aboutsummaryrefslogtreecommitdiffstats
path: root/fs/nfs/pagelist.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/nfs/pagelist.c')
-rw-r--r--fs/nfs/pagelist.c120
1 files changed, 58 insertions, 62 deletions
diff --git a/fs/nfs/pagelist.c b/fs/nfs/pagelist.c
index 5e70918f6c95..ecb3d4cdbc85 100644
--- a/fs/nfs/pagelist.c
+++ b/fs/nfs/pagelist.c
@@ -484,8 +484,7 @@ EXPORT_SYMBOL_GPL(nfs_pgio_header_free);
484static bool nfs_pgio_data_init(struct nfs_pgio_header *hdr, 484static bool nfs_pgio_data_init(struct nfs_pgio_header *hdr,
485 unsigned int pagecount) 485 unsigned int pagecount)
486{ 486{
487 if (nfs_pgarray_set(&hdr->data.page_array, pagecount)) { 487 if (nfs_pgarray_set(&hdr->page_array, pagecount)) {
488 hdr->data.header = hdr;
489 atomic_inc(&hdr->refcnt); 488 atomic_inc(&hdr->refcnt);
490 return true; 489 return true;
491 } 490 }
@@ -493,16 +492,14 @@ static bool nfs_pgio_data_init(struct nfs_pgio_header *hdr,
493} 492}
494 493
495/** 494/**
496 * nfs_pgio_data_destroy - Properly free pageio data 495 * nfs_pgio_data_destroy - Properly release pageio data
497 * @data: The data to destroy 496 * @hdr: The header with data to destroy
498 */ 497 */
499void nfs_pgio_data_destroy(struct nfs_pgio_data *data) 498void nfs_pgio_data_destroy(struct nfs_pgio_header *hdr)
500{ 499{
501 struct nfs_pgio_header *hdr = data->header; 500 put_nfs_open_context(hdr->args.context);
502 501 if (hdr->page_array.pagevec != hdr->page_array.page_array)
503 put_nfs_open_context(data->args.context); 502 kfree(hdr->page_array.pagevec);
504 if (data->page_array.pagevec != data->page_array.page_array)
505 kfree(data->page_array.pagevec);
506 if (atomic_dec_and_test(&hdr->refcnt)) 503 if (atomic_dec_and_test(&hdr->refcnt))
507 hdr->completion_ops->completion(hdr); 504 hdr->completion_ops->completion(hdr);
508} 505}
@@ -510,31 +507,31 @@ EXPORT_SYMBOL_GPL(nfs_pgio_data_destroy);
510 507
511/** 508/**
512 * nfs_pgio_rpcsetup - Set up arguments for a pageio call 509 * nfs_pgio_rpcsetup - Set up arguments for a pageio call
513 * @data: The pageio data 510 * @hdr: The pageio hdr
514 * @count: Number of bytes to read 511 * @count: Number of bytes to read
515 * @offset: Initial offset 512 * @offset: Initial offset
516 * @how: How to commit data (writes only) 513 * @how: How to commit data (writes only)
517 * @cinfo: Commit information for the call (writes only) 514 * @cinfo: Commit information for the call (writes only)
518 */ 515 */
519static void nfs_pgio_rpcsetup(struct nfs_pgio_data *data, 516static void nfs_pgio_rpcsetup(struct nfs_pgio_header *hdr,
520 unsigned int count, unsigned int offset, 517 unsigned int count, unsigned int offset,
521 int how, struct nfs_commit_info *cinfo) 518 int how, struct nfs_commit_info *cinfo)
522{ 519{
523 struct nfs_page *req = data->header->req; 520 struct nfs_page *req = hdr->req;
524 521
525 /* Set up the RPC argument and reply structs 522 /* Set up the RPC argument and reply structs
526 * NB: take care not to mess about with data->commit et al. */ 523 * NB: take care not to mess about with hdr->commit et al. */
527 524
528 data->args.fh = NFS_FH(data->header->inode); 525 hdr->args.fh = NFS_FH(hdr->inode);
529 data->args.offset = req_offset(req) + offset; 526 hdr->args.offset = req_offset(req) + offset;
530 /* pnfs_set_layoutcommit needs this */ 527 /* pnfs_set_layoutcommit needs this */
531 data->mds_offset = data->args.offset; 528 hdr->mds_offset = hdr->args.offset;
532 data->args.pgbase = req->wb_pgbase + offset; 529 hdr->args.pgbase = req->wb_pgbase + offset;
533 data->args.pages = data->page_array.pagevec; 530 hdr->args.pages = hdr->page_array.pagevec;
534 data->args.count = count; 531 hdr->args.count = count;
535 data->args.context = get_nfs_open_context(req->wb_context); 532 hdr->args.context = get_nfs_open_context(req->wb_context);
536 data->args.lock_context = req->wb_lock_context; 533 hdr->args.lock_context = req->wb_lock_context;
537 data->args.stable = NFS_UNSTABLE; 534 hdr->args.stable = NFS_UNSTABLE;
538 switch (how & (FLUSH_STABLE | FLUSH_COND_STABLE)) { 535 switch (how & (FLUSH_STABLE | FLUSH_COND_STABLE)) {
539 case 0: 536 case 0:
540 break; 537 break;
@@ -542,59 +539,60 @@ static void nfs_pgio_rpcsetup(struct nfs_pgio_data *data,
542 if (nfs_reqs_to_commit(cinfo)) 539 if (nfs_reqs_to_commit(cinfo))
543 break; 540 break;
544 default: 541 default:
545 data->args.stable = NFS_FILE_SYNC; 542 hdr->args.stable = NFS_FILE_SYNC;
546 } 543 }
547 544
548 data->res.fattr = &data->fattr; 545 hdr->res.fattr = &hdr->fattr;
549 data->res.count = count; 546 hdr->res.count = count;
550 data->res.eof = 0; 547 hdr->res.eof = 0;
551 data->res.verf = &data->writeverf; 548 hdr->res.verf = &hdr->writeverf;
552 nfs_fattr_init(&data->fattr); 549 nfs_fattr_init(&hdr->fattr);
553} 550}
554 551
555/** 552/**
556 * nfs_pgio_prepare - Prepare pageio data to go over the wire 553 * nfs_pgio_prepare - Prepare pageio hdr to go over the wire
557 * @task: The current task 554 * @task: The current task
558 * @calldata: pageio data to prepare 555 * @calldata: pageio header to prepare
559 */ 556 */
560static void nfs_pgio_prepare(struct rpc_task *task, void *calldata) 557static void nfs_pgio_prepare(struct rpc_task *task, void *calldata)
561{ 558{
562 struct nfs_pgio_data *data = calldata; 559 struct nfs_pgio_header *hdr = calldata;
563 int err; 560 int err;
564 err = NFS_PROTO(data->header->inode)->pgio_rpc_prepare(task, data); 561 err = NFS_PROTO(hdr->inode)->pgio_rpc_prepare(task, hdr);
565 if (err) 562 if (err)
566 rpc_exit(task, err); 563 rpc_exit(task, err);
567} 564}
568 565
569int nfs_initiate_pgio(struct rpc_clnt *clnt, struct nfs_pgio_data *data, 566int nfs_initiate_pgio(struct rpc_clnt *clnt, struct nfs_pgio_header *hdr,
570 const struct rpc_call_ops *call_ops, int how, int flags) 567 const struct rpc_call_ops *call_ops, int how, int flags)
571{ 568{
569 struct inode *inode = hdr->inode;
572 struct rpc_task *task; 570 struct rpc_task *task;
573 struct rpc_message msg = { 571 struct rpc_message msg = {
574 .rpc_argp = &data->args, 572 .rpc_argp = &hdr->args,
575 .rpc_resp = &data->res, 573 .rpc_resp = &hdr->res,
576 .rpc_cred = data->header->cred, 574 .rpc_cred = hdr->cred,
577 }; 575 };
578 struct rpc_task_setup task_setup_data = { 576 struct rpc_task_setup task_setup_data = {
579 .rpc_client = clnt, 577 .rpc_client = clnt,
580 .task = &data->task, 578 .task = &hdr->task,
581 .rpc_message = &msg, 579 .rpc_message = &msg,
582 .callback_ops = call_ops, 580 .callback_ops = call_ops,
583 .callback_data = data, 581 .callback_data = hdr,
584 .workqueue = nfsiod_workqueue, 582 .workqueue = nfsiod_workqueue,
585 .flags = RPC_TASK_ASYNC | flags, 583 .flags = RPC_TASK_ASYNC | flags,
586 }; 584 };
587 int ret = 0; 585 int ret = 0;
588 586
589 data->header->rw_ops->rw_initiate(data, &msg, &task_setup_data, how); 587 hdr->rw_ops->rw_initiate(hdr, &msg, &task_setup_data, how);
590 588
591 dprintk("NFS: %5u initiated pgio call " 589 dprintk("NFS: %5u initiated pgio call "
592 "(req %s/%llu, %u bytes @ offset %llu)\n", 590 "(req %s/%llu, %u bytes @ offset %llu)\n",
593 data->task.tk_pid, 591 hdr->task.tk_pid,
594 data->header->inode->i_sb->s_id, 592 inode->i_sb->s_id,
595 (unsigned long long)NFS_FILEID(data->header->inode), 593 (unsigned long long)NFS_FILEID(inode),
596 data->args.count, 594 hdr->args.count,
597 (unsigned long long)data->args.offset); 595 (unsigned long long)hdr->args.offset);
598 596
599 task = rpc_run_task(&task_setup_data); 597 task = rpc_run_task(&task_setup_data);
600 if (IS_ERR(task)) { 598 if (IS_ERR(task)) {
@@ -621,21 +619,21 @@ static int nfs_pgio_error(struct nfs_pageio_descriptor *desc,
621 struct nfs_pgio_header *hdr) 619 struct nfs_pgio_header *hdr)
622{ 620{
623 set_bit(NFS_IOHDR_REDO, &hdr->flags); 621 set_bit(NFS_IOHDR_REDO, &hdr->flags);
624 nfs_pgio_data_destroy(&hdr->data); 622 nfs_pgio_data_destroy(hdr);
625 desc->pg_completion_ops->error_cleanup(&desc->pg_list); 623 desc->pg_completion_ops->error_cleanup(&desc->pg_list);
626 return -ENOMEM; 624 return -ENOMEM;
627} 625}
628 626
629/** 627/**
630 * nfs_pgio_release - Release pageio data 628 * nfs_pgio_release - Release pageio data
631 * @calldata: The pageio data to release 629 * @calldata: The pageio header to release
632 */ 630 */
633static void nfs_pgio_release(void *calldata) 631static void nfs_pgio_release(void *calldata)
634{ 632{
635 struct nfs_pgio_data *data = calldata; 633 struct nfs_pgio_header *hdr = calldata;
636 if (data->header->rw_ops->rw_release) 634 if (hdr->rw_ops->rw_release)
637 data->header->rw_ops->rw_release(data); 635 hdr->rw_ops->rw_release(hdr);
638 nfs_pgio_data_destroy(data); 636 nfs_pgio_data_destroy(hdr);
639} 637}
640 638
641/** 639/**
@@ -676,22 +674,22 @@ EXPORT_SYMBOL_GPL(nfs_pageio_init);
676/** 674/**
677 * nfs_pgio_result - Basic pageio error handling 675 * nfs_pgio_result - Basic pageio error handling
678 * @task: The task that ran 676 * @task: The task that ran
679 * @calldata: Pageio data to check 677 * @calldata: Pageio header to check
680 */ 678 */
681static void nfs_pgio_result(struct rpc_task *task, void *calldata) 679static void nfs_pgio_result(struct rpc_task *task, void *calldata)
682{ 680{
683 struct nfs_pgio_data *data = calldata; 681 struct nfs_pgio_header *hdr = calldata;
684 struct inode *inode = data->header->inode; 682 struct inode *inode = hdr->inode;
685 683
686 dprintk("NFS: %s: %5u, (status %d)\n", __func__, 684 dprintk("NFS: %s: %5u, (status %d)\n", __func__,
687 task->tk_pid, task->tk_status); 685 task->tk_pid, task->tk_status);
688 686
689 if (data->header->rw_ops->rw_done(task, data, inode) != 0) 687 if (hdr->rw_ops->rw_done(task, hdr, inode) != 0)
690 return; 688 return;
691 if (task->tk_status < 0) 689 if (task->tk_status < 0)
692 nfs_set_pgio_error(data->header, task->tk_status, data->args.offset); 690 nfs_set_pgio_error(hdr, task->tk_status, hdr->args.offset);
693 else 691 else
694 data->header->rw_ops->rw_result(task, data); 692 hdr->rw_ops->rw_result(task, hdr);
695} 693}
696 694
697/* 695/*
@@ -707,7 +705,6 @@ int nfs_generic_pgio(struct nfs_pageio_descriptor *desc,
707{ 705{
708 struct nfs_page *req; 706 struct nfs_page *req;
709 struct page **pages; 707 struct page **pages;
710 struct nfs_pgio_data *data;
711 struct list_head *head = &desc->pg_list; 708 struct list_head *head = &desc->pg_list;
712 struct nfs_commit_info cinfo; 709 struct nfs_commit_info cinfo;
713 710
@@ -715,9 +712,8 @@ int nfs_generic_pgio(struct nfs_pageio_descriptor *desc,
715 desc->pg_count))) 712 desc->pg_count)))
716 return nfs_pgio_error(desc, hdr); 713 return nfs_pgio_error(desc, hdr);
717 714
718 data = &hdr->data;
719 nfs_init_cinfo(&cinfo, desc->pg_inode, desc->pg_dreq); 715 nfs_init_cinfo(&cinfo, desc->pg_inode, desc->pg_dreq);
720 pages = data->page_array.pagevec; 716 pages = hdr->page_array.pagevec;
721 while (!list_empty(head)) { 717 while (!list_empty(head)) {
722 req = nfs_list_entry(head->next); 718 req = nfs_list_entry(head->next);
723 nfs_list_remove_request(req); 719 nfs_list_remove_request(req);
@@ -730,7 +726,7 @@ int nfs_generic_pgio(struct nfs_pageio_descriptor *desc,
730 desc->pg_ioflags &= ~FLUSH_COND_STABLE; 726 desc->pg_ioflags &= ~FLUSH_COND_STABLE;
731 727
732 /* Set up the argument struct */ 728 /* Set up the argument struct */
733 nfs_pgio_rpcsetup(data, desc->pg_count, 0, desc->pg_ioflags, &cinfo); 729 nfs_pgio_rpcsetup(hdr, desc->pg_count, 0, desc->pg_ioflags, &cinfo);
734 desc->pg_rpc_callops = &nfs_pgio_common_ops; 730 desc->pg_rpc_callops = &nfs_pgio_common_ops;
735 return 0; 731 return 0;
736} 732}
@@ -751,7 +747,7 @@ static int nfs_generic_pg_pgios(struct nfs_pageio_descriptor *desc)
751 ret = nfs_generic_pgio(desc, hdr); 747 ret = nfs_generic_pgio(desc, hdr);
752 if (ret == 0) 748 if (ret == 0)
753 ret = nfs_initiate_pgio(NFS_CLIENT(hdr->inode), 749 ret = nfs_initiate_pgio(NFS_CLIENT(hdr->inode),
754 &hdr->data, desc->pg_rpc_callops, 750 hdr, desc->pg_rpc_callops,
755 desc->pg_ioflags, 0); 751 desc->pg_ioflags, 0);
756 if (atomic_dec_and_test(&hdr->refcnt)) 752 if (atomic_dec_and_test(&hdr->refcnt))
757 hdr->completion_ops->completion(hdr); 753 hdr->completion_ops->completion(hdr);