aboutsummaryrefslogtreecommitdiffstats
path: root/fs/nfs
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
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')
-rw-r--r--fs/nfs/callback_proc.c39
-rw-r--r--fs/nfs/client.c35
-rw-r--r--fs/nfs/internal.h1
3 files changed, 58 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;
diff --git a/fs/nfs/client.c b/fs/nfs/client.c
index 7a15832369e9..685c43f810c1 100644
--- a/fs/nfs/client.c
+++ b/fs/nfs/client.c
@@ -276,6 +276,41 @@ struct nfs_client *nfs_find_client(const struct sockaddr *addr, u32 nfsversion)
276} 276}
277 277
278/* 278/*
279 * Find a client by IP address and protocol version
280 * - returns NULL if no such client
281 */
282struct nfs_client *nfs_find_client_next(struct nfs_client *clp)
283{
284 struct sockaddr *sap = (struct sockaddr *)&clp->cl_addr;
285 u32 nfsvers = clp->rpc_ops->version;
286
287 spin_lock(&nfs_client_lock);
288 list_for_each_entry_continue(clp, &nfs_client_list, cl_share_link) {
289 struct sockaddr *clap = (struct sockaddr *)&clp->cl_addr;
290
291 /* Don't match clients that failed to initialise properly */
292 if (clp->cl_cons_state != NFS_CS_READY)
293 continue;
294
295 /* Different NFS versions cannot share the same nfs_client */
296 if (clp->rpc_ops->version != nfsvers)
297 continue;
298
299 if (sap->sa_family != clap->sa_family)
300 continue;
301 /* Match only the IP address, not the port number */
302 if (!nfs_sockaddr_match_ipaddr(sap, clap))
303 continue;
304
305 atomic_inc(&clp->cl_count);
306 spin_unlock(&nfs_client_lock);
307 return clp;
308 }
309 spin_unlock(&nfs_client_lock);
310 return NULL;
311}
312
313/*
279 * Find an nfs_client on the list that matches the initialisation data 314 * Find an nfs_client on the list that matches the initialisation data
280 * that is supplied. 315 * that is supplied.
281 */ 316 */
diff --git a/fs/nfs/internal.h b/fs/nfs/internal.h
index a80621199086..0f5619611b8d 100644
--- a/fs/nfs/internal.h
+++ b/fs/nfs/internal.h
@@ -64,6 +64,7 @@ extern struct rpc_program nfs_program;
64 64
65extern void nfs_put_client(struct nfs_client *); 65extern void nfs_put_client(struct nfs_client *);
66extern struct nfs_client *nfs_find_client(const struct sockaddr *, u32); 66extern struct nfs_client *nfs_find_client(const struct sockaddr *, u32);
67extern struct nfs_client *nfs_find_client_next(struct nfs_client *);
67extern struct nfs_server *nfs_create_server( 68extern struct nfs_server *nfs_create_server(
68 const struct nfs_parsed_mount_data *, 69 const struct nfs_parsed_mount_data *,
69 struct nfs_fh *); 70 struct nfs_fh *);