diff options
author | Serge E. Hallyn <serue@us.ibm.com> | 2007-01-29 16:19:40 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-01-30 11:26:44 -0500 |
commit | 7a238fcba0629b6f2edbcd37458bae56fcf36be5 (patch) | |
tree | ad556f0ec00637df5b4c4a2063c6b3325666d2f1 /kernel/exit.c | |
parent | c0d4d573feed199b16094c072e7cb07afb01c598 (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/exit.c')
-rw-r--r-- | kernel/exit.c | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/kernel/exit.c b/kernel/exit.c index 35401720635b..a5bf5329ff97 100644 --- a/kernel/exit.c +++ b/kernel/exit.c | |||
@@ -396,7 +396,7 @@ void daemonize(const char *name, ...) | |||
396 | current->fs = fs; | 396 | current->fs = fs; |
397 | atomic_inc(&fs->count); | 397 | atomic_inc(&fs->count); |
398 | 398 | ||
399 | exit_task_namespaces(current); | 399 | put_and_finalize_nsproxy(current->nsproxy); |
400 | current->nsproxy = init_task.nsproxy; | 400 | current->nsproxy = init_task.nsproxy; |
401 | get_task_namespaces(current); | 401 | get_task_namespaces(current); |
402 | 402 | ||
@@ -853,6 +853,7 @@ static void exit_notify(struct task_struct *tsk) | |||
853 | fastcall NORET_TYPE void do_exit(long code) | 853 | fastcall NORET_TYPE void do_exit(long code) |
854 | { | 854 | { |
855 | struct task_struct *tsk = current; | 855 | struct task_struct *tsk = current; |
856 | struct nsproxy *ns; | ||
856 | int group_dead; | 857 | int group_dead; |
857 | 858 | ||
858 | profile_task_exit(tsk); | 859 | profile_task_exit(tsk); |
@@ -938,8 +939,9 @@ fastcall NORET_TYPE void do_exit(long code) | |||
938 | 939 | ||
939 | tsk->exit_code = code; | 940 | tsk->exit_code = code; |
940 | proc_exit_connector(tsk); | 941 | proc_exit_connector(tsk); |
942 | ns = preexit_task_namespaces(tsk); | ||
941 | exit_notify(tsk); | 943 | exit_notify(tsk); |
942 | exit_task_namespaces(tsk); | 944 | exit_task_namespaces(tsk, ns); |
943 | #ifdef CONFIG_NUMA | 945 | #ifdef CONFIG_NUMA |
944 | mpol_free(tsk->mempolicy); | 946 | mpol_free(tsk->mempolicy); |
945 | tsk->mempolicy = NULL; | 947 | tsk->mempolicy = NULL; |