mirror of
https://github.com/torvalds/linux.git
synced 2024-11-29 07:31:29 +00:00
nfsd: nfsd_file cache entries should be per net namespace
Ensure that we can safely clear out the file cache entries when the nfs server is shut down on a container. Otherwise, the file cache may end up pinning the mounts. Signed-off-by: Trond Myklebust <trond.myklebust@hammerspace.com> Signed-off-by: J. Bruce Fields <bfields@redhat.com>
This commit is contained in:
parent
2b86e3aaf9
commit
5e113224c1
@ -240,7 +240,7 @@ static void expkey_flush(void)
|
|||||||
* destroyed while we're in the middle of flushing.
|
* destroyed while we're in the middle of flushing.
|
||||||
*/
|
*/
|
||||||
mutex_lock(&nfsd_mutex);
|
mutex_lock(&nfsd_mutex);
|
||||||
nfsd_file_cache_purge();
|
nfsd_file_cache_purge(current->nsproxy->net_ns);
|
||||||
mutex_unlock(&nfsd_mutex);
|
mutex_unlock(&nfsd_mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -16,6 +16,7 @@
|
|||||||
#include "vfs.h"
|
#include "vfs.h"
|
||||||
#include "nfsd.h"
|
#include "nfsd.h"
|
||||||
#include "nfsfh.h"
|
#include "nfsfh.h"
|
||||||
|
#include "netns.h"
|
||||||
#include "filecache.h"
|
#include "filecache.h"
|
||||||
#include "trace.h"
|
#include "trace.h"
|
||||||
|
|
||||||
@ -167,7 +168,8 @@ nfsd_file_mark_find_or_create(struct nfsd_file *nf)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static struct nfsd_file *
|
static struct nfsd_file *
|
||||||
nfsd_file_alloc(struct inode *inode, unsigned int may, unsigned int hashval)
|
nfsd_file_alloc(struct inode *inode, unsigned int may, unsigned int hashval,
|
||||||
|
struct net *net)
|
||||||
{
|
{
|
||||||
struct nfsd_file *nf;
|
struct nfsd_file *nf;
|
||||||
|
|
||||||
@ -177,6 +179,7 @@ nfsd_file_alloc(struct inode *inode, unsigned int may, unsigned int hashval)
|
|||||||
INIT_LIST_HEAD(&nf->nf_lru);
|
INIT_LIST_HEAD(&nf->nf_lru);
|
||||||
nf->nf_file = NULL;
|
nf->nf_file = NULL;
|
||||||
nf->nf_cred = get_current_cred();
|
nf->nf_cred = get_current_cred();
|
||||||
|
nf->nf_net = net;
|
||||||
nf->nf_flags = 0;
|
nf->nf_flags = 0;
|
||||||
nf->nf_inode = inode;
|
nf->nf_inode = inode;
|
||||||
nf->nf_hashval = hashval;
|
nf->nf_hashval = hashval;
|
||||||
@ -607,10 +610,11 @@ out_err:
|
|||||||
* Note this can deadlock with nfsd_file_lru_cb.
|
* Note this can deadlock with nfsd_file_lru_cb.
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
nfsd_file_cache_purge(void)
|
nfsd_file_cache_purge(struct net *net)
|
||||||
{
|
{
|
||||||
unsigned int i;
|
unsigned int i;
|
||||||
struct nfsd_file *nf;
|
struct nfsd_file *nf;
|
||||||
|
struct hlist_node *next;
|
||||||
LIST_HEAD(dispose);
|
LIST_HEAD(dispose);
|
||||||
bool del;
|
bool del;
|
||||||
|
|
||||||
@ -618,10 +622,12 @@ nfsd_file_cache_purge(void)
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
for (i = 0; i < NFSD_FILE_HASH_SIZE; i++) {
|
for (i = 0; i < NFSD_FILE_HASH_SIZE; i++) {
|
||||||
spin_lock(&nfsd_file_hashtbl[i].nfb_lock);
|
struct nfsd_fcache_bucket *nfb = &nfsd_file_hashtbl[i];
|
||||||
while(!hlist_empty(&nfsd_file_hashtbl[i].nfb_head)) {
|
|
||||||
nf = hlist_entry(nfsd_file_hashtbl[i].nfb_head.first,
|
spin_lock(&nfb->nfb_lock);
|
||||||
struct nfsd_file, nf_node);
|
hlist_for_each_entry_safe(nf, next, &nfb->nfb_head, nf_node) {
|
||||||
|
if (net && nf->nf_net != net)
|
||||||
|
continue;
|
||||||
del = nfsd_file_unhash_and_release_locked(nf, &dispose);
|
del = nfsd_file_unhash_and_release_locked(nf, &dispose);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -630,7 +636,7 @@ nfsd_file_cache_purge(void)
|
|||||||
*/
|
*/
|
||||||
WARN_ON_ONCE(!del);
|
WARN_ON_ONCE(!del);
|
||||||
}
|
}
|
||||||
spin_unlock(&nfsd_file_hashtbl[i].nfb_lock);
|
spin_unlock(&nfb->nfb_lock);
|
||||||
nfsd_file_dispose_list(&dispose);
|
nfsd_file_dispose_list(&dispose);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -649,7 +655,7 @@ nfsd_file_cache_shutdown(void)
|
|||||||
* calling nfsd_file_cache_purge
|
* calling nfsd_file_cache_purge
|
||||||
*/
|
*/
|
||||||
cancel_delayed_work_sync(&nfsd_filecache_laundrette);
|
cancel_delayed_work_sync(&nfsd_filecache_laundrette);
|
||||||
nfsd_file_cache_purge();
|
nfsd_file_cache_purge(NULL);
|
||||||
list_lru_destroy(&nfsd_file_lru);
|
list_lru_destroy(&nfsd_file_lru);
|
||||||
rcu_barrier();
|
rcu_barrier();
|
||||||
fsnotify_put_group(nfsd_file_fsnotify_group);
|
fsnotify_put_group(nfsd_file_fsnotify_group);
|
||||||
@ -685,7 +691,7 @@ nfsd_match_cred(const struct cred *c1, const struct cred *c2)
|
|||||||
|
|
||||||
static struct nfsd_file *
|
static struct nfsd_file *
|
||||||
nfsd_file_find_locked(struct inode *inode, unsigned int may_flags,
|
nfsd_file_find_locked(struct inode *inode, unsigned int may_flags,
|
||||||
unsigned int hashval)
|
unsigned int hashval, struct net *net)
|
||||||
{
|
{
|
||||||
struct nfsd_file *nf;
|
struct nfsd_file *nf;
|
||||||
unsigned char need = may_flags & NFSD_FILE_MAY_MASK;
|
unsigned char need = may_flags & NFSD_FILE_MAY_MASK;
|
||||||
@ -696,6 +702,8 @@ nfsd_file_find_locked(struct inode *inode, unsigned int may_flags,
|
|||||||
continue;
|
continue;
|
||||||
if (nf->nf_inode != inode)
|
if (nf->nf_inode != inode)
|
||||||
continue;
|
continue;
|
||||||
|
if (nf->nf_net != net)
|
||||||
|
continue;
|
||||||
if (!nfsd_match_cred(nf->nf_cred, current_cred()))
|
if (!nfsd_match_cred(nf->nf_cred, current_cred()))
|
||||||
continue;
|
continue;
|
||||||
if (nfsd_file_get(nf) != NULL)
|
if (nfsd_file_get(nf) != NULL)
|
||||||
@ -738,6 +746,7 @@ nfsd_file_acquire(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
|||||||
unsigned int may_flags, struct nfsd_file **pnf)
|
unsigned int may_flags, struct nfsd_file **pnf)
|
||||||
{
|
{
|
||||||
__be32 status;
|
__be32 status;
|
||||||
|
struct net *net = SVC_NET(rqstp);
|
||||||
struct nfsd_file *nf, *new;
|
struct nfsd_file *nf, *new;
|
||||||
struct inode *inode;
|
struct inode *inode;
|
||||||
unsigned int hashval;
|
unsigned int hashval;
|
||||||
@ -752,12 +761,12 @@ nfsd_file_acquire(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
|||||||
hashval = (unsigned int)hash_long(inode->i_ino, NFSD_FILE_HASH_BITS);
|
hashval = (unsigned int)hash_long(inode->i_ino, NFSD_FILE_HASH_BITS);
|
||||||
retry:
|
retry:
|
||||||
rcu_read_lock();
|
rcu_read_lock();
|
||||||
nf = nfsd_file_find_locked(inode, may_flags, hashval);
|
nf = nfsd_file_find_locked(inode, may_flags, hashval, net);
|
||||||
rcu_read_unlock();
|
rcu_read_unlock();
|
||||||
if (nf)
|
if (nf)
|
||||||
goto wait_for_construction;
|
goto wait_for_construction;
|
||||||
|
|
||||||
new = nfsd_file_alloc(inode, may_flags, hashval);
|
new = nfsd_file_alloc(inode, may_flags, hashval, net);
|
||||||
if (!new) {
|
if (!new) {
|
||||||
trace_nfsd_file_acquire(rqstp, hashval, inode, may_flags,
|
trace_nfsd_file_acquire(rqstp, hashval, inode, may_flags,
|
||||||
NULL, nfserr_jukebox);
|
NULL, nfserr_jukebox);
|
||||||
@ -765,7 +774,7 @@ retry:
|
|||||||
}
|
}
|
||||||
|
|
||||||
spin_lock(&nfsd_file_hashtbl[hashval].nfb_lock);
|
spin_lock(&nfsd_file_hashtbl[hashval].nfb_lock);
|
||||||
nf = nfsd_file_find_locked(inode, may_flags, hashval);
|
nf = nfsd_file_find_locked(inode, may_flags, hashval, net);
|
||||||
if (nf == NULL)
|
if (nf == NULL)
|
||||||
goto open_file;
|
goto open_file;
|
||||||
spin_unlock(&nfsd_file_hashtbl[hashval].nfb_lock);
|
spin_unlock(&nfsd_file_hashtbl[hashval].nfb_lock);
|
||||||
|
@ -34,6 +34,7 @@ struct nfsd_file {
|
|||||||
struct rcu_head nf_rcu;
|
struct rcu_head nf_rcu;
|
||||||
struct file *nf_file;
|
struct file *nf_file;
|
||||||
const struct cred *nf_cred;
|
const struct cred *nf_cred;
|
||||||
|
struct net *nf_net;
|
||||||
#define NFSD_FILE_HASHED (0)
|
#define NFSD_FILE_HASHED (0)
|
||||||
#define NFSD_FILE_PENDING (1)
|
#define NFSD_FILE_PENDING (1)
|
||||||
#define NFSD_FILE_BREAK_READ (2)
|
#define NFSD_FILE_BREAK_READ (2)
|
||||||
@ -48,7 +49,7 @@ struct nfsd_file {
|
|||||||
};
|
};
|
||||||
|
|
||||||
int nfsd_file_cache_init(void);
|
int nfsd_file_cache_init(void);
|
||||||
void nfsd_file_cache_purge(void);
|
void nfsd_file_cache_purge(struct net *);
|
||||||
void nfsd_file_cache_shutdown(void);
|
void nfsd_file_cache_shutdown(void);
|
||||||
void nfsd_file_put(struct nfsd_file *nf);
|
void nfsd_file_put(struct nfsd_file *nf);
|
||||||
struct nfsd_file *nfsd_file_get(struct nfsd_file *nf);
|
struct nfsd_file *nfsd_file_get(struct nfsd_file *nf);
|
||||||
|
@ -387,6 +387,7 @@ static void nfsd_shutdown_net(struct net *net)
|
|||||||
{
|
{
|
||||||
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
|
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
|
||||||
|
|
||||||
|
nfsd_file_cache_purge(net);
|
||||||
nfs4_state_shutdown_net(net);
|
nfs4_state_shutdown_net(net);
|
||||||
if (nn->lockd_up) {
|
if (nn->lockd_up) {
|
||||||
lockd_down(net);
|
lockd_down(net);
|
||||||
|
Loading…
Reference in New Issue
Block a user