diff options
author | Badari Pulavarty <pbadari@us.ibm.com> | 2007-05-08 03:25:21 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-05-08 14:15:00 -0400 |
commit | e3222c4ecc649c4ae568e61dda9349482401b501 (patch) | |
tree | d96614ef67d947a3dd8ab0929a4755bce9fdbcc1 /ipc | |
parent | 4fc75ff4816c3483b4b772b2f6cb3d8fd88ca547 (diff) |
Merge sys_clone()/sys_unshare() nsproxy and namespace handling
sys_clone() and sys_unshare() both makes copies of nsproxy and its associated
namespaces. But they have different code paths.
This patch merges all the nsproxy and its associated namespace copy/clone
handling (as much as possible). Posted on container list earlier for
feedback.
- Create a new nsproxy and its associated namespaces and pass it back to
caller to attach it to right process.
- Changed all copy_*_ns() routines to return a new copy of namespace
instead of attaching it to task->nsproxy.
- Moved the CAP_SYS_ADMIN checks out of copy_*_ns() routines.
- Removed unnessary !ns checks from copy_*_ns() and added BUG_ON()
just incase.
- Get rid of all individual unshare_*_ns() routines and make use of
copy_*_ns() instead.
[akpm@osdl.org: cleanups, warning fix]
[clg@fr.ibm.com: remove dup_namespaces() declaration]
[serue@us.ibm.com: fix CONFIG_IPC_NS=n, clone(CLONE_NEWIPC) retval]
[akpm@linux-foundation.org: fix build with CONFIG_SYSVIPC=n]
Signed-off-by: Badari Pulavarty <pbadari@us.ibm.com>
Signed-off-by: Serge Hallyn <serue@us.ibm.com>
Cc: Cedric Le Goater <clg@fr.ibm.com>
Cc: "Eric W. Biederman" <ebiederm@xmission.com>
Cc: <containers@lists.osdl.org>
Signed-off-by: Cedric Le Goater <clg@fr.ibm.com>
Cc: Oleg Nesterov <oleg@tv-sign.ru>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'ipc')
-rw-r--r-- | ipc/util.c | 53 |
1 files changed, 10 insertions, 43 deletions
diff --git a/ipc/util.c b/ipc/util.c index 0b652387d169..0ad5b1c3ca01 100644 --- a/ipc/util.c +++ b/ipc/util.c | |||
@@ -85,53 +85,20 @@ err_mem: | |||
85 | return ERR_PTR(err); | 85 | return ERR_PTR(err); |
86 | } | 86 | } |
87 | 87 | ||
88 | int unshare_ipcs(unsigned long unshare_flags, struct ipc_namespace **new_ipc) | 88 | struct ipc_namespace *copy_ipcs(unsigned long flags, struct ipc_namespace *ns) |
89 | { | 89 | { |
90 | struct ipc_namespace *new; | ||
91 | |||
92 | if (unshare_flags & CLONE_NEWIPC) { | ||
93 | if (!capable(CAP_SYS_ADMIN)) | ||
94 | return -EPERM; | ||
95 | |||
96 | new = clone_ipc_ns(current->nsproxy->ipc_ns); | ||
97 | if (IS_ERR(new)) | ||
98 | return PTR_ERR(new); | ||
99 | |||
100 | *new_ipc = new; | ||
101 | } | ||
102 | |||
103 | return 0; | ||
104 | } | ||
105 | |||
106 | int copy_ipcs(unsigned long flags, struct task_struct *tsk) | ||
107 | { | ||
108 | struct ipc_namespace *old_ns = tsk->nsproxy->ipc_ns; | ||
109 | struct ipc_namespace *new_ns; | 90 | struct ipc_namespace *new_ns; |
110 | int err = 0; | ||
111 | 91 | ||
112 | if (!old_ns) | 92 | BUG_ON(!ns); |
113 | return 0; | 93 | get_ipc_ns(ns); |
114 | |||
115 | get_ipc_ns(old_ns); | ||
116 | 94 | ||
117 | if (!(flags & CLONE_NEWIPC)) | 95 | if (!(flags & CLONE_NEWIPC)) |
118 | return 0; | 96 | return ns; |
119 | 97 | ||
120 | if (!capable(CAP_SYS_ADMIN)) { | 98 | new_ns = clone_ipc_ns(ns); |
121 | err = -EPERM; | ||
122 | goto out; | ||
123 | } | ||
124 | 99 | ||
125 | new_ns = clone_ipc_ns(old_ns); | 100 | put_ipc_ns(ns); |
126 | if (!new_ns) { | 101 | return new_ns; |
127 | err = -ENOMEM; | ||
128 | goto out; | ||
129 | } | ||
130 | |||
131 | tsk->nsproxy->ipc_ns = new_ns; | ||
132 | out: | ||
133 | put_ipc_ns(old_ns); | ||
134 | return err; | ||
135 | } | 102 | } |
136 | 103 | ||
137 | void free_ipc_ns(struct kref *kref) | 104 | void free_ipc_ns(struct kref *kref) |
@@ -145,11 +112,11 @@ void free_ipc_ns(struct kref *kref) | |||
145 | kfree(ns); | 112 | kfree(ns); |
146 | } | 113 | } |
147 | #else | 114 | #else |
148 | int copy_ipcs(unsigned long flags, struct task_struct *tsk) | 115 | struct ipc_namespace *copy_ipcs(unsigned long flags, struct ipc_namespace *ns) |
149 | { | 116 | { |
150 | if (flags & CLONE_NEWIPC) | 117 | if (flags & CLONE_NEWIPC) |
151 | return -EINVAL; | 118 | return ERR_PTR(-EINVAL); |
152 | return 0; | 119 | return ns; |
153 | } | 120 | } |
154 | #endif | 121 | #endif |
155 | 122 | ||