diff options
| -rw-r--r-- | fs/nfsd/nfs4state.c | 27 | ||||
| -rw-r--r-- | fs/nfsd/nfs4xdr.c | 3 | ||||
| -rw-r--r-- | fs/nfsd/state.h | 1 |
3 files changed, 27 insertions, 4 deletions
diff --git a/fs/nfsd/nfs4state.c b/fs/nfsd/nfs4state.c index 98aa7e8827b9..cc0e9117dd16 100644 --- a/fs/nfsd/nfs4state.c +++ b/fs/nfsd/nfs4state.c | |||
| @@ -701,6 +701,22 @@ free_client(struct nfs4_client *clp) | |||
| 701 | kfree(clp); | 701 | kfree(clp); |
| 702 | } | 702 | } |
| 703 | 703 | ||
| 704 | void | ||
| 705 | release_session_client(struct nfsd4_session *session) | ||
| 706 | { | ||
| 707 | struct nfs4_client *clp = session->se_client; | ||
| 708 | |||
| 709 | if (!atomic_dec_and_lock(&clp->cl_refcount, &client_lock)) | ||
| 710 | return; | ||
| 711 | if (is_client_expired(clp)) { | ||
| 712 | free_client(clp); | ||
| 713 | session->se_client = NULL; | ||
| 714 | } else | ||
| 715 | renew_client_locked(clp); | ||
| 716 | spin_unlock(&client_lock); | ||
| 717 | nfsd4_put_session(session); | ||
| 718 | } | ||
| 719 | |||
| 704 | /* must be called under the client_lock */ | 720 | /* must be called under the client_lock */ |
| 705 | static inline void | 721 | static inline void |
| 706 | unhash_client_locked(struct nfs4_client *clp) | 722 | unhash_client_locked(struct nfs4_client *clp) |
| @@ -1476,8 +1492,7 @@ out: | |||
| 1476 | /* Hold a session reference until done processing the compound. */ | 1492 | /* Hold a session reference until done processing the compound. */ |
| 1477 | if (cstate->session) { | 1493 | if (cstate->session) { |
| 1478 | nfsd4_get_session(cstate->session); | 1494 | nfsd4_get_session(cstate->session); |
| 1479 | /* Renew the clientid on success and on replay */ | 1495 | atomic_inc(&session->se_client->cl_refcount); |
| 1480 | renew_client_locked(session->se_client); | ||
| 1481 | } | 1496 | } |
| 1482 | spin_unlock(&client_lock); | 1497 | spin_unlock(&client_lock); |
| 1483 | dprintk("%s: return %d\n", __func__, ntohl(status)); | 1498 | dprintk("%s: return %d\n", __func__, ntohl(status)); |
| @@ -2598,7 +2613,13 @@ nfs4_laundromat(void) | |||
| 2598 | clientid_val = t; | 2613 | clientid_val = t; |
| 2599 | break; | 2614 | break; |
| 2600 | } | 2615 | } |
| 2601 | list_move(&clp->cl_lru, &reaplist); | 2616 | if (atomic_read(&clp->cl_refcount)) { |
| 2617 | dprintk("NFSD: client in use (clientid %08x)\n", | ||
| 2618 | clp->cl_clientid.cl_id); | ||
| 2619 | continue; | ||
| 2620 | } | ||
| 2621 | unhash_client_locked(clp); | ||
| 2622 | list_add(&clp->cl_lru, &reaplist); | ||
| 2602 | } | 2623 | } |
| 2603 | spin_unlock(&client_lock); | 2624 | spin_unlock(&client_lock); |
| 2604 | list_for_each_safe(pos, next, &reaplist) { | 2625 | list_for_each_safe(pos, next, &reaplist) { |
diff --git a/fs/nfsd/nfs4xdr.c b/fs/nfsd/nfs4xdr.c index 5c2de471329a..126d0caabb3c 100644 --- a/fs/nfsd/nfs4xdr.c +++ b/fs/nfsd/nfs4xdr.c | |||
| @@ -3313,7 +3313,8 @@ nfs4svc_encode_compoundres(struct svc_rqst *rqstp, __be32 *p, struct nfsd4_compo | |||
| 3313 | dprintk("%s: SET SLOT STATE TO AVAILABLE\n", __func__); | 3313 | dprintk("%s: SET SLOT STATE TO AVAILABLE\n", __func__); |
| 3314 | cs->slot->sl_inuse = false; | 3314 | cs->slot->sl_inuse = false; |
| 3315 | } | 3315 | } |
| 3316 | nfsd4_put_session(cs->session); | 3316 | /* Renew the clientid on success and on replay */ |
| 3317 | release_session_client(cs->session); | ||
| 3317 | } | 3318 | } |
| 3318 | return 1; | 3319 | return 1; |
| 3319 | } | 3320 | } |
diff --git a/fs/nfsd/state.h b/fs/nfsd/state.h index cfd743ea4b79..006c84230c7c 100644 --- a/fs/nfsd/state.h +++ b/fs/nfsd/state.h | |||
| @@ -420,6 +420,7 @@ extern int nfs4_has_reclaimed_state(const char *name, bool use_exchange_id); | |||
| 420 | extern void nfsd4_recdir_purge_old(void); | 420 | extern void nfsd4_recdir_purge_old(void); |
| 421 | extern int nfsd4_create_clid_dir(struct nfs4_client *clp); | 421 | extern int nfsd4_create_clid_dir(struct nfs4_client *clp); |
| 422 | extern void nfsd4_remove_clid_dir(struct nfs4_client *clp); | 422 | extern void nfsd4_remove_clid_dir(struct nfs4_client *clp); |
| 423 | extern void release_session_client(struct nfsd4_session *); | ||
| 423 | 424 | ||
| 424 | static inline void | 425 | static inline void |
| 425 | nfs4_put_stateowner(struct nfs4_stateowner *so) | 426 | nfs4_put_stateowner(struct nfs4_stateowner *so) |
