Commit 8c62d127 authored by NeilBrown's avatar NeilBrown Committed by Chuck Lever

SUNRPC/NFSD: clean up get/put functions.

svc_destroy() is poorly named - it doesn't necessarily destroy the svc,
it might just reduce the ref count.
nfsd_destroy() is poorly named for the same reason.

This patch:
 - removes the refcount functionality from svc_destroy(), moving it to
   a new svc_put().  Almost all previous callers of svc_destroy() now
   call svc_put().
 - renames nfsd_destroy() to nfsd_put() and improves the code, using
   the new svc_destroy() rather than svc_put()
 - removes a few comments that explain the important for balanced
   get/put calls.  This should be obvious.

The only non-trivial part of this is that svc_destroy() would call
svc_sock_update() on a non-final decrement.  It can no longer do that,
and svc_put() isn't really a good place of it.  This call is now made
from svc_exit_thread() which seems like a good place.  This makes the
call *before* sv_nrthreads is decremented rather than after.  This
is not particularly important as the call just sets a flag which
causes sv_nrthreads set be checked later.  A subsequent patch will
improve the ordering.
Signed-off-by: default avatarNeilBrown <neilb@suse.de>
Signed-off-by: default avatarChuck Lever <chuck.lever@oracle.com>
parent df5e49c8
...@@ -431,10 +431,6 @@ static struct svc_serv *lockd_create_svc(void) ...@@ -431,10 +431,6 @@ static struct svc_serv *lockd_create_svc(void)
* Check whether we're already up and running. * Check whether we're already up and running.
*/ */
if (nlmsvc_rqst) if (nlmsvc_rqst)
/*
* Note: increase service usage, because later in case of error
* svc_destroy() will be called.
*/
return svc_get(nlmsvc_rqst->rq_server); return svc_get(nlmsvc_rqst->rq_server);
/* /*
...@@ -495,7 +491,7 @@ int lockd_up(struct net *net, const struct cred *cred) ...@@ -495,7 +491,7 @@ int lockd_up(struct net *net, const struct cred *cred)
* so we exit through here on both success and failure. * so we exit through here on both success and failure.
*/ */
err_put: err_put:
svc_destroy(serv); svc_put(serv);
err_create: err_create:
mutex_unlock(&nlmsvc_mutex); mutex_unlock(&nlmsvc_mutex);
return error; return error;
......
...@@ -267,10 +267,6 @@ static struct svc_serv *nfs_callback_create_svc(int minorversion) ...@@ -267,10 +267,6 @@ static struct svc_serv *nfs_callback_create_svc(int minorversion)
* Check whether we're already up and running. * Check whether we're already up and running.
*/ */
if (cb_info->serv) if (cb_info->serv)
/*
* Note: increase service usage, because later in case of error
* svc_destroy() will be called.
*/
return svc_get(cb_info->serv); return svc_get(cb_info->serv);
switch (minorversion) { switch (minorversion) {
...@@ -333,16 +329,10 @@ int nfs_callback_up(u32 minorversion, struct rpc_xprt *xprt) ...@@ -333,16 +329,10 @@ int nfs_callback_up(u32 minorversion, struct rpc_xprt *xprt)
goto err_start; goto err_start;
cb_info->users++; cb_info->users++;
/*
* svc_create creates the svc_serv with sv_nrthreads == 1, and then
* svc_prepare_thread increments that. So we need to call svc_destroy
* on both success and failure so that the refcount is 1 when the
* thread exits.
*/
err_net: err_net:
if (!cb_info->users) if (!cb_info->users)
cb_info->serv = NULL; cb_info->serv = NULL;
svc_destroy(serv); svc_put(serv);
err_create: err_create:
mutex_unlock(&nfs_callback_mutex); mutex_unlock(&nfs_callback_mutex);
return ret; return ret;
...@@ -368,7 +358,7 @@ void nfs_callback_down(int minorversion, struct net *net) ...@@ -368,7 +358,7 @@ void nfs_callback_down(int minorversion, struct net *net)
if (cb_info->users == 0) { if (cb_info->users == 0) {
svc_get(serv); svc_get(serv);
serv->sv_ops->svo_setup(serv, NULL, 0); serv->sv_ops->svo_setup(serv, NULL, 0);
svc_destroy(serv); svc_put(serv);
dprintk("nfs_callback_down: service destroyed\n"); dprintk("nfs_callback_down: service destroyed\n");
cb_info->serv = NULL; cb_info->serv = NULL;
} }
......
...@@ -743,7 +743,7 @@ static ssize_t __write_ports_addfd(char *buf, struct net *net, const struct cred ...@@ -743,7 +743,7 @@ static ssize_t __write_ports_addfd(char *buf, struct net *net, const struct cred
err = svc_addsock(nn->nfsd_serv, fd, buf, SIMPLE_TRANSACTION_LIMIT, cred); err = svc_addsock(nn->nfsd_serv, fd, buf, SIMPLE_TRANSACTION_LIMIT, cred);
if (err < 0 && list_empty(&nn->nfsd_serv->sv_permsocks)) { if (err < 0 && list_empty(&nn->nfsd_serv->sv_permsocks)) {
nfsd_destroy(net); nfsd_put(net);
return err; return err;
} }
...@@ -796,7 +796,7 @@ static ssize_t __write_ports_addxprt(char *buf, struct net *net, const struct cr ...@@ -796,7 +796,7 @@ static ssize_t __write_ports_addxprt(char *buf, struct net *net, const struct cr
if (!list_empty(&nn->nfsd_serv->sv_permsocks)) if (!list_empty(&nn->nfsd_serv->sv_permsocks))
nn->nfsd_serv->sv_nrthreads--; nn->nfsd_serv->sv_nrthreads--;
else else
nfsd_destroy(net); nfsd_put(net);
return err; return err;
} }
......
...@@ -97,7 +97,7 @@ int nfsd_pool_stats_open(struct inode *, struct file *); ...@@ -97,7 +97,7 @@ int nfsd_pool_stats_open(struct inode *, struct file *);
int nfsd_pool_stats_release(struct inode *, struct file *); int nfsd_pool_stats_release(struct inode *, struct file *);
void nfsd_shutdown_threads(struct net *net); void nfsd_shutdown_threads(struct net *net);
void nfsd_destroy(struct net *net); void nfsd_put(struct net *net);
bool i_am_nfsd(void); bool i_am_nfsd(void);
......
...@@ -623,7 +623,7 @@ void nfsd_shutdown_threads(struct net *net) ...@@ -623,7 +623,7 @@ void nfsd_shutdown_threads(struct net *net)
svc_get(serv); svc_get(serv);
/* Kill outstanding nfsd threads */ /* Kill outstanding nfsd threads */
serv->sv_ops->svo_setup(serv, NULL, 0); serv->sv_ops->svo_setup(serv, NULL, 0);
nfsd_destroy(net); nfsd_put(net);
mutex_unlock(&nfsd_mutex); mutex_unlock(&nfsd_mutex);
/* Wait for shutdown of nfsd_serv to complete */ /* Wait for shutdown of nfsd_serv to complete */
wait_for_completion(&nn->nfsd_shutdown_complete); wait_for_completion(&nn->nfsd_shutdown_complete);
...@@ -656,7 +656,10 @@ int nfsd_create_serv(struct net *net) ...@@ -656,7 +656,10 @@ int nfsd_create_serv(struct net *net)
nn->nfsd_serv->sv_maxconn = nn->max_connections; nn->nfsd_serv->sv_maxconn = nn->max_connections;
error = svc_bind(nn->nfsd_serv, net); error = svc_bind(nn->nfsd_serv, net);
if (error < 0) { if (error < 0) {
svc_destroy(nn->nfsd_serv); /* NOT nfsd_put() as notifiers (see below) haven't
* been set up yet.
*/
svc_put(nn->nfsd_serv);
nfsd_complete_shutdown(net); nfsd_complete_shutdown(net);
return error; return error;
} }
...@@ -697,16 +700,16 @@ int nfsd_get_nrthreads(int n, int *nthreads, struct net *net) ...@@ -697,16 +700,16 @@ int nfsd_get_nrthreads(int n, int *nthreads, struct net *net)
return 0; return 0;
} }
void nfsd_destroy(struct net *net) void nfsd_put(struct net *net)
{ {
struct nfsd_net *nn = net_generic(net, nfsd_net_id); struct nfsd_net *nn = net_generic(net, nfsd_net_id);
int destroy = (nn->nfsd_serv->sv_nrthreads == 1);
if (destroy) nn->nfsd_serv->sv_nrthreads -= 1;
if (nn->nfsd_serv->sv_nrthreads == 0) {
svc_shutdown_net(nn->nfsd_serv, net); svc_shutdown_net(nn->nfsd_serv, net);
svc_destroy(nn->nfsd_serv); svc_destroy(nn->nfsd_serv);
if (destroy)
nfsd_complete_shutdown(net); nfsd_complete_shutdown(net);
}
} }
int nfsd_set_nrthreads(int n, int *nthreads, struct net *net) int nfsd_set_nrthreads(int n, int *nthreads, struct net *net)
...@@ -758,7 +761,7 @@ int nfsd_set_nrthreads(int n, int *nthreads, struct net *net) ...@@ -758,7 +761,7 @@ int nfsd_set_nrthreads(int n, int *nthreads, struct net *net)
if (err) if (err)
break; break;
} }
nfsd_destroy(net); nfsd_put(net);
return err; return err;
} }
...@@ -795,7 +798,7 @@ nfsd_svc(int nrservs, struct net *net, const struct cred *cred) ...@@ -795,7 +798,7 @@ nfsd_svc(int nrservs, struct net *net, const struct cred *cred)
error = nfsd_startup_net(net, cred); error = nfsd_startup_net(net, cred);
if (error) if (error)
goto out_destroy; goto out_put;
error = nn->nfsd_serv->sv_ops->svo_setup(nn->nfsd_serv, error = nn->nfsd_serv->sv_ops->svo_setup(nn->nfsd_serv,
NULL, nrservs); NULL, nrservs);
if (error) if (error)
...@@ -808,8 +811,8 @@ nfsd_svc(int nrservs, struct net *net, const struct cred *cred) ...@@ -808,8 +811,8 @@ nfsd_svc(int nrservs, struct net *net, const struct cred *cred)
out_shutdown: out_shutdown:
if (error < 0 && !nfsd_up_before) if (error < 0 && !nfsd_up_before)
nfsd_shutdown_net(net); nfsd_shutdown_net(net);
out_destroy: out_put:
nfsd_destroy(net); /* Release server */ nfsd_put(net);
out: out:
mutex_unlock(&nfsd_mutex); mutex_unlock(&nfsd_mutex);
return error; return error;
...@@ -982,7 +985,7 @@ nfsd(void *vrqstp) ...@@ -982,7 +985,7 @@ nfsd(void *vrqstp)
/* Release the thread */ /* Release the thread */
svc_exit_thread(rqstp); svc_exit_thread(rqstp);
nfsd_destroy(net); nfsd_put(net);
/* Release module */ /* Release module */
mutex_unlock(&nfsd_mutex); mutex_unlock(&nfsd_mutex);
...@@ -1109,8 +1112,7 @@ int nfsd_pool_stats_release(struct inode *inode, struct file *file) ...@@ -1109,8 +1112,7 @@ int nfsd_pool_stats_release(struct inode *inode, struct file *file)
struct net *net = inode->i_sb->s_fs_info; struct net *net = inode->i_sb->s_fs_info;
mutex_lock(&nfsd_mutex); mutex_lock(&nfsd_mutex);
/* this function really, really should have been called svc_put() */ nfsd_put(net);
nfsd_destroy(net);
mutex_unlock(&nfsd_mutex); mutex_unlock(&nfsd_mutex);
return ret; return ret;
} }
...@@ -114,8 +114,13 @@ struct svc_serv { ...@@ -114,8 +114,13 @@ struct svc_serv {
#endif /* CONFIG_SUNRPC_BACKCHANNEL */ #endif /* CONFIG_SUNRPC_BACKCHANNEL */
}; };
/* /**
* We use sv_nrthreads as a reference count. svc_destroy() drops * svc_get() - increment reference count on a SUNRPC serv
* @serv: the svc_serv to have count incremented
*
* Returns: the svc_serv that was passed in.
*
* We use sv_nrthreads as a reference count. svc_put() drops
* this refcount, so we need to bump it up around operations that * this refcount, so we need to bump it up around operations that
* change the number of threads. Horrible, but there it is. * change the number of threads. Horrible, but there it is.
* Should be called with the "service mutex" held. * Should be called with the "service mutex" held.
...@@ -126,6 +131,22 @@ static inline struct svc_serv *svc_get(struct svc_serv *serv) ...@@ -126,6 +131,22 @@ static inline struct svc_serv *svc_get(struct svc_serv *serv)
return serv; return serv;
} }
void svc_destroy(struct svc_serv *serv);
/**
* svc_put - decrement reference count on a SUNRPC serv
* @serv: the svc_serv to have count decremented
*
* When the reference count reaches zero, svc_destroy()
* is called to clean up and free the serv.
*/
static inline void svc_put(struct svc_serv *serv)
{
serv->sv_nrthreads -= 1;
if (serv->sv_nrthreads == 0)
svc_destroy(serv);
}
/* /*
* Maximum payload size supported by a kernel RPC server. * Maximum payload size supported by a kernel RPC server.
* This is use to determine the max number of pages nfsd is * This is use to determine the max number of pages nfsd is
...@@ -515,7 +536,6 @@ struct svc_serv * svc_create_pooled(struct svc_program *, unsigned int, ...@@ -515,7 +536,6 @@ struct svc_serv * svc_create_pooled(struct svc_program *, unsigned int,
int svc_set_num_threads(struct svc_serv *, struct svc_pool *, int); int svc_set_num_threads(struct svc_serv *, struct svc_pool *, int);
int svc_set_num_threads_sync(struct svc_serv *, struct svc_pool *, int); int svc_set_num_threads_sync(struct svc_serv *, struct svc_pool *, int);
int svc_pool_stats_open(struct svc_serv *serv, struct file *file); int svc_pool_stats_open(struct svc_serv *serv, struct file *file);
void svc_destroy(struct svc_serv *);
void svc_shutdown_net(struct svc_serv *, struct net *); void svc_shutdown_net(struct svc_serv *, struct net *);
int svc_process(struct svc_rqst *); int svc_process(struct svc_rqst *);
int bc_svc_process(struct svc_serv *, struct rpc_rqst *, int bc_svc_process(struct svc_serv *, struct rpc_rqst *,
......
...@@ -528,17 +528,7 @@ EXPORT_SYMBOL_GPL(svc_shutdown_net); ...@@ -528,17 +528,7 @@ EXPORT_SYMBOL_GPL(svc_shutdown_net);
void void
svc_destroy(struct svc_serv *serv) svc_destroy(struct svc_serv *serv)
{ {
dprintk("svc: svc_destroy(%s, %d)\n", dprintk("svc: svc_destroy(%s)\n", serv->sv_program->pg_name);
serv->sv_program->pg_name,
serv->sv_nrthreads);
if (serv->sv_nrthreads) {
if (--(serv->sv_nrthreads) != 0) {
svc_sock_update_bufs(serv);
return;
}
} else
printk("svc_destroy: no threads for serv=%p!\n", serv);
del_timer_sync(&serv->sv_temptimer); del_timer_sync(&serv->sv_temptimer);
...@@ -892,9 +882,10 @@ svc_exit_thread(struct svc_rqst *rqstp) ...@@ -892,9 +882,10 @@ svc_exit_thread(struct svc_rqst *rqstp)
svc_rqst_free(rqstp); svc_rqst_free(rqstp);
/* Release the server */ if (!serv)
if (serv) return;
svc_destroy(serv); svc_sock_update_bufs(serv);
svc_destroy(serv);
} }
EXPORT_SYMBOL_GPL(svc_exit_thread); EXPORT_SYMBOL_GPL(svc_exit_thread);
......
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