aboutsummaryrefslogtreecommitdiffstats
path: root/fs/nfs/callback_proc.c
diff options
context:
space:
mode:
authorTrond Myklebust <Trond.Myklebust@netapp.com>2008-01-26 01:06:40 -0500
committerTrond Myklebust <Trond.Myklebust@netapp.com>2008-01-30 02:06:12 -0500
commit3fbd67ad61f6d5a09ea717b56c50bc5c3d8042a8 (patch)
treeffac0a032211517e9d0bc9ff21a09f8c21fdf1c3 /fs/nfs/callback_proc.c
parent57bfa89171e50cddf51a4f62c90e47c6259857b4 (diff)
NFSv4: Iterate through all nfs_clients when the server recalls a delegation
The same delegation may have been handed out to more than one nfs_client. Ensure that if a recall occurs, we return all instances. Signed-off-by: Trond Myklebust <Trond.Myklebust@netapp.com>
Diffstat (limited to 'fs/nfs/callback_proc.c')
-rw-r--r--fs/nfs/callback_proc.c39
1 files changed, 22 insertions, 17 deletions
diff --git a/fs/nfs/callback_proc.c b/fs/nfs/callback_proc.c
index e89a9007c91c..15f7785048d3 100644
--- a/fs/nfs/callback_proc.c
+++ b/fs/nfs/callback_proc.c
@@ -75,23 +75,28 @@ __be32 nfs4_callback_recall(struct cb_recallargs *args, void *dummy)
75 dprintk("NFS: RECALL callback request from %s\n", 75 dprintk("NFS: RECALL callback request from %s\n",
76 rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR)); 76 rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR));
77 77
78 inode = nfs_delegation_find_inode(clp, &args->fh); 78 do {
79 if (inode == NULL) 79 struct nfs_client *prev = clp;
80 goto out_putclient; 80
81 /* Set up a helper thread to actually return the delegation */ 81 inode = nfs_delegation_find_inode(clp, &args->fh);
82 switch(nfs_async_inode_return_delegation(inode, &args->stateid)) { 82 if (inode != NULL) {
83 case 0: 83 /* Set up a helper thread to actually return the delegation */
84 res = 0; 84 switch(nfs_async_inode_return_delegation(inode, &args->stateid)) {
85 break; 85 case 0:
86 case -ENOENT: 86 res = 0;
87 res = htonl(NFS4ERR_BAD_STATEID); 87 break;
88 break; 88 case -ENOENT:
89 default: 89 if (res != 0)
90 res = htonl(NFS4ERR_RESOURCE); 90 res = htonl(NFS4ERR_BAD_STATEID);
91 } 91 break;
92 iput(inode); 92 default:
93out_putclient: 93 res = htonl(NFS4ERR_RESOURCE);
94 nfs_put_client(clp); 94 }
95 iput(inode);
96 }
97 clp = nfs_find_client_next(prev);
98 nfs_put_client(prev);
99 } while (clp != NULL);
95out: 100out:
96 dprintk("%s: exit with status = %d\n", __FUNCTION__, ntohl(res)); 101 dprintk("%s: exit with status = %d\n", __FUNCTION__, ntohl(res));
97 return res; 102 return res;