Skip to content

Commit 6c75dc0

Browse files
Fred IsamanTrond Myklebust
authored andcommitted
NFS: merge _full and _partial write rpc_ops
Decouple nfs_pgio_header and nfs_write_data, and have (possibly multiple) nfs_write_datas each take a refcount on nfs_pgio_header. For the moment keeps nfs_write_header as a way to preallocate a single nfs_write_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_write_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>
1 parent 4db6e0b commit 6c75dc0

File tree

7 files changed

+227
-239
lines changed

7 files changed

+227
-239
lines changed

fs/nfs/direct.c

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -768,11 +768,17 @@ static ssize_t nfs_direct_write_schedule_segment(struct nfs_direct_req *dreq,
768768
bytes = min(wsize,count);
769769

770770
result = -ENOMEM;
771-
whdr = nfs_writehdr_alloc(nfs_page_array_len(pgbase, bytes));
771+
whdr = nfs_writehdr_alloc();
772772
if (unlikely(!whdr))
773773
break;
774774

775-
data = &whdr->rpc_data;
775+
data = nfs_writedata_alloc(&whdr->header, nfs_page_array_len(pgbase, bytes));
776+
if (!data) {
777+
nfs_writehdr_free(&whdr->header);
778+
break;
779+
}
780+
data->header = &whdr->header;
781+
atomic_inc(&data->header->refcnt);
776782
pages = &data->pages;
777783

778784
down_read(&current->mm->mmap_sem);

fs/nfs/internal.h

Lines changed: 6 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -319,10 +319,14 @@ extern void nfs_pageio_reset_read_mds(struct nfs_pageio_descriptor *pgio);
319319
extern void nfs_readdata_release(struct nfs_read_data *rdata);
320320

321321
/* write.c */
322-
extern struct nfs_write_header *nfs_writehdr_alloc(unsigned int npages);
322+
extern void nfs_async_write_error(struct list_head *head);
323+
extern struct nfs_write_header *nfs_writehdr_alloc(void);
323324
extern void nfs_writehdr_free(struct nfs_pgio_header *hdr);
325+
extern struct nfs_write_data *nfs_writedata_alloc(struct nfs_pgio_header *hdr,
326+
unsigned int pagecount);
327+
extern void nfs_write_completion(struct nfs_pgio_header *hdr);
324328
extern int nfs_generic_flush(struct nfs_pageio_descriptor *desc,
325-
struct list_head *head);
329+
struct nfs_pgio_header *hdr);
326330
extern void nfs_pageio_init_write_mds(struct nfs_pageio_descriptor *pgio,
327331
struct inode *inode, int ioflags);
328332
extern void nfs_pageio_reset_write_mds(struct nfs_pageio_descriptor *pgio);

fs/nfs/nfs4filelayout.c

Lines changed: 0 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -314,7 +314,6 @@ static void filelayout_write_release(void *data)
314314
{
315315
struct nfs_write_data *wdata = data;
316316

317-
put_lseg(wdata->header->lseg);
318317
wdata->header->mds_ops->rpc_release(data);
319318
}
320319

fs/nfs/nfs4proc.c

Lines changed: 1 addition & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -3431,8 +3431,6 @@ void nfs4_reset_write(struct rpc_task *task, struct nfs_write_data *data)
34313431
struct inode *inode = hdr->inode;
34323432

34333433
dprintk("%s Reset task for i/o through\n", __func__);
3434-
put_lseg(hdr->lseg);
3435-
hdr->lseg = NULL;
34363434
data->ds_clp = NULL;
34373435
data->write_done_cb = nfs4_write_done_cb;
34383436
data->args.fh = NFS_FH(inode);
@@ -3448,7 +3446,7 @@ static void nfs4_proc_write_setup(struct nfs_write_data *data, struct rpc_messag
34483446
{
34493447
struct nfs_server *server = NFS_SERVER(data->header->inode);
34503448

3451-
if (data->header->lseg) {
3449+
if (data->ds_clp) {
34523450
data->args.bitmask = NULL;
34533451
data->res.fattr = NULL;
34543452
} else

fs/nfs/pnfs.c

Lines changed: 36 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -1199,7 +1199,9 @@ static void pnfs_ld_handle_write_error(struct nfs_write_data *data)
11991199
clear_bit(NFS_INO_LAYOUTCOMMIT, &NFS_I(hdr->inode)->flags);
12001200
pnfs_return_layout(hdr->inode);
12011201
}
1202-
data->task.tk_status = pnfs_write_done_resend_to_mds(hdr->inode, &hdr->pages);
1202+
if (!test_and_set_bit(NFS_IOHDR_REDO, &hdr->flags))
1203+
data->task.tk_status = pnfs_write_done_resend_to_mds(hdr->inode,
1204+
&hdr->pages);
12031205
}
12041206

12051207
/*
@@ -1214,7 +1216,6 @@ void pnfs_ld_write_done(struct nfs_write_data *data)
12141216
hdr->mds_ops->rpc_call_done(&data->task, data);
12151217
} else
12161218
pnfs_ld_handle_write_error(data);
1217-
put_lseg(hdr->lseg);
12181219
hdr->mds_ops->rpc_release(data);
12191220
}
12201221
EXPORT_SYMBOL_GPL(pnfs_ld_write_done);
@@ -1225,12 +1226,11 @@ pnfs_write_through_mds(struct nfs_pageio_descriptor *desc,
12251226
{
12261227
struct nfs_pgio_header *hdr = data->header;
12271228

1228-
list_splice_tail_init(&hdr->pages, &desc->pg_list);
1229-
if (hdr->req && list_empty(&hdr->req->wb_list))
1230-
nfs_list_add_request(hdr->req, &desc->pg_list);
1231-
nfs_pageio_reset_write_mds(desc);
1232-
desc->pg_recoalesce = 1;
1233-
put_lseg(hdr->lseg);
1229+
if (!test_and_set_bit(NFS_IOHDR_REDO, &hdr->flags)) {
1230+
list_splice_tail_init(&hdr->pages, &desc->pg_list);
1231+
nfs_pageio_reset_write_mds(desc);
1232+
desc->pg_recoalesce = 1;
1233+
}
12341234
nfs_writedata_release(data);
12351235
}
12361236

@@ -1246,18 +1246,12 @@ pnfs_try_to_write_data(struct nfs_write_data *wdata,
12461246
struct nfs_server *nfss = NFS_SERVER(inode);
12471247

12481248
hdr->mds_ops = call_ops;
1249-
hdr->lseg = get_lseg(lseg);
12501249

12511250
dprintk("%s: Writing ino:%lu %u@%llu (how %d)\n", __func__,
12521251
inode->i_ino, wdata->args.count, wdata->args.offset, how);
1253-
12541252
trypnfs = nfss->pnfs_curr_ld->write_pagelist(wdata, how);
1255-
if (trypnfs == PNFS_NOT_ATTEMPTED) {
1256-
put_lseg(hdr->lseg);
1257-
hdr->lseg = NULL;
1258-
} else
1253+
if (trypnfs != PNFS_NOT_ATTEMPTED)
12591254
nfs_inc_stats(inode, NFSIOS_PNFS_WRITE);
1260-
12611255
dprintk("%s End (trypnfs:%d)\n", __func__, trypnfs);
12621256
return trypnfs;
12631257
}
@@ -1273,7 +1267,7 @@ pnfs_do_multiple_writes(struct nfs_pageio_descriptor *desc, struct list_head *he
12731267
while (!list_empty(head)) {
12741268
enum pnfs_try_status trypnfs;
12751269

1276-
data = list_entry(head->next, struct nfs_write_data, list);
1270+
data = list_first_entry(head, struct nfs_write_data, list);
12771271
list_del_init(&data->list);
12781272

12791273
trypnfs = pnfs_try_to_write_data(data, call_ops, lseg, how);
@@ -1283,20 +1277,40 @@ pnfs_do_multiple_writes(struct nfs_pageio_descriptor *desc, struct list_head *he
12831277
put_lseg(lseg);
12841278
}
12851279

1280+
static void pnfs_writehdr_free(struct nfs_pgio_header *hdr)
1281+
{
1282+
put_lseg(hdr->lseg);
1283+
nfs_writehdr_free(hdr);
1284+
}
1285+
12861286
int
12871287
pnfs_generic_pg_writepages(struct nfs_pageio_descriptor *desc)
12881288
{
1289-
LIST_HEAD(head);
1289+
struct nfs_write_header *whdr;
1290+
struct nfs_pgio_header *hdr;
12901291
int ret;
12911292

1292-
ret = nfs_generic_flush(desc, &head);
1293-
if (ret != 0) {
1293+
whdr = nfs_writehdr_alloc();
1294+
if (!whdr) {
1295+
nfs_async_write_error(&desc->pg_list);
12941296
put_lseg(desc->pg_lseg);
12951297
desc->pg_lseg = NULL;
1296-
return ret;
1298+
return -ENOMEM;
12971299
}
1298-
pnfs_do_multiple_writes(desc, &head, desc->pg_ioflags);
1299-
return 0;
1300+
hdr = &whdr->header;
1301+
nfs_pgheader_init(desc, hdr, pnfs_writehdr_free);
1302+
hdr->lseg = get_lseg(desc->pg_lseg);
1303+
atomic_inc(&hdr->refcnt);
1304+
ret = nfs_generic_flush(desc, hdr);
1305+
if (ret != 0) {
1306+
put_lseg(desc->pg_lseg);
1307+
desc->pg_lseg = NULL;
1308+
set_bit(NFS_IOHDR_REDO, &hdr->flags);
1309+
} else
1310+
pnfs_do_multiple_writes(desc, &hdr->rpc_list, desc->pg_ioflags);
1311+
if (atomic_dec_and_test(&hdr->refcnt))
1312+
nfs_write_completion(hdr);
1313+
return ret;
13001314
}
13011315
EXPORT_SYMBOL_GPL(pnfs_generic_pg_writepages);
13021316

0 commit comments

Comments
 (0)