aboutsummaryrefslogtreecommitdiffstats
path: root/fs/nfs/nfs4proc.c
diff options
context:
space:
mode:
authorTrond Myklebust <Trond.Myklebust@netapp.com>2009-06-17 16:22:59 -0400
committerTrond Myklebust <Trond.Myklebust@netapp.com>2009-06-17 16:22:59 -0400
commit965b5d679146c9f69bc0325388bb9ed357863c4f (patch)
treed70fe13004404f1d3a203c8b9b90aa5f61150115 /fs/nfs/nfs4proc.c
parentd5122201a7f90b2aa73092f158b84d1d74f1134d (diff)
NFSv4: Handle more errors when recovering open file and locking state
It is possible for servers to return NFS4ERR_BAD_STATEID when the state management code is recovering locks or is reclaiming state when returning a delegation. Ensure that we handle that case. While we're at it, add in handlers for NFS4ERR_STALE, NFS4ERR_ADMIN_REVOKED, NFS4ERR_OPENMODE, NFS4ERR_DENIED and NFS4ERR_STALE_STATEID, since the protocol appears to allow for them too. Also handle ENOMEM... Finally, rather than add new NFSv4.0-specific errors and error handling into the generic delegation code, move that open file and locking state error handling into the NFSv4 layer. Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Diffstat (limited to 'fs/nfs/nfs4proc.c')
-rw-r--r--fs/nfs/nfs4proc.c44
1 files changed, 38 insertions, 6 deletions
diff --git a/fs/nfs/nfs4proc.c b/fs/nfs/nfs4proc.c
index d326193c9785..55314e721632 100644
--- a/fs/nfs/nfs4proc.c
+++ b/fs/nfs/nfs4proc.c
@@ -804,16 +804,30 @@ int nfs4_open_delegation_recall(struct nfs_open_context *ctx, struct nfs4_state
804 err = _nfs4_open_delegation_recall(ctx, state, stateid); 804 err = _nfs4_open_delegation_recall(ctx, state, stateid);
805 switch (err) { 805 switch (err) {
806 case 0: 806 case 0:
807 return err; 807 case -ENOENT:
808 case -ESTALE:
809 goto out;
808 case -NFS4ERR_STALE_CLIENTID: 810 case -NFS4ERR_STALE_CLIENTID:
809 case -NFS4ERR_STALE_STATEID: 811 case -NFS4ERR_STALE_STATEID:
810 case -NFS4ERR_EXPIRED: 812 case -NFS4ERR_EXPIRED:
811 /* Don't recall a delegation if it was lost */ 813 /* Don't recall a delegation if it was lost */
812 nfs4_schedule_state_recovery(server->nfs_client); 814 nfs4_schedule_state_recovery(server->nfs_client);
813 return err; 815 goto out;
816 case -ERESTARTSYS:
817 /*
818 * The show must go on: exit, but mark the
819 * stateid as needing recovery.
820 */
821 case -NFS4ERR_ADMIN_REVOKED:
822 case -NFS4ERR_BAD_STATEID:
823 nfs4_state_mark_reclaim_nograce(server->nfs_client, state);
824 case -ENOMEM:
825 err = 0;
826 goto out;
814 } 827 }
815 err = nfs4_handle_exception(server, err, &exception); 828 err = nfs4_handle_exception(server, err, &exception);
816 } while (exception.retry); 829 } while (exception.retry);
830out:
817 return err; 831 return err;
818} 832}
819 833
@@ -3487,8 +3501,6 @@ static int _nfs4_do_setlk(struct nfs4_state *state, int cmd, struct file_lock *f
3487 ret = nfs4_wait_for_completion_rpc_task(task); 3501 ret = nfs4_wait_for_completion_rpc_task(task);
3488 if (ret == 0) { 3502 if (ret == 0) {
3489 ret = data->rpc_status; 3503 ret = data->rpc_status;
3490 if (ret == -NFS4ERR_DENIED)
3491 ret = -EAGAIN;
3492 } else 3504 } else
3493 data->cancelled = 1; 3505 data->cancelled = 1;
3494 rpc_put_task(task); 3506 rpc_put_task(task);
@@ -3576,9 +3588,11 @@ static int nfs4_proc_setlk(struct nfs4_state *state, int cmd, struct file_lock *
3576 int err; 3588 int err;
3577 3589
3578 do { 3590 do {
3591 err = _nfs4_proc_setlk(state, cmd, request);
3592 if (err == -NFS4ERR_DENIED)
3593 err = -EAGAIN;
3579 err = nfs4_handle_exception(NFS_SERVER(state->inode), 3594 err = nfs4_handle_exception(NFS_SERVER(state->inode),
3580 _nfs4_proc_setlk(state, cmd, request), 3595 err, &exception);
3581 &exception);
3582 } while (exception.retry); 3596 } while (exception.retry);
3583 return err; 3597 return err;
3584} 3598}
@@ -3635,11 +3649,29 @@ int nfs4_lock_delegation_recall(struct nfs4_state *state, struct file_lock *fl)
3635 printk(KERN_ERR "%s: unhandled error %d.\n", 3649 printk(KERN_ERR "%s: unhandled error %d.\n",
3636 __func__, err); 3650 __func__, err);
3637 case 0: 3651 case 0:
3652 case -ESTALE:
3638 goto out; 3653 goto out;
3639 case -NFS4ERR_EXPIRED: 3654 case -NFS4ERR_EXPIRED:
3640 case -NFS4ERR_STALE_CLIENTID: 3655 case -NFS4ERR_STALE_CLIENTID:
3656 case -NFS4ERR_STALE_STATEID:
3641 nfs4_schedule_state_recovery(server->nfs_client); 3657 nfs4_schedule_state_recovery(server->nfs_client);
3642 goto out; 3658 goto out;
3659 case -ERESTARTSYS:
3660 /*
3661 * The show must go on: exit, but mark the
3662 * stateid as needing recovery.
3663 */
3664 case -NFS4ERR_ADMIN_REVOKED:
3665 case -NFS4ERR_BAD_STATEID:
3666 case -NFS4ERR_OPENMODE:
3667 nfs4_state_mark_reclaim_nograce(server->nfs_client, state);
3668 err = 0;
3669 goto out;
3670 case -ENOMEM:
3671 case -NFS4ERR_DENIED:
3672 /* kill_proc(fl->fl_pid, SIGLOST, 1); */
3673 err = 0;
3674 goto out;
3643 case -NFS4ERR_DELAY: 3675 case -NFS4ERR_DELAY:
3644 break; 3676 break;
3645 } 3677 }