diff options
author | J. Bruce Fields <bfields@redhat.com> | 2011-12-22 18:22:49 -0700 |
---|---|---|
committer | Willy Tarreau <w@1wt.eu> | 2012-02-11 15:38:28 +0100 |
commit | b57fdc838c5af0c587e36b03050ea4c98344b8ac (patch) | |
tree | a18d7271f9fa16e0c294096e90daf6ad2b788e79 /net/sunrpc | |
parent | 2e9633af99b28c996ec0d1b53b7503a4dade6d8e (diff) | |
download | lwn-b57fdc838c5af0c587e36b03050ea4c98344b8ac.tar.gz lwn-b57fdc838c5af0c587e36b03050ea4c98344b8ac.zip |
svcrpc: fix double-free on shutdown of nfsd after changing pool mode
commit 61c8504c428edcebf23b97775a129c5b393a302b upstream.
The pool_to and to_pool fields of the global svc_pool_map are freed on
shutdown, but are initialized in nfsd startup only in the
SVC_POOL_PERCPU and SVC_POOL_PERNODE cases.
They *are* initialized to zero on kernel startup. So as long as you use
only SVC_POOL_GLOBAL (the default), this will never be a problem.
You're also OK if you only ever use SVC_POOL_PERCPU or SVC_POOL_PERNODE.
However, the following sequence events leads to a double-free:
1. set SVC_POOL_PERCPU or SVC_POOL_PERNODE
2. start nfsd: both fields are initialized.
3. shutdown nfsd: both fields are freed.
4. set SVC_POOL_GLOBAL
5. start nfsd: the fields are left untouched.
6. shutdown nfsd: now we try to free them again.
Step 4 is actually unnecessary, since (for some bizarre reason), nfsd
automatically resets the pool mode to SVC_POOL_GLOBAL on shutdown.
Signed-off-by: J. Bruce Fields <bfields@redhat.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
Signed-off-by: Willy Tarreau <w@1wt.eu>
Diffstat (limited to 'net/sunrpc')
-rw-r--r-- | net/sunrpc/svc.c | 3 |
1 files changed, 3 insertions, 0 deletions
diff --git a/net/sunrpc/svc.c b/net/sunrpc/svc.c index 5a32cb7c4bb4..8a41977bc3d4 100644 --- a/net/sunrpc/svc.c +++ b/net/sunrpc/svc.c @@ -163,6 +163,7 @@ svc_pool_map_alloc_arrays(struct svc_pool_map *m, unsigned int maxpools) fail_free: kfree(m->to_pool); + m->to_pool = NULL; fail: return -ENOMEM; } @@ -283,7 +284,9 @@ svc_pool_map_put(void) if (!--m->count) { m->mode = SVC_POOL_DEFAULT; kfree(m->to_pool); + m->to_pool = NULL; kfree(m->pool_to); + m->pool_to = NULL; m->npools = 0; } |