/* * Returns a pointer to a held 'struct nfs_client' that matches the server's * address, major version number, and session ID. It is the caller's * responsibility to release the returned reference. * * Returns NULL if there are no connections with sessions, or if no session * matches the one of interest. */ static struct nfs_client *find_client_with_session( const struct sockaddr *addr, u32 nfsversion, struct nfs4_sessionid *sessionid) { struct nfs_client *clp; clp = nfs_find_client(addr, 4); if (clp == NULL) return NULL; do { struct nfs_client *prev = clp; if (clp->cl_session != NULL) { if (memcmp(clp->cl_session->sess_id.data, sessionid->data, NFS4_MAX_SESSIONID_LEN) == 0) { /* Returns a held reference to clp */ return clp; } } clp = nfs_find_client_next(prev); nfs_put_client(prev); } while (clp != NULL); return NULL; }
__be32 nfs4_callback_recall(struct cb_recallargs *args, void *dummy) { struct nfs_client *clp; struct inode *inode; __be32 res; res = htonl(NFS4ERR_BADHANDLE); clp = nfs_find_client(args->addr, 4); if (clp == NULL) goto out; dprintk("NFS: RECALL callback request from %s\n", rpc_peeraddr2str(clp->cl_rpcclient, RPC_DISPLAY_ADDR)); do { struct nfs_client *prev = clp; inode = nfs_delegation_find_inode(clp, &args->fh); if (inode != NULL) { /* Set up a helper thread to actually return the delegation */ switch (nfs_async_inode_return_delegation(inode, &args->stateid, nfs_validate_delegation_stateid(clp))) { case 0: res = 0; break; case -ENOENT: if (res != 0) res = htonl(NFS4ERR_BAD_STATEID); break; default: res = htonl(NFS4ERR_RESOURCE); } iput(inode); } clp = nfs_find_client_next(prev); nfs_put_client(prev); } while (clp != NULL); out: dprintk("%s: exit with status = %d\n", __func__, ntohl(res)); return res; }