diff options
author | Trond Myklebust <Trond.Myklebust@netapp.com> | 2009-06-17 16:22:59 -0400 |
---|---|---|
committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2009-06-17 16:22:59 -0400 |
commit | 965b5d679146c9f69bc0325388bb9ed357863c4f (patch) | |
tree | d70fe13004404f1d3a203c8b9b90aa5f61150115 /fs/nfs/nfs4proc.c | |
parent | d5122201a7f90b2aa73092f158b84d1d74f1134d (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.c | 44 |
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); |
830 | out: | ||
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 | } |