mirror of
https://github.com/AuxXxilium/linux_dsm_epyc7002.git
synced 2025-02-22 00:00:03 +07:00
NFS: merge _full and _partial read rpc_ops
Decouple nfs_pgio_header and nfs_read_data, and have (possibly multiple) nfs_read_datas each take a refcount on nfs_pgio_header. For the moment keeps nfs_read_header as a way to preallocate a single nfs_read_data with the nfs_pgio_header. The code doesn't need this, and would be prettier without, but given the amount of churn I am already introducing I didn't want to play with tuning new mempools. This also fixes bug in pnfs_ld_handle_read_error. In the case of desc->pg_bsize < PAGE_CACHE_SIZE, the pages list was empty, causing replay attempt to do nothing. Signed-off-by: Fred Isaman <iisaman@netapp.com> Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
This commit is contained in:
parent
30dd374f6f
commit
4db6e0b74c
@ -319,10 +319,16 @@ static ssize_t nfs_direct_read_schedule_segment(struct nfs_direct_req *dreq,
|
|||||||
bytes = min(rsize,count);
|
bytes = min(rsize,count);
|
||||||
|
|
||||||
result = -ENOMEM;
|
result = -ENOMEM;
|
||||||
rhdr = nfs_readhdr_alloc(nfs_page_array_len(pgbase, bytes));
|
rhdr = nfs_readhdr_alloc();
|
||||||
if (unlikely(!rhdr))
|
if (unlikely(!rhdr))
|
||||||
break;
|
break;
|
||||||
data = &rhdr->rpc_data;
|
data = nfs_readdata_alloc(&rhdr->header, nfs_page_array_len(pgbase, bytes));
|
||||||
|
if (!data) {
|
||||||
|
nfs_readhdr_free(&rhdr->header);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
data->header = &rhdr->header;
|
||||||
|
atomic_inc(&data->header->refcnt);
|
||||||
pages = &data->pages;
|
pages = &data->pages;
|
||||||
|
|
||||||
down_read(¤t->mm->mmap_sem);
|
down_read(¤t->mm->mmap_sem);
|
||||||
|
@ -200,6 +200,7 @@ struct vfsmount *nfs_do_refmount(struct rpc_clnt *client, struct dentry *dentry)
|
|||||||
extern struct svc_version nfs4_callback_version1;
|
extern struct svc_version nfs4_callback_version1;
|
||||||
extern struct svc_version nfs4_callback_version4;
|
extern struct svc_version nfs4_callback_version4;
|
||||||
|
|
||||||
|
struct nfs_pageio_descriptor;
|
||||||
/* pagelist.c */
|
/* pagelist.c */
|
||||||
extern int __init nfs_init_nfspagecache(void);
|
extern int __init nfs_init_nfspagecache(void);
|
||||||
extern void nfs_destroy_nfspagecache(void);
|
extern void nfs_destroy_nfspagecache(void);
|
||||||
@ -211,6 +212,10 @@ extern void nfs_destroy_writepagecache(void);
|
|||||||
extern int __init nfs_init_directcache(void);
|
extern int __init nfs_init_directcache(void);
|
||||||
extern void nfs_destroy_directcache(void);
|
extern void nfs_destroy_directcache(void);
|
||||||
extern bool nfs_pgarray_set(struct nfs_page_array *p, unsigned int pagecount);
|
extern bool nfs_pgarray_set(struct nfs_page_array *p, unsigned int pagecount);
|
||||||
|
extern void nfs_pgheader_init(struct nfs_pageio_descriptor *desc,
|
||||||
|
struct nfs_pgio_header *hdr,
|
||||||
|
void (*release)(struct nfs_pgio_header *hdr));
|
||||||
|
void nfs_set_pgio_error(struct nfs_pgio_header *hdr, int error, loff_t pos);
|
||||||
|
|
||||||
/* nfs2xdr.c */
|
/* nfs2xdr.c */
|
||||||
extern int nfs_stat_to_errno(enum nfs_stat);
|
extern int nfs_stat_to_errno(enum nfs_stat);
|
||||||
@ -295,17 +300,19 @@ extern struct dentry *nfs4_get_root(struct super_block *, struct nfs_fh *,
|
|||||||
extern int nfs4_get_rootfh(struct nfs_server *server, struct nfs_fh *mntfh);
|
extern int nfs4_get_rootfh(struct nfs_server *server, struct nfs_fh *mntfh);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct nfs_pageio_descriptor;
|
|
||||||
/* read.c */
|
/* read.c */
|
||||||
extern struct nfs_read_header *nfs_readhdr_alloc(unsigned int npages);
|
extern void nfs_async_read_error(struct list_head *head);
|
||||||
|
extern struct nfs_read_header *nfs_readhdr_alloc(void);
|
||||||
extern void nfs_readhdr_free(struct nfs_pgio_header *hdr);
|
extern void nfs_readhdr_free(struct nfs_pgio_header *hdr);
|
||||||
|
extern void nfs_read_completion(struct nfs_pgio_header *hdr);
|
||||||
|
extern struct nfs_read_data *nfs_readdata_alloc(struct nfs_pgio_header *hdr,
|
||||||
|
unsigned int pagecount);
|
||||||
extern int nfs_initiate_read(struct rpc_clnt *clnt,
|
extern int nfs_initiate_read(struct rpc_clnt *clnt,
|
||||||
struct nfs_read_data *data,
|
struct nfs_read_data *data,
|
||||||
const struct rpc_call_ops *call_ops);
|
const struct rpc_call_ops *call_ops);
|
||||||
extern void nfs_read_prepare(struct rpc_task *task, void *calldata);
|
extern void nfs_read_prepare(struct rpc_task *task, void *calldata);
|
||||||
extern int nfs_generic_pagein(struct nfs_pageio_descriptor *desc,
|
extern int nfs_generic_pagein(struct nfs_pageio_descriptor *desc,
|
||||||
struct list_head *head);
|
struct nfs_pgio_header *hdr);
|
||||||
|
|
||||||
extern void nfs_pageio_init_read_mds(struct nfs_pageio_descriptor *pgio,
|
extern void nfs_pageio_init_read_mds(struct nfs_pageio_descriptor *pgio,
|
||||||
struct inode *inode);
|
struct inode *inode);
|
||||||
extern void nfs_pageio_reset_read_mds(struct nfs_pageio_descriptor *pgio);
|
extern void nfs_pageio_reset_read_mds(struct nfs_pageio_descriptor *pgio);
|
||||||
|
@ -227,7 +227,6 @@ static void filelayout_read_release(void *data)
|
|||||||
{
|
{
|
||||||
struct nfs_read_data *rdata = data;
|
struct nfs_read_data *rdata = data;
|
||||||
|
|
||||||
put_lseg(rdata->header->lseg);
|
|
||||||
rdata->header->mds_ops->rpc_release(data);
|
rdata->header->mds_ops->rpc_release(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3391,8 +3391,6 @@ void nfs4_reset_read(struct rpc_task *task, struct nfs_read_data *data)
|
|||||||
struct inode *inode = hdr->inode;
|
struct inode *inode = hdr->inode;
|
||||||
|
|
||||||
dprintk("%s Reset task for i/o through\n", __func__);
|
dprintk("%s Reset task for i/o through\n", __func__);
|
||||||
put_lseg(hdr->lseg);
|
|
||||||
hdr->lseg = NULL;
|
|
||||||
data->ds_clp = NULL;
|
data->ds_clp = NULL;
|
||||||
/* offsets will differ in the dense stripe case */
|
/* offsets will differ in the dense stripe case */
|
||||||
data->args.offset = data->mds_offset;
|
data->args.offset = data->mds_offset;
|
||||||
|
@ -39,6 +39,30 @@ bool nfs_pgarray_set(struct nfs_page_array *p, unsigned int pagecount)
|
|||||||
return p->pagevec != NULL;
|
return p->pagevec != NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void nfs_pgheader_init(struct nfs_pageio_descriptor *desc,
|
||||||
|
struct nfs_pgio_header *hdr,
|
||||||
|
void (*release)(struct nfs_pgio_header *hdr))
|
||||||
|
{
|
||||||
|
hdr->req = nfs_list_entry(desc->pg_list.next);
|
||||||
|
hdr->inode = desc->pg_inode;
|
||||||
|
hdr->cred = hdr->req->wb_context->cred;
|
||||||
|
hdr->io_start = req_offset(hdr->req);
|
||||||
|
hdr->good_bytes = desc->pg_count;
|
||||||
|
hdr->release = release;
|
||||||
|
}
|
||||||
|
|
||||||
|
void nfs_set_pgio_error(struct nfs_pgio_header *hdr, int error, loff_t pos)
|
||||||
|
{
|
||||||
|
spin_lock(&hdr->lock);
|
||||||
|
if (pos < hdr->io_start + hdr->good_bytes) {
|
||||||
|
set_bit(NFS_IOHDR_ERROR, &hdr->flags);
|
||||||
|
clear_bit(NFS_IOHDR_EOF, &hdr->flags);
|
||||||
|
hdr->good_bytes = pos - hdr->io_start;
|
||||||
|
hdr->error = error;
|
||||||
|
}
|
||||||
|
spin_unlock(&hdr->lock);
|
||||||
|
}
|
||||||
|
|
||||||
static inline struct nfs_page *
|
static inline struct nfs_page *
|
||||||
nfs_page_alloc(void)
|
nfs_page_alloc(void)
|
||||||
{
|
{
|
||||||
|
@ -1333,7 +1333,9 @@ static void pnfs_ld_handle_read_error(struct nfs_read_data *data)
|
|||||||
clear_bit(NFS_INO_LAYOUTCOMMIT, &NFS_I(hdr->inode)->flags);
|
clear_bit(NFS_INO_LAYOUTCOMMIT, &NFS_I(hdr->inode)->flags);
|
||||||
pnfs_return_layout(hdr->inode);
|
pnfs_return_layout(hdr->inode);
|
||||||
}
|
}
|
||||||
data->task.tk_status = pnfs_read_done_resend_to_mds(hdr->inode, &hdr->pages);
|
if (!test_and_set_bit(NFS_IOHDR_REDO, &hdr->flags))
|
||||||
|
data->task.tk_status = pnfs_read_done_resend_to_mds(hdr->inode,
|
||||||
|
&hdr->pages);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -1348,7 +1350,6 @@ void pnfs_ld_read_done(struct nfs_read_data *data)
|
|||||||
hdr->mds_ops->rpc_call_done(&data->task, data);
|
hdr->mds_ops->rpc_call_done(&data->task, data);
|
||||||
} else
|
} else
|
||||||
pnfs_ld_handle_read_error(data);
|
pnfs_ld_handle_read_error(data);
|
||||||
put_lseg(hdr->lseg);
|
|
||||||
hdr->mds_ops->rpc_release(data);
|
hdr->mds_ops->rpc_release(data);
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(pnfs_ld_read_done);
|
EXPORT_SYMBOL_GPL(pnfs_ld_read_done);
|
||||||
@ -1359,11 +1360,11 @@ pnfs_read_through_mds(struct nfs_pageio_descriptor *desc,
|
|||||||
{
|
{
|
||||||
struct nfs_pgio_header *hdr = data->header;
|
struct nfs_pgio_header *hdr = data->header;
|
||||||
|
|
||||||
list_splice_tail_init(&hdr->pages, &desc->pg_list);
|
if (!test_and_set_bit(NFS_IOHDR_REDO, &hdr->flags)) {
|
||||||
if (hdr->req && list_empty(&hdr->req->wb_list))
|
list_splice_tail_init(&hdr->pages, &desc->pg_list);
|
||||||
nfs_list_add_request(hdr->req, &desc->pg_list);
|
nfs_pageio_reset_read_mds(desc);
|
||||||
nfs_pageio_reset_read_mds(desc);
|
desc->pg_recoalesce = 1;
|
||||||
desc->pg_recoalesce = 1;
|
}
|
||||||
nfs_readdata_release(data);
|
nfs_readdata_release(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1381,18 +1382,13 @@ pnfs_try_to_read_data(struct nfs_read_data *rdata,
|
|||||||
enum pnfs_try_status trypnfs;
|
enum pnfs_try_status trypnfs;
|
||||||
|
|
||||||
hdr->mds_ops = call_ops;
|
hdr->mds_ops = call_ops;
|
||||||
hdr->lseg = get_lseg(lseg);
|
|
||||||
|
|
||||||
dprintk("%s: Reading ino:%lu %u@%llu\n",
|
dprintk("%s: Reading ino:%lu %u@%llu\n",
|
||||||
__func__, inode->i_ino, rdata->args.count, rdata->args.offset);
|
__func__, inode->i_ino, rdata->args.count, rdata->args.offset);
|
||||||
|
|
||||||
trypnfs = nfss->pnfs_curr_ld->read_pagelist(rdata);
|
trypnfs = nfss->pnfs_curr_ld->read_pagelist(rdata);
|
||||||
if (trypnfs == PNFS_NOT_ATTEMPTED) {
|
if (trypnfs != PNFS_NOT_ATTEMPTED)
|
||||||
put_lseg(hdr->lseg);
|
|
||||||
hdr->lseg = NULL;
|
|
||||||
} else {
|
|
||||||
nfs_inc_stats(inode, NFSIOS_PNFS_READ);
|
nfs_inc_stats(inode, NFSIOS_PNFS_READ);
|
||||||
}
|
|
||||||
dprintk("%s End (trypnfs:%d)\n", __func__, trypnfs);
|
dprintk("%s End (trypnfs:%d)\n", __func__, trypnfs);
|
||||||
return trypnfs;
|
return trypnfs;
|
||||||
}
|
}
|
||||||
@ -1408,7 +1404,7 @@ pnfs_do_multiple_reads(struct nfs_pageio_descriptor *desc, struct list_head *hea
|
|||||||
while (!list_empty(head)) {
|
while (!list_empty(head)) {
|
||||||
enum pnfs_try_status trypnfs;
|
enum pnfs_try_status trypnfs;
|
||||||
|
|
||||||
data = list_entry(head->next, struct nfs_read_data, list);
|
data = list_first_entry(head, struct nfs_read_data, list);
|
||||||
list_del_init(&data->list);
|
list_del_init(&data->list);
|
||||||
|
|
||||||
trypnfs = pnfs_try_to_read_data(data, call_ops, lseg);
|
trypnfs = pnfs_try_to_read_data(data, call_ops, lseg);
|
||||||
@ -1418,20 +1414,41 @@ pnfs_do_multiple_reads(struct nfs_pageio_descriptor *desc, struct list_head *hea
|
|||||||
put_lseg(lseg);
|
put_lseg(lseg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void pnfs_readhdr_free(struct nfs_pgio_header *hdr)
|
||||||
|
{
|
||||||
|
put_lseg(hdr->lseg);
|
||||||
|
nfs_readhdr_free(hdr);
|
||||||
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
pnfs_generic_pg_readpages(struct nfs_pageio_descriptor *desc)
|
pnfs_generic_pg_readpages(struct nfs_pageio_descriptor *desc)
|
||||||
{
|
{
|
||||||
LIST_HEAD(head);
|
struct nfs_read_header *rhdr;
|
||||||
|
struct nfs_pgio_header *hdr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = nfs_generic_pagein(desc, &head);
|
rhdr = nfs_readhdr_alloc();
|
||||||
if (ret != 0) {
|
if (!rhdr) {
|
||||||
|
nfs_async_read_error(&desc->pg_list);
|
||||||
|
ret = -ENOMEM;
|
||||||
put_lseg(desc->pg_lseg);
|
put_lseg(desc->pg_lseg);
|
||||||
desc->pg_lseg = NULL;
|
desc->pg_lseg = NULL;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
pnfs_do_multiple_reads(desc, &head);
|
hdr = &rhdr->header;
|
||||||
return 0;
|
nfs_pgheader_init(desc, hdr, pnfs_readhdr_free);
|
||||||
|
hdr->lseg = get_lseg(desc->pg_lseg);
|
||||||
|
atomic_inc(&hdr->refcnt);
|
||||||
|
ret = nfs_generic_pagein(desc, hdr);
|
||||||
|
if (ret != 0) {
|
||||||
|
put_lseg(desc->pg_lseg);
|
||||||
|
desc->pg_lseg = NULL;
|
||||||
|
set_bit(NFS_IOHDR_REDO, &hdr->flags);
|
||||||
|
} else
|
||||||
|
pnfs_do_multiple_reads(desc, &hdr->rpc_list);
|
||||||
|
if (atomic_dec_and_test(&hdr->refcnt))
|
||||||
|
nfs_read_completion(hdr);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(pnfs_generic_pg_readpages);
|
EXPORT_SYMBOL_GPL(pnfs_generic_pg_readpages);
|
||||||
|
|
||||||
|
338
fs/nfs/read.c
338
fs/nfs/read.c
@ -30,29 +30,49 @@
|
|||||||
#define NFSDBG_FACILITY NFSDBG_PAGECACHE
|
#define NFSDBG_FACILITY NFSDBG_PAGECACHE
|
||||||
|
|
||||||
static const struct nfs_pageio_ops nfs_pageio_read_ops;
|
static const struct nfs_pageio_ops nfs_pageio_read_ops;
|
||||||
static const struct rpc_call_ops nfs_read_partial_ops;
|
static const struct rpc_call_ops nfs_read_common_ops;
|
||||||
static const struct rpc_call_ops nfs_read_full_ops;
|
|
||||||
|
|
||||||
static struct kmem_cache *nfs_rdata_cachep;
|
static struct kmem_cache *nfs_rdata_cachep;
|
||||||
|
|
||||||
struct nfs_read_header *nfs_readhdr_alloc(unsigned int pagecount)
|
struct nfs_read_header *nfs_readhdr_alloc()
|
||||||
{
|
{
|
||||||
struct nfs_read_header *p;
|
struct nfs_read_header *rhdr;
|
||||||
|
|
||||||
p = kmem_cache_zalloc(nfs_rdata_cachep, GFP_KERNEL);
|
rhdr = kmem_cache_zalloc(nfs_rdata_cachep, GFP_KERNEL);
|
||||||
if (p) {
|
if (rhdr) {
|
||||||
struct nfs_pgio_header *hdr = &p->header;
|
struct nfs_pgio_header *hdr = &rhdr->header;
|
||||||
struct nfs_read_data *data = &p->rpc_data;
|
|
||||||
|
|
||||||
INIT_LIST_HEAD(&hdr->pages);
|
INIT_LIST_HEAD(&hdr->pages);
|
||||||
INIT_LIST_HEAD(&data->list);
|
INIT_LIST_HEAD(&hdr->rpc_list);
|
||||||
data->header = hdr;
|
spin_lock_init(&hdr->lock);
|
||||||
if (!nfs_pgarray_set(&data->pages, pagecount)) {
|
atomic_set(&hdr->refcnt, 0);
|
||||||
kmem_cache_free(nfs_rdata_cachep, p);
|
|
||||||
p = NULL;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return p;
|
return rhdr;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct nfs_read_data *nfs_readdata_alloc(struct nfs_pgio_header *hdr,
|
||||||
|
unsigned int pagecount)
|
||||||
|
{
|
||||||
|
struct nfs_read_data *data, *prealloc;
|
||||||
|
|
||||||
|
prealloc = &container_of(hdr, struct nfs_read_header, header)->rpc_data;
|
||||||
|
if (prealloc->header == NULL)
|
||||||
|
data = prealloc;
|
||||||
|
else
|
||||||
|
data = kzalloc(sizeof(*data), GFP_KERNEL);
|
||||||
|
if (!data)
|
||||||
|
goto out;
|
||||||
|
|
||||||
|
if (nfs_pgarray_set(&data->pages, pagecount)) {
|
||||||
|
data->header = hdr;
|
||||||
|
atomic_inc(&hdr->refcnt);
|
||||||
|
} else {
|
||||||
|
if (data != prealloc)
|
||||||
|
kfree(data);
|
||||||
|
data = NULL;
|
||||||
|
}
|
||||||
|
out:
|
||||||
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
void nfs_readhdr_free(struct nfs_pgio_header *hdr)
|
void nfs_readhdr_free(struct nfs_pgio_header *hdr)
|
||||||
@ -64,10 +84,18 @@ void nfs_readhdr_free(struct nfs_pgio_header *hdr)
|
|||||||
|
|
||||||
void nfs_readdata_release(struct nfs_read_data *rdata)
|
void nfs_readdata_release(struct nfs_read_data *rdata)
|
||||||
{
|
{
|
||||||
|
struct nfs_pgio_header *hdr = rdata->header;
|
||||||
|
struct nfs_read_header *read_header = container_of(hdr, struct nfs_read_header, header);
|
||||||
|
|
||||||
put_nfs_open_context(rdata->args.context);
|
put_nfs_open_context(rdata->args.context);
|
||||||
if (rdata->pages.pagevec != rdata->pages.page_array)
|
if (rdata->pages.pagevec != rdata->pages.page_array)
|
||||||
kfree(rdata->pages.pagevec);
|
kfree(rdata->pages.pagevec);
|
||||||
nfs_readhdr_free(rdata->header);
|
if (rdata != &read_header->rpc_data)
|
||||||
|
kfree(rdata);
|
||||||
|
else
|
||||||
|
rdata->header = NULL;
|
||||||
|
if (atomic_dec_and_test(&hdr->refcnt))
|
||||||
|
nfs_read_completion(hdr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static
|
static
|
||||||
@ -79,35 +107,6 @@ int nfs_return_empty_page(struct page *page)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void nfs_readpage_truncate_uninitialised_page(struct nfs_read_data *data)
|
|
||||||
{
|
|
||||||
unsigned int remainder = data->args.count - data->res.count;
|
|
||||||
unsigned int base = data->args.pgbase + data->res.count;
|
|
||||||
unsigned int pglen;
|
|
||||||
struct page **pages;
|
|
||||||
|
|
||||||
if (data->res.eof == 0 || remainder == 0)
|
|
||||||
return;
|
|
||||||
/*
|
|
||||||
* Note: "remainder" can never be negative, since we check for
|
|
||||||
* this in the XDR code.
|
|
||||||
*/
|
|
||||||
pages = &data->args.pages[base >> PAGE_CACHE_SHIFT];
|
|
||||||
base &= ~PAGE_CACHE_MASK;
|
|
||||||
pglen = PAGE_CACHE_SIZE - base;
|
|
||||||
for (;;) {
|
|
||||||
if (remainder <= pglen) {
|
|
||||||
zero_user(*pages, base, remainder);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
zero_user(*pages, base, pglen);
|
|
||||||
pages++;
|
|
||||||
remainder -= pglen;
|
|
||||||
pglen = PAGE_CACHE_SIZE;
|
|
||||||
base = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void nfs_pageio_init_read_mds(struct nfs_pageio_descriptor *pgio,
|
void nfs_pageio_init_read_mds(struct nfs_pageio_descriptor *pgio,
|
||||||
struct inode *inode)
|
struct inode *inode)
|
||||||
{
|
{
|
||||||
@ -170,6 +169,46 @@ static void nfs_readpage_release(struct nfs_page *req)
|
|||||||
nfs_release_request(req);
|
nfs_release_request(req);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Note io was page aligned */
|
||||||
|
void nfs_read_completion(struct nfs_pgio_header *hdr)
|
||||||
|
{
|
||||||
|
unsigned long bytes = 0;
|
||||||
|
|
||||||
|
if (test_bit(NFS_IOHDR_REDO, &hdr->flags))
|
||||||
|
goto out;
|
||||||
|
if (!test_bit(NFS_IOHDR_ERROR, &hdr->flags)) {
|
||||||
|
while (!list_empty(&hdr->pages)) {
|
||||||
|
struct nfs_page *req = nfs_list_entry(hdr->pages.next);
|
||||||
|
struct page *page = req->wb_page;
|
||||||
|
|
||||||
|
if (test_bit(NFS_IOHDR_EOF, &hdr->flags)) {
|
||||||
|
if (bytes > hdr->good_bytes)
|
||||||
|
zero_user(page, 0, PAGE_SIZE);
|
||||||
|
else if (hdr->good_bytes - bytes < PAGE_SIZE)
|
||||||
|
zero_user_segment(page,
|
||||||
|
hdr->good_bytes & ~PAGE_MASK,
|
||||||
|
PAGE_SIZE);
|
||||||
|
}
|
||||||
|
SetPageUptodate(page);
|
||||||
|
nfs_list_remove_request(req);
|
||||||
|
nfs_readpage_release(req);
|
||||||
|
bytes += PAGE_SIZE;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
while (!list_empty(&hdr->pages)) {
|
||||||
|
struct nfs_page *req = nfs_list_entry(hdr->pages.next);
|
||||||
|
|
||||||
|
bytes += req->wb_bytes;
|
||||||
|
if (bytes <= hdr->good_bytes)
|
||||||
|
SetPageUptodate(req->wb_page);
|
||||||
|
nfs_list_remove_request(req);
|
||||||
|
nfs_readpage_release(req);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out:
|
||||||
|
hdr->release(hdr);
|
||||||
|
}
|
||||||
|
|
||||||
int nfs_initiate_read(struct rpc_clnt *clnt,
|
int nfs_initiate_read(struct rpc_clnt *clnt,
|
||||||
struct nfs_read_data *data,
|
struct nfs_read_data *data,
|
||||||
const struct rpc_call_ops *call_ops)
|
const struct rpc_call_ops *call_ops)
|
||||||
@ -214,16 +253,12 @@ EXPORT_SYMBOL_GPL(nfs_initiate_read);
|
|||||||
/*
|
/*
|
||||||
* Set up the NFS read request struct
|
* Set up the NFS read request struct
|
||||||
*/
|
*/
|
||||||
static void nfs_read_rpcsetup(struct nfs_page *req, struct nfs_read_data *data,
|
static void nfs_read_rpcsetup(struct nfs_read_data *data,
|
||||||
unsigned int count, unsigned int offset)
|
unsigned int count, unsigned int offset)
|
||||||
{
|
{
|
||||||
struct inode *inode = data->header->inode;
|
struct nfs_page *req = data->header->req;
|
||||||
|
|
||||||
data->header->req = req;
|
data->args.fh = NFS_FH(data->header->inode);
|
||||||
data->header->inode = inode;
|
|
||||||
data->header->cred = req->wb_context->cred;
|
|
||||||
|
|
||||||
data->args.fh = NFS_FH(inode);
|
|
||||||
data->args.offset = req_offset(req) + offset;
|
data->args.offset = req_offset(req) + offset;
|
||||||
data->args.pgbase = req->wb_pgbase + offset;
|
data->args.pgbase = req->wb_pgbase + offset;
|
||||||
data->args.pages = data->pages.pagevec;
|
data->args.pages = data->pages.pagevec;
|
||||||
@ -255,7 +290,7 @@ nfs_do_multiple_reads(struct list_head *head,
|
|||||||
while (!list_empty(head)) {
|
while (!list_empty(head)) {
|
||||||
int ret2;
|
int ret2;
|
||||||
|
|
||||||
data = list_entry(head->next, struct nfs_read_data, list);
|
data = list_first_entry(head, struct nfs_read_data, list);
|
||||||
list_del_init(&data->list);
|
list_del_init(&data->list);
|
||||||
|
|
||||||
ret2 = nfs_do_read(data, call_ops);
|
ret2 = nfs_do_read(data, call_ops);
|
||||||
@ -265,7 +300,7 @@ nfs_do_multiple_reads(struct list_head *head,
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
void
|
||||||
nfs_async_read_error(struct list_head *head)
|
nfs_async_read_error(struct list_head *head)
|
||||||
{
|
{
|
||||||
struct nfs_page *req;
|
struct nfs_page *req;
|
||||||
@ -290,11 +325,11 @@ nfs_async_read_error(struct list_head *head)
|
|||||||
* won't see the new data until our attribute cache is updated. This is more
|
* won't see the new data until our attribute cache is updated. This is more
|
||||||
* or less conventional NFS client behavior.
|
* or less conventional NFS client behavior.
|
||||||
*/
|
*/
|
||||||
static int nfs_pagein_multi(struct nfs_pageio_descriptor *desc, struct list_head *res)
|
static int nfs_pagein_multi(struct nfs_pageio_descriptor *desc,
|
||||||
|
struct nfs_pgio_header *hdr)
|
||||||
{
|
{
|
||||||
struct nfs_page *req = nfs_list_entry(desc->pg_list.next);
|
struct nfs_page *req = hdr->req;
|
||||||
struct page *page = req->wb_page;
|
struct page *page = req->wb_page;
|
||||||
struct nfs_read_header *rhdr;
|
|
||||||
struct nfs_read_data *data;
|
struct nfs_read_data *data;
|
||||||
size_t rsize = desc->pg_bsize, nbytes;
|
size_t rsize = desc->pg_bsize, nbytes;
|
||||||
unsigned int offset;
|
unsigned int offset;
|
||||||
@ -302,85 +337,97 @@ static int nfs_pagein_multi(struct nfs_pageio_descriptor *desc, struct list_head
|
|||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
nfs_list_remove_request(req);
|
nfs_list_remove_request(req);
|
||||||
|
nfs_list_add_request(req, &hdr->pages);
|
||||||
|
|
||||||
offset = 0;
|
offset = 0;
|
||||||
nbytes = desc->pg_count;
|
nbytes = desc->pg_count;
|
||||||
do {
|
do {
|
||||||
size_t len = min(nbytes,rsize);
|
size_t len = min(nbytes,rsize);
|
||||||
|
|
||||||
rhdr = nfs_readhdr_alloc(1);
|
data = nfs_readdata_alloc(hdr, 1);
|
||||||
if (!rhdr)
|
if (!data)
|
||||||
goto out_bad;
|
goto out_bad;
|
||||||
data = &rhdr->rpc_data;
|
|
||||||
data->pages.pagevec[0] = page;
|
data->pages.pagevec[0] = page;
|
||||||
nfs_read_rpcsetup(req, data, len, offset);
|
nfs_read_rpcsetup(data, len, offset);
|
||||||
list_add(&data->list, res);
|
list_add(&data->list, &hdr->rpc_list);
|
||||||
requests++;
|
requests++;
|
||||||
nbytes -= len;
|
nbytes -= len;
|
||||||
offset += len;
|
offset += len;
|
||||||
} while(nbytes != 0);
|
} while(nbytes != 0);
|
||||||
atomic_set(&req->wb_complete, requests);
|
desc->pg_rpc_callops = &nfs_read_common_ops;
|
||||||
desc->pg_rpc_callops = &nfs_read_partial_ops;
|
|
||||||
return ret;
|
return ret;
|
||||||
out_bad:
|
out_bad:
|
||||||
while (!list_empty(res)) {
|
while (!list_empty(&hdr->rpc_list)) {
|
||||||
data = list_entry(res->next, struct nfs_read_data, list);
|
data = list_first_entry(&hdr->rpc_list, struct nfs_read_data, list);
|
||||||
list_del(&data->list);
|
list_del(&data->list);
|
||||||
nfs_readdata_release(data);
|
nfs_readdata_release(data);
|
||||||
}
|
}
|
||||||
nfs_readpage_release(req);
|
nfs_async_read_error(&hdr->pages);
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nfs_pagein_one(struct nfs_pageio_descriptor *desc, struct list_head *res)
|
static int nfs_pagein_one(struct nfs_pageio_descriptor *desc,
|
||||||
|
struct nfs_pgio_header *hdr)
|
||||||
{
|
{
|
||||||
struct nfs_page *req;
|
struct nfs_page *req;
|
||||||
struct page **pages;
|
struct page **pages;
|
||||||
struct nfs_read_header *rhdr;
|
struct nfs_read_data *data;
|
||||||
struct nfs_read_data *data;
|
|
||||||
struct list_head *head = &desc->pg_list;
|
struct list_head *head = &desc->pg_list;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
|
|
||||||
rhdr = nfs_readhdr_alloc(nfs_page_array_len(desc->pg_base,
|
data = nfs_readdata_alloc(hdr, nfs_page_array_len(desc->pg_base,
|
||||||
desc->pg_count));
|
desc->pg_count));
|
||||||
if (!rhdr) {
|
if (!data) {
|
||||||
nfs_async_read_error(head);
|
nfs_async_read_error(head);
|
||||||
ret = -ENOMEM;
|
ret = -ENOMEM;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
data = &rhdr->rpc_data;
|
|
||||||
pages = data->pages.pagevec;
|
pages = data->pages.pagevec;
|
||||||
while (!list_empty(head)) {
|
while (!list_empty(head)) {
|
||||||
req = nfs_list_entry(head->next);
|
req = nfs_list_entry(head->next);
|
||||||
nfs_list_remove_request(req);
|
nfs_list_remove_request(req);
|
||||||
nfs_list_add_request(req, &rhdr->header.pages);
|
nfs_list_add_request(req, &hdr->pages);
|
||||||
*pages++ = req->wb_page;
|
*pages++ = req->wb_page;
|
||||||
}
|
}
|
||||||
req = nfs_list_entry(rhdr->header.pages.next);
|
|
||||||
|
|
||||||
nfs_read_rpcsetup(req, data, desc->pg_count, 0);
|
nfs_read_rpcsetup(data, desc->pg_count, 0);
|
||||||
list_add(&data->list, res);
|
list_add(&data->list, &hdr->rpc_list);
|
||||||
desc->pg_rpc_callops = &nfs_read_full_ops;
|
desc->pg_rpc_callops = &nfs_read_common_ops;
|
||||||
out:
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
int nfs_generic_pagein(struct nfs_pageio_descriptor *desc, struct list_head *head)
|
int nfs_generic_pagein(struct nfs_pageio_descriptor *desc,
|
||||||
|
struct nfs_pgio_header *hdr)
|
||||||
{
|
{
|
||||||
if (desc->pg_bsize < PAGE_CACHE_SIZE)
|
if (desc->pg_bsize < PAGE_CACHE_SIZE)
|
||||||
return nfs_pagein_multi(desc, head);
|
return nfs_pagein_multi(desc, hdr);
|
||||||
return nfs_pagein_one(desc, head);
|
return nfs_pagein_one(desc, hdr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int nfs_generic_pg_readpages(struct nfs_pageio_descriptor *desc)
|
static int nfs_generic_pg_readpages(struct nfs_pageio_descriptor *desc)
|
||||||
{
|
{
|
||||||
LIST_HEAD(head);
|
struct nfs_read_header *rhdr;
|
||||||
|
struct nfs_pgio_header *hdr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = nfs_generic_pagein(desc, &head);
|
rhdr = nfs_readhdr_alloc();
|
||||||
|
if (!rhdr) {
|
||||||
|
nfs_async_read_error(&desc->pg_list);
|
||||||
|
return -ENOMEM;
|
||||||
|
}
|
||||||
|
hdr = &rhdr->header;
|
||||||
|
nfs_pgheader_init(desc, hdr, nfs_readhdr_free);
|
||||||
|
atomic_inc(&hdr->refcnt);
|
||||||
|
ret = nfs_generic_pagein(desc, hdr);
|
||||||
if (ret == 0)
|
if (ret == 0)
|
||||||
ret = nfs_do_multiple_reads(&head, desc->pg_rpc_callops);
|
ret = nfs_do_multiple_reads(&hdr->rpc_list,
|
||||||
|
desc->pg_rpc_callops);
|
||||||
|
else
|
||||||
|
set_bit(NFS_IOHDR_REDO, &hdr->flags);
|
||||||
|
if (atomic_dec_and_test(&hdr->refcnt))
|
||||||
|
nfs_read_completion(hdr);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -419,15 +466,13 @@ static void nfs_readpage_retry(struct rpc_task *task, struct nfs_read_data *data
|
|||||||
struct nfs_readargs *argp = &data->args;
|
struct nfs_readargs *argp = &data->args;
|
||||||
struct nfs_readres *resp = &data->res;
|
struct nfs_readres *resp = &data->res;
|
||||||
|
|
||||||
if (resp->eof || resp->count == argp->count)
|
|
||||||
return;
|
|
||||||
|
|
||||||
/* This is a short read! */
|
/* This is a short read! */
|
||||||
nfs_inc_stats(data->header->inode, NFSIOS_SHORTREAD);
|
nfs_inc_stats(data->header->inode, NFSIOS_SHORTREAD);
|
||||||
/* Has the server at least made some progress? */
|
/* Has the server at least made some progress? */
|
||||||
if (resp->count == 0)
|
if (resp->count == 0) {
|
||||||
|
nfs_set_pgio_error(data->header, -EIO, argp->offset);
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
/* Yes, so retry the read at the end of the data */
|
/* Yes, so retry the read at the end of the data */
|
||||||
data->mds_offset += resp->count;
|
data->mds_offset += resp->count;
|
||||||
argp->offset += resp->count;
|
argp->offset += resp->count;
|
||||||
@ -436,38 +481,34 @@ static void nfs_readpage_retry(struct rpc_task *task, struct nfs_read_data *data
|
|||||||
rpc_restart_call_prepare(task);
|
rpc_restart_call_prepare(task);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
static void nfs_readpage_result_common(struct rpc_task *task, void *calldata)
|
||||||
* Handle a read reply that fills part of a page.
|
|
||||||
*/
|
|
||||||
static void nfs_readpage_result_partial(struct rpc_task *task, void *calldata)
|
|
||||||
{
|
{
|
||||||
struct nfs_read_data *data = calldata;
|
struct nfs_read_data *data = calldata;
|
||||||
|
struct nfs_pgio_header *hdr = data->header;
|
||||||
|
|
||||||
|
/* Note the only returns of nfs_readpage_result are 0 and -EAGAIN */
|
||||||
if (nfs_readpage_result(task, data) != 0)
|
if (nfs_readpage_result(task, data) != 0)
|
||||||
return;
|
return;
|
||||||
if (task->tk_status < 0)
|
if (task->tk_status < 0)
|
||||||
return;
|
nfs_set_pgio_error(hdr, task->tk_status, data->args.offset);
|
||||||
|
else if (data->res.eof) {
|
||||||
|
loff_t bound;
|
||||||
|
|
||||||
nfs_readpage_truncate_uninitialised_page(data);
|
bound = data->args.offset + data->res.count;
|
||||||
nfs_readpage_retry(task, data);
|
spin_lock(&hdr->lock);
|
||||||
|
if (bound < hdr->io_start + hdr->good_bytes) {
|
||||||
|
set_bit(NFS_IOHDR_EOF, &hdr->flags);
|
||||||
|
clear_bit(NFS_IOHDR_ERROR, &hdr->flags);
|
||||||
|
hdr->good_bytes = bound - hdr->io_start;
|
||||||
|
}
|
||||||
|
spin_unlock(&hdr->lock);
|
||||||
|
} else if (data->res.count != data->args.count)
|
||||||
|
nfs_readpage_retry(task, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void nfs_readpage_release_partial(void *calldata)
|
static void nfs_readpage_release_common(void *calldata)
|
||||||
{
|
{
|
||||||
struct nfs_read_data *data = calldata;
|
nfs_readdata_release(calldata);
|
||||||
struct nfs_page *req = data->header->req;
|
|
||||||
struct page *page = req->wb_page;
|
|
||||||
int status = data->task.tk_status;
|
|
||||||
|
|
||||||
if (status < 0)
|
|
||||||
set_bit(PG_PARTIAL_READ_FAILED, &req->wb_flags);
|
|
||||||
|
|
||||||
if (atomic_dec_and_test(&req->wb_complete)) {
|
|
||||||
if (!test_bit(PG_PARTIAL_READ_FAILED, &req->wb_flags))
|
|
||||||
SetPageUptodate(page);
|
|
||||||
nfs_readpage_release(req);
|
|
||||||
}
|
|
||||||
nfs_readdata_release(data);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void nfs_read_prepare(struct rpc_task *task, void *calldata)
|
void nfs_read_prepare(struct rpc_task *task, void *calldata)
|
||||||
@ -476,75 +517,10 @@ void nfs_read_prepare(struct rpc_task *task, void *calldata)
|
|||||||
NFS_PROTO(data->header->inode)->read_rpc_prepare(task, data);
|
NFS_PROTO(data->header->inode)->read_rpc_prepare(task, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
static const struct rpc_call_ops nfs_read_partial_ops = {
|
static const struct rpc_call_ops nfs_read_common_ops = {
|
||||||
.rpc_call_prepare = nfs_read_prepare,
|
.rpc_call_prepare = nfs_read_prepare,
|
||||||
.rpc_call_done = nfs_readpage_result_partial,
|
.rpc_call_done = nfs_readpage_result_common,
|
||||||
.rpc_release = nfs_readpage_release_partial,
|
.rpc_release = nfs_readpage_release_common,
|
||||||
};
|
|
||||||
|
|
||||||
static void nfs_readpage_set_pages_uptodate(struct nfs_read_data *data)
|
|
||||||
{
|
|
||||||
unsigned int count = data->res.count;
|
|
||||||
unsigned int base = data->args.pgbase;
|
|
||||||
struct page **pages;
|
|
||||||
|
|
||||||
if (data->res.eof)
|
|
||||||
count = data->args.count;
|
|
||||||
if (unlikely(count == 0))
|
|
||||||
return;
|
|
||||||
pages = &data->args.pages[base >> PAGE_CACHE_SHIFT];
|
|
||||||
base &= ~PAGE_CACHE_MASK;
|
|
||||||
count += base;
|
|
||||||
for (;count >= PAGE_CACHE_SIZE; count -= PAGE_CACHE_SIZE, pages++)
|
|
||||||
SetPageUptodate(*pages);
|
|
||||||
if (count == 0)
|
|
||||||
return;
|
|
||||||
/* Was this a short read? */
|
|
||||||
if (data->res.eof || data->res.count == data->args.count)
|
|
||||||
SetPageUptodate(*pages);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This is the callback from RPC telling us whether a reply was
|
|
||||||
* received or some error occurred (timeout or socket shutdown).
|
|
||||||
*/
|
|
||||||
static void nfs_readpage_result_full(struct rpc_task *task, void *calldata)
|
|
||||||
{
|
|
||||||
struct nfs_read_data *data = calldata;
|
|
||||||
|
|
||||||
if (nfs_readpage_result(task, data) != 0)
|
|
||||||
return;
|
|
||||||
if (task->tk_status < 0)
|
|
||||||
return;
|
|
||||||
/*
|
|
||||||
* Note: nfs_readpage_retry may change the values of
|
|
||||||
* data->args. In the multi-page case, we therefore need
|
|
||||||
* to ensure that we call nfs_readpage_set_pages_uptodate()
|
|
||||||
* first.
|
|
||||||
*/
|
|
||||||
nfs_readpage_truncate_uninitialised_page(data);
|
|
||||||
nfs_readpage_set_pages_uptodate(data);
|
|
||||||
nfs_readpage_retry(task, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void nfs_readpage_release_full(void *calldata)
|
|
||||||
{
|
|
||||||
struct nfs_read_data *data = calldata;
|
|
||||||
struct nfs_pgio_header *hdr = data->header;
|
|
||||||
|
|
||||||
while (!list_empty(&hdr->pages)) {
|
|
||||||
struct nfs_page *req = nfs_list_entry(hdr->pages.next);
|
|
||||||
|
|
||||||
nfs_list_remove_request(req);
|
|
||||||
nfs_readpage_release(req);
|
|
||||||
}
|
|
||||||
nfs_readdata_release(calldata);
|
|
||||||
}
|
|
||||||
|
|
||||||
static const struct rpc_call_ops nfs_read_full_ops = {
|
|
||||||
.rpc_call_prepare = nfs_read_prepare,
|
|
||||||
.rpc_call_done = nfs_readpage_result_full,
|
|
||||||
.rpc_release = nfs_readpage_release_full,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -27,7 +27,6 @@ enum {
|
|||||||
PG_CLEAN,
|
PG_CLEAN,
|
||||||
PG_NEED_COMMIT,
|
PG_NEED_COMMIT,
|
||||||
PG_NEED_RESCHED,
|
PG_NEED_RESCHED,
|
||||||
PG_PARTIAL_READ_FAILED,
|
|
||||||
PG_COMMIT_TO_DS,
|
PG_COMMIT_TO_DS,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1187,14 +1187,30 @@ struct nfs_read_data {
|
|||||||
struct nfs_client *ds_clp; /* pNFS data server */
|
struct nfs_client *ds_clp; /* pNFS data server */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/* used as flag bits in nfs_pgio_header */
|
||||||
|
enum {
|
||||||
|
NFS_IOHDR_ERROR = 0,
|
||||||
|
NFS_IOHDR_EOF,
|
||||||
|
NFS_IOHDR_REDO,
|
||||||
|
};
|
||||||
|
|
||||||
struct nfs_pgio_header {
|
struct nfs_pgio_header {
|
||||||
struct inode *inode;
|
struct inode *inode;
|
||||||
struct rpc_cred *cred;
|
struct rpc_cred *cred;
|
||||||
struct list_head pages;
|
struct list_head pages;
|
||||||
|
struct list_head rpc_list;
|
||||||
|
atomic_t refcnt;
|
||||||
struct nfs_page *req;
|
struct nfs_page *req;
|
||||||
struct pnfs_layout_segment *lseg;
|
struct pnfs_layout_segment *lseg;
|
||||||
|
loff_t io_start;
|
||||||
const struct rpc_call_ops *mds_ops;
|
const struct rpc_call_ops *mds_ops;
|
||||||
|
void (*release) (struct nfs_pgio_header *hdr);
|
||||||
|
spinlock_t lock;
|
||||||
|
/* fields protected by lock */
|
||||||
int pnfs_error;
|
int pnfs_error;
|
||||||
|
int error; /* merge with pnfs_error */
|
||||||
|
unsigned long good_bytes; /* boundary of good data */
|
||||||
|
unsigned long flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct nfs_read_header {
|
struct nfs_read_header {
|
||||||
|
Loading…
Reference in New Issue
Block a user