diff options
author | Trond Myklebust <Trond.Myklebust@netapp.com> | 2008-01-26 01:06:40 -0500 |
---|---|---|
committer | Trond Myklebust <Trond.Myklebust@netapp.com> | 2008-01-30 02:06:12 -0500 |
commit | 3fbd67ad61f6d5a09ea717b56c50bc5c3d8042a8 (patch) | |
tree | ffac0a032211517e9d0bc9ff21a09f8c21fdf1c3 /fs/nfs | |
parent | 57bfa89171e50cddf51a4f62c90e47c6259857b4 (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.c | 39 | ||||
-rw-r--r-- | fs/nfs/client.c | 35 | ||||
-rw-r--r-- | fs/nfs/internal.h | 1 |
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: |
93 | out_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); | ||
95 | out: | 100 | out: |
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 | */ | ||
282 | struct 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 | ||
65 | extern void nfs_put_client(struct nfs_client *); | 65 | extern void nfs_put_client(struct nfs_client *); |
66 | extern struct nfs_client *nfs_find_client(const struct sockaddr *, u32); | 66 | extern struct nfs_client *nfs_find_client(const struct sockaddr *, u32); |
67 | extern struct nfs_client *nfs_find_client_next(struct nfs_client *); | ||
67 | extern struct nfs_server *nfs_create_server( | 68 | extern 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 *); |