Commit 4bdbba54 authored by Jeff Layton's avatar Jeff Layton Committed by Chuck Lever

nfsd: don't free files unconditionally in __nfsd_file_cache_purge

nfsd_file_cache_purge is called when the server is shutting down, in
which case, tearing things down is generally fine, but it also gets
called when the exports cache is flushed.

Instead of walking the cache and freeing everything unconditionally,
handle it the same as when we have a notification of conflicting access.

Fixes: ac3a2585 ("nfsd: rework refcounting in filecache")
Reported-by: default avatarRuben Vestergaard <rubenv@drcmr.dk>
Reported-by: default avatarTorkil Svensgaard <torkil@drcmr.dk>
Reported-by: default avatarShachar Kagan <skagan@nvidia.com>
Signed-off-by: default avatarJeff Layton <jlayton@kernel.org>
Tested-by: default avatarShachar Kagan <skagan@nvidia.com>
Signed-off-by: default avatarChuck Lever <chuck.lever@oracle.com>
parent 7c24fa22
...@@ -661,6 +661,39 @@ static struct shrinker nfsd_file_shrinker = { ...@@ -661,6 +661,39 @@ static struct shrinker nfsd_file_shrinker = {
.seeks = 1, .seeks = 1,
}; };
/**
* nfsd_file_cond_queue - conditionally unhash and queue a nfsd_file
* @nf: nfsd_file to attempt to queue
* @dispose: private list to queue successfully-put objects
*
* Unhash an nfsd_file, try to get a reference to it, and then put that
* reference. If it's the last reference, queue it to the dispose list.
*/
static void
nfsd_file_cond_queue(struct nfsd_file *nf, struct list_head *dispose)
__must_hold(RCU)
{
int decrement = 1;
/* If we raced with someone else unhashing, ignore it */
if (!nfsd_file_unhash(nf))
return;
/* If we can't get a reference, ignore it */
if (!nfsd_file_get(nf))
return;
/* Extra decrement if we remove from the LRU */
if (nfsd_file_lru_remove(nf))
++decrement;
/* If refcount goes to 0, then put on the dispose list */
if (refcount_sub_and_test(decrement, &nf->nf_ref)) {
list_add(&nf->nf_lru, dispose);
trace_nfsd_file_closing(nf);
}
}
/** /**
* nfsd_file_queue_for_close: try to close out any open nfsd_files for an inode * nfsd_file_queue_for_close: try to close out any open nfsd_files for an inode
* @inode: inode on which to close out nfsd_files * @inode: inode on which to close out nfsd_files
...@@ -688,30 +721,11 @@ nfsd_file_queue_for_close(struct inode *inode, struct list_head *dispose) ...@@ -688,30 +721,11 @@ nfsd_file_queue_for_close(struct inode *inode, struct list_head *dispose)
rcu_read_lock(); rcu_read_lock();
do { do {
int decrement = 1;
nf = rhashtable_lookup(&nfsd_file_rhash_tbl, &key, nf = rhashtable_lookup(&nfsd_file_rhash_tbl, &key,
nfsd_file_rhash_params); nfsd_file_rhash_params);
if (!nf) if (!nf)
break; break;
nfsd_file_cond_queue(nf, dispose);
/* If we raced with someone else unhashing, ignore it */
if (!nfsd_file_unhash(nf))
continue;
/* If we can't get a reference, ignore it */
if (!nfsd_file_get(nf))
continue;
/* Extra decrement if we remove from the LRU */
if (nfsd_file_lru_remove(nf))
++decrement;
/* If refcount goes to 0, then put on the dispose list */
if (refcount_sub_and_test(decrement, &nf->nf_ref)) {
list_add(&nf->nf_lru, dispose);
trace_nfsd_file_closing(nf);
}
} while (1); } while (1);
rcu_read_unlock(); rcu_read_unlock();
} }
...@@ -928,11 +942,8 @@ __nfsd_file_cache_purge(struct net *net) ...@@ -928,11 +942,8 @@ __nfsd_file_cache_purge(struct net *net)
nf = rhashtable_walk_next(&iter); nf = rhashtable_walk_next(&iter);
while (!IS_ERR_OR_NULL(nf)) { while (!IS_ERR_OR_NULL(nf)) {
if (!net || nf->nf_net == net) { if (!net || nf->nf_net == net)
nfsd_file_unhash(nf); nfsd_file_cond_queue(nf, &dispose);
nfsd_file_lru_remove(nf);
list_add(&nf->nf_lru, &dispose);
}
nf = rhashtable_walk_next(&iter); nf = rhashtable_walk_next(&iter);
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment