aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/nsproxy.c
diff options
context:
space:
mode:
authorSerge E. Hallyn <serue@us.ibm.com>2007-01-29 16:19:40 -0500
committerLinus Torvalds <torvalds@woody.linux-foundation.org>2007-01-30 11:26:44 -0500
commit7a238fcba0629b6f2edbcd37458bae56fcf36be5 (patch)
treead556f0ec00637df5b4c4a2063c6b3325666d2f1 /kernel/nsproxy.c
parentc0d4d573feed199b16094c072e7cb07afb01c598 (diff)
[PATCH] namespaces: fix exit race by splitting exit
Fix exit race by splitting the nsproxy putting into two pieces. First piece reduces the nsproxy refcount. If we dropped the last reference, then it puts the mnt_ns, and returns the nsproxy as a hint to the caller. Else it returns NULL. The second piece of exiting task namespaces sets tsk->nsproxy to NULL, and drops the references to other namespaces and frees the nsproxy only if an nsproxy was passed in. A little awkward and should probably be reworked, but hopefully it fixes the NFS oops. Signed-off-by: Serge E. Hallyn <serue@us.ibm.com> Cc: Herbert Poetzl <herbert@13thfloor.at> Cc: Oleg Nesterov <oleg@tv-sign.ru> Cc: "Eric W. Biederman" <ebiederm@xmission.com> Cc: Cedric Le Goater <clg@fr.ibm.com> Cc: Daniel Hokka Zakrisson <daniel@hozac.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'kernel/nsproxy.c')
-rw-r--r--kernel/nsproxy.c16
1 files changed, 15 insertions, 1 deletions
diff --git a/kernel/nsproxy.c b/kernel/nsproxy.c
index f5b9ee6f6bbb..7b05bce75cde 100644
--- a/kernel/nsproxy.c
+++ b/kernel/nsproxy.c
@@ -117,7 +117,7 @@ int copy_namespaces(int flags, struct task_struct *tsk)
117 goto out_pid; 117 goto out_pid;
118 118
119out: 119out:
120 put_nsproxy(old_ns); 120 put_and_finalize_nsproxy(old_ns);
121 return err; 121 return err;
122 122
123out_pid: 123out_pid:
@@ -135,6 +135,20 @@ out_ns:
135 goto out; 135 goto out;
136} 136}
137 137
138struct nsproxy *put_nsproxy(struct nsproxy *ns)
139{
140 if (ns) {
141 if (atomic_dec_and_test(&ns->count)) {
142 if (ns->mnt_ns) {
143 put_mnt_ns(ns->mnt_ns);
144 ns->mnt_ns = NULL;
145 }
146 return ns;
147 }
148 }
149 return NULL;
150}
151
138void free_nsproxy(struct nsproxy *ns) 152void free_nsproxy(struct nsproxy *ns)
139{ 153{
140 if (ns->mnt_ns) 154 if (ns->mnt_ns)