diff options
Diffstat (limited to 'fs')
-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) |