diff options
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/exit.c | 6 | ||||
-rw-r--r-- | kernel/fork.c | 4 | ||||
-rw-r--r-- | kernel/nsproxy.c | 16 |
3 files changed, 21 insertions, 5 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; |
diff --git a/kernel/fork.c b/kernel/fork.c index fc723e595cd5..4cf868458f06 100644 --- a/kernel/fork.c +++ b/kernel/fork.c | |||
@@ -1265,7 +1265,7 @@ static struct task_struct *copy_process(unsigned long clone_flags, | |||
1265 | return p; | 1265 | return p; |
1266 | 1266 | ||
1267 | bad_fork_cleanup_namespaces: | 1267 | bad_fork_cleanup_namespaces: |
1268 | exit_task_namespaces(p); | 1268 | put_and_finalize_nsproxy(p->nsproxy); |
1269 | bad_fork_cleanup_keys: | 1269 | bad_fork_cleanup_keys: |
1270 | exit_keys(p); | 1270 | exit_keys(p); |
1271 | bad_fork_cleanup_mm: | 1271 | bad_fork_cleanup_mm: |
@@ -1711,7 +1711,7 @@ asmlinkage long sys_unshare(unsigned long unshare_flags) | |||
1711 | } | 1711 | } |
1712 | 1712 | ||
1713 | if (new_nsproxy) | 1713 | if (new_nsproxy) |
1714 | put_nsproxy(new_nsproxy); | 1714 | put_and_finalize_nsproxy(new_nsproxy); |
1715 | 1715 | ||
1716 | bad_unshare_cleanup_ipc: | 1716 | bad_unshare_cleanup_ipc: |
1717 | if (new_ipc) | 1717 | if (new_ipc) |
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 | ||
119 | out: | 119 | out: |
120 | put_nsproxy(old_ns); | 120 | put_and_finalize_nsproxy(old_ns); |
121 | return err; | 121 | return err; |
122 | 122 | ||
123 | out_pid: | 123 | out_pid: |
@@ -135,6 +135,20 @@ out_ns: | |||
135 | goto out; | 135 | goto out; |
136 | } | 136 | } |
137 | 137 | ||
138 | struct 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 | |||
138 | void free_nsproxy(struct nsproxy *ns) | 152 | void free_nsproxy(struct nsproxy *ns) |
139 | { | 153 | { |
140 | if (ns->mnt_ns) | 154 | if (ns->mnt_ns) |