Commit f2151df5 authored by Jingbo Xu's avatar Jingbo Xu Committed by Gao Xiang

erofs: make iov_iter describe target buffers over fscache

So far the fscache mode supports uncompressed data only, and the data
read from fscache is put directly into the target page cache.  As the
support for compressed data in fscache mode is going to be introduced,
rework the fscache internals so that the following compressed part
could make the raw data read from fscache be directed to the target
buffer it wants, decompress the raw data, and finally fill the page
cache with the decompressed data.

As the first step, a new structure, i.e. erofs_fscache_io (io), is
introduced to describe a generic read request from the fscache, while
the caller can specify the target buffer it wants in the iov_iter
structure (io->iter).  Besides, the caller can also specify its
completion callback and private data through erofs_fscache_io, which
will be called to make further handling, e.g. unlocking the page cache
for uncompressed data or decompressing the read raw data, when the read
request from the fscache completes.  Now erofs_fscache_read_io_async()
serves as a generic interface for reading raw data from fscache for both
compressed and uncompressed data.

The erofs_fscache_rq structure is kept to describe a request to fill the
page cache in the specified range.
Signed-off-by: default avatarJingbo Xu <jefflexu@linux.alibaba.com>
Reviewed-by: default avatarGao Xiang <hsiangkao@linux.alibaba.com>
Link: https://lore.kernel.org/r/20240308094159.40547-1-jefflexu@linux.alibaba.comSigned-off-by: default avatarGao Xiang <hsiangkao@linux.alibaba.com>
parent 0f28be64
...@@ -25,9 +25,15 @@ static struct file_system_type erofs_anon_fs_type = { ...@@ -25,9 +25,15 @@ static struct file_system_type erofs_anon_fs_type = {
.kill_sb = kill_anon_super, .kill_sb = kill_anon_super,
}; };
struct erofs_fscache_request { struct erofs_fscache_io {
struct erofs_fscache_request *primary; struct netfs_cache_resources cres;
struct netfs_cache_resources cache_resources; struct iov_iter iter;
netfs_io_terminated_t end_io;
void *private;
refcount_t ref;
};
struct erofs_fscache_rq {
struct address_space *mapping; /* The mapping being accessed */ struct address_space *mapping; /* The mapping being accessed */
loff_t start; /* Start position */ loff_t start; /* Start position */
size_t len; /* Length of the request */ size_t len; /* Length of the request */
...@@ -36,44 +42,17 @@ struct erofs_fscache_request { ...@@ -36,44 +42,17 @@ struct erofs_fscache_request {
refcount_t ref; refcount_t ref;
}; };
static struct erofs_fscache_request *erofs_fscache_req_alloc(struct address_space *mapping, static bool erofs_fscache_io_put(struct erofs_fscache_io *io)
loff_t start, size_t len)
{
struct erofs_fscache_request *req;
req = kzalloc(sizeof(struct erofs_fscache_request), GFP_KERNEL);
if (!req)
return ERR_PTR(-ENOMEM);
req->mapping = mapping;
req->start = start;
req->len = len;
refcount_set(&req->ref, 1);
return req;
}
static struct erofs_fscache_request *erofs_fscache_req_chain(struct erofs_fscache_request *primary,
size_t len)
{ {
struct erofs_fscache_request *req; if (!refcount_dec_and_test(&io->ref))
return false;
/* use primary request for the first submission */ if (io->cres.ops)
if (!primary->submitted) { io->cres.ops->end_operation(&io->cres);
refcount_inc(&primary->ref); kfree(io);
return primary; return true;
}
req = erofs_fscache_req_alloc(primary->mapping,
primary->start + primary->submitted, len);
if (!IS_ERR(req)) {
req->primary = primary;
refcount_inc(&primary->ref);
}
return req;
} }
static void erofs_fscache_req_complete(struct erofs_fscache_request *req) static void erofs_fscache_req_complete(struct erofs_fscache_rq *req)
{ {
struct folio *folio; struct folio *folio;
bool failed = req->error; bool failed = req->error;
...@@ -93,120 +72,149 @@ static void erofs_fscache_req_complete(struct erofs_fscache_request *req) ...@@ -93,120 +72,149 @@ static void erofs_fscache_req_complete(struct erofs_fscache_request *req)
rcu_read_unlock(); rcu_read_unlock();
} }
static void erofs_fscache_req_put(struct erofs_fscache_request *req) static void erofs_fscache_req_put(struct erofs_fscache_rq *req)
{ {
if (refcount_dec_and_test(&req->ref)) { if (!refcount_dec_and_test(&req->ref))
if (req->cache_resources.ops) return;
req->cache_resources.ops->end_operation(&req->cache_resources);
if (!req->primary)
erofs_fscache_req_complete(req); erofs_fscache_req_complete(req);
else
erofs_fscache_req_put(req->primary);
kfree(req); kfree(req);
}
} }
static void erofs_fscache_subreq_complete(void *priv, static struct erofs_fscache_rq *erofs_fscache_req_alloc(struct address_space *mapping,
loff_t start, size_t len)
{
struct erofs_fscache_rq *req = kzalloc(sizeof(*req), GFP_KERNEL);
if (!req)
return NULL;
req->mapping = mapping;
req->start = start;
req->len = len;
refcount_set(&req->ref, 1);
return req;
}
static void erofs_fscache_req_io_put(struct erofs_fscache_io *io)
{
struct erofs_fscache_rq *req = io->private;
if (erofs_fscache_io_put(io))
erofs_fscache_req_put(req);
}
static void erofs_fscache_req_end_io(void *priv,
ssize_t transferred_or_error, bool was_async) ssize_t transferred_or_error, bool was_async)
{ {
struct erofs_fscache_request *req = priv; struct erofs_fscache_io *io = priv;
struct erofs_fscache_rq *req = io->private;
if (IS_ERR_VALUE(transferred_or_error)) { if (IS_ERR_VALUE(transferred_or_error))
if (req->primary)
req->primary->error = transferred_or_error;
else
req->error = transferred_or_error; req->error = transferred_or_error;
} erofs_fscache_req_io_put(io);
erofs_fscache_req_put(req); }
static struct erofs_fscache_io *erofs_fscache_req_io_alloc(struct erofs_fscache_rq *req)
{
struct erofs_fscache_io *io = kzalloc(sizeof(*io), GFP_KERNEL);
if (!io)
return NULL;
io->end_io = erofs_fscache_req_end_io;
io->private = req;
refcount_inc(&req->ref);
refcount_set(&io->ref, 1);
return io;
} }
/* /*
* Read data from fscache (cookie, pstart, len), and fill the read data into * Read data from fscache described by cookie at pstart physical address
* page cache described by (req->mapping, lstart, len). @pstart describeis the * offset, and fill the read data into buffer described by io->iter.
* start physical address in the cache file.
*/ */
static int erofs_fscache_read_folios_async(struct fscache_cookie *cookie, static int erofs_fscache_read_io_async(struct fscache_cookie *cookie,
struct erofs_fscache_request *req, loff_t pstart, size_t len) loff_t pstart, struct erofs_fscache_io *io)
{ {
enum netfs_io_source source; enum netfs_io_source source;
struct super_block *sb = req->mapping->host->i_sb; struct netfs_cache_resources *cres = &io->cres;
struct netfs_cache_resources *cres = &req->cache_resources; struct iov_iter *iter = &io->iter;
struct iov_iter iter;
loff_t lstart = req->start + req->submitted;
size_t done = 0;
int ret; int ret;
DBG_BUGON(len > req->len - req->submitted);
ret = fscache_begin_read_operation(cres, cookie); ret = fscache_begin_read_operation(cres, cookie);
if (ret) if (ret)
return ret; return ret;
while (done < len) { while (iov_iter_count(iter)) {
loff_t sstart = pstart + done; size_t orig_count = iov_iter_count(iter), len = orig_count;
size_t slen = len - done;
unsigned long flags = 1 << NETFS_SREQ_ONDEMAND; unsigned long flags = 1 << NETFS_SREQ_ONDEMAND;
source = cres->ops->prepare_ondemand_read(cres, source = cres->ops->prepare_ondemand_read(cres,
sstart, &slen, LLONG_MAX, &flags, 0); pstart, &len, LLONG_MAX, &flags, 0);
if (WARN_ON(slen == 0)) if (WARN_ON(len == 0))
source = NETFS_INVALID_READ; source = NETFS_INVALID_READ;
if (source != NETFS_READ_FROM_CACHE) { if (source != NETFS_READ_FROM_CACHE) {
erofs_err(sb, "failed to fscache prepare_read (source %d)", source); erofs_err(NULL, "prepare_read failed (source %d)", source);
return -EIO; return -EIO;
} }
refcount_inc(&req->ref); iov_iter_truncate(iter, len);
iov_iter_xarray(&iter, ITER_DEST, &req->mapping->i_pages, refcount_inc(&io->ref);
lstart + done, slen); ret = fscache_read(cres, pstart, iter, NETFS_READ_HOLE_FAIL,
io->end_io, io);
ret = fscache_read(cres, sstart, &iter, NETFS_READ_HOLE_FAIL,
erofs_fscache_subreq_complete, req);
if (ret == -EIOCBQUEUED) if (ret == -EIOCBQUEUED)
ret = 0; ret = 0;
if (ret) { if (ret) {
erofs_err(sb, "failed to fscache_read (ret %d)", ret); erofs_err(NULL, "fscache_read failed (ret %d)", ret);
return ret; return ret;
} }
if (WARN_ON(iov_iter_count(iter)))
return -EIO;
done += slen; iov_iter_reexpand(iter, orig_count - len);
pstart += len;
} }
DBG_BUGON(done != len);
return 0; return 0;
} }
static int erofs_fscache_meta_read_folio(struct file *data, struct folio *folio) static int erofs_fscache_meta_read_folio(struct file *data, struct folio *folio)
{ {
int ret;
struct erofs_fscache *ctx = folio->mapping->host->i_private; struct erofs_fscache *ctx = folio->mapping->host->i_private;
struct erofs_fscache_request *req; int ret = -ENOMEM;
struct erofs_fscache_rq *req;
struct erofs_fscache_io *io;
req = erofs_fscache_req_alloc(folio->mapping, req = erofs_fscache_req_alloc(folio->mapping,
folio_pos(folio), folio_size(folio)); folio_pos(folio), folio_size(folio));
if (IS_ERR(req)) { if (!req) {
folio_unlock(folio); folio_unlock(folio);
return PTR_ERR(req); return ret;
} }
ret = erofs_fscache_read_folios_async(ctx->cookie, req, io = erofs_fscache_req_io_alloc(req);
if (!io) {
req->error = ret;
goto out;
}
iov_iter_xarray(&io->iter, ITER_DEST, &folio->mapping->i_pages,
folio_pos(folio), folio_size(folio)); folio_pos(folio), folio_size(folio));
ret = erofs_fscache_read_io_async(ctx->cookie, folio_pos(folio), io);
if (ret) if (ret)
req->error = ret; req->error = ret;
erofs_fscache_req_io_put(io);
out:
erofs_fscache_req_put(req); erofs_fscache_req_put(req);
return ret; return ret;
} }
static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary) static int erofs_fscache_data_read_slice(struct erofs_fscache_rq *req)
{ {
struct address_space *mapping = primary->mapping; struct address_space *mapping = req->mapping;
struct inode *inode = mapping->host; struct inode *inode = mapping->host;
struct super_block *sb = inode->i_sb; struct super_block *sb = inode->i_sb;
struct erofs_fscache_request *req; struct erofs_fscache_io *io;
struct erofs_map_blocks map; struct erofs_map_blocks map;
struct erofs_map_dev mdev; struct erofs_map_dev mdev;
struct iov_iter iter; loff_t pos = req->start + req->submitted;
loff_t pos = primary->start + primary->submitted;
size_t count; size_t count;
int ret; int ret;
...@@ -217,6 +225,7 @@ static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary) ...@@ -217,6 +225,7 @@ static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary)
if (map.m_flags & EROFS_MAP_META) { if (map.m_flags & EROFS_MAP_META) {
struct erofs_buf buf = __EROFS_BUF_INITIALIZER; struct erofs_buf buf = __EROFS_BUF_INITIALIZER;
struct iov_iter iter;
erofs_blk_t blknr; erofs_blk_t blknr;
size_t offset, size; size_t offset, size;
void *src; void *src;
...@@ -237,15 +246,17 @@ static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary) ...@@ -237,15 +246,17 @@ static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary)
} }
iov_iter_zero(PAGE_SIZE - size, &iter); iov_iter_zero(PAGE_SIZE - size, &iter);
erofs_put_metabuf(&buf); erofs_put_metabuf(&buf);
primary->submitted += PAGE_SIZE; req->submitted += PAGE_SIZE;
return 0; return 0;
} }
count = primary->len - primary->submitted; count = req->len - req->submitted;
if (!(map.m_flags & EROFS_MAP_MAPPED)) { if (!(map.m_flags & EROFS_MAP_MAPPED)) {
struct iov_iter iter;
iov_iter_xarray(&iter, ITER_DEST, &mapping->i_pages, pos, count); iov_iter_xarray(&iter, ITER_DEST, &mapping->i_pages, pos, count);
iov_iter_zero(count, &iter); iov_iter_zero(count, &iter);
primary->submitted += count; req->submitted += count;
return 0; return 0;
} }
...@@ -260,18 +271,19 @@ static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary) ...@@ -260,18 +271,19 @@ static int erofs_fscache_data_read_slice(struct erofs_fscache_request *primary)
if (ret) if (ret)
return ret; return ret;
req = erofs_fscache_req_chain(primary, count); io = erofs_fscache_req_io_alloc(req);
if (IS_ERR(req)) if (!io)
return PTR_ERR(req); return -ENOMEM;
iov_iter_xarray(&io->iter, ITER_DEST, &mapping->i_pages, pos, count);
ret = erofs_fscache_read_io_async(mdev.m_fscache->cookie,
mdev.m_pa + (pos - map.m_la), io);
erofs_fscache_req_io_put(io);
ret = erofs_fscache_read_folios_async(mdev.m_fscache->cookie, req->submitted += count;
req, mdev.m_pa + (pos - map.m_la), count);
erofs_fscache_req_put(req);
primary->submitted += count;
return ret; return ret;
} }
static int erofs_fscache_data_read(struct erofs_fscache_request *req) static int erofs_fscache_data_read(struct erofs_fscache_rq *req)
{ {
int ret; int ret;
...@@ -280,20 +292,19 @@ static int erofs_fscache_data_read(struct erofs_fscache_request *req) ...@@ -280,20 +292,19 @@ static int erofs_fscache_data_read(struct erofs_fscache_request *req)
if (ret) if (ret)
req->error = ret; req->error = ret;
} while (!ret && req->submitted < req->len); } while (!ret && req->submitted < req->len);
return ret; return ret;
} }
static int erofs_fscache_read_folio(struct file *file, struct folio *folio) static int erofs_fscache_read_folio(struct file *file, struct folio *folio)
{ {
struct erofs_fscache_request *req; struct erofs_fscache_rq *req;
int ret; int ret;
req = erofs_fscache_req_alloc(folio->mapping, req = erofs_fscache_req_alloc(folio->mapping,
folio_pos(folio), folio_size(folio)); folio_pos(folio), folio_size(folio));
if (IS_ERR(req)) { if (!req) {
folio_unlock(folio); folio_unlock(folio);
return PTR_ERR(req); return -ENOMEM;
} }
ret = erofs_fscache_data_read(req); ret = erofs_fscache_data_read(req);
...@@ -303,14 +314,14 @@ static int erofs_fscache_read_folio(struct file *file, struct folio *folio) ...@@ -303,14 +314,14 @@ static int erofs_fscache_read_folio(struct file *file, struct folio *folio)
static void erofs_fscache_readahead(struct readahead_control *rac) static void erofs_fscache_readahead(struct readahead_control *rac)
{ {
struct erofs_fscache_request *req; struct erofs_fscache_rq *req;
if (!readahead_count(rac)) if (!readahead_count(rac))
return; return;
req = erofs_fscache_req_alloc(rac->mapping, req = erofs_fscache_req_alloc(rac->mapping,
readahead_pos(rac), readahead_length(rac)); readahead_pos(rac), readahead_length(rac));
if (IS_ERR(req)) if (!req)
return; return;
/* The request completion will drop refs on the folios. */ /* The request completion will drop refs on the folios. */
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment