Blame view

kernel/nsproxy.c 6.43 KB
ab516013a   Serge E. Hallyn   [PATCH] namespace...
1
2
3
4
5
6
7
8
9
  /*
   *  Copyright (C) 2006 IBM Corporation
   *
   *  Author: Serge Hallyn <serue@us.ibm.com>
   *
   *  This program is free software; you can redistribute it and/or
   *  modify it under the terms of the GNU General Public License as
   *  published by the Free Software Foundation, version 2 of the
   *  License.
25b21cb2f   Kirill Korotaev   [PATCH] IPC names...
10
11
12
13
   *
   *  Jun 2006 - namespaces support
   *             OpenVZ, SWsoft Inc.
   *             Pavel Emelianov <xemul@openvz.org>
ab516013a   Serge E. Hallyn   [PATCH] namespace...
14
   */
5a0e3ad6a   Tejun Heo   include cleanup: ...
15
  #include <linux/slab.h>
9984de1a5   Paul Gortmaker   kernel: Map most ...
16
  #include <linux/export.h>
ab516013a   Serge E. Hallyn   [PATCH] namespace...
17
  #include <linux/nsproxy.h>
0437eb594   Serge E. Hallyn   [PATCH] nsproxy: ...
18
  #include <linux/init_task.h>
6b3286ed1   Kirill Korotaev   [PATCH] rename st...
19
  #include <linux/mnt_namespace.h>
4865ecf13   Serge E. Hallyn   [PATCH] namespace...
20
  #include <linux/utsname.h>
9a575a92d   Cedric Le Goater   [PATCH] to nsproxy
21
  #include <linux/pid_namespace.h>
9dd776b6d   Eric W. Biederman   [NET]: Add networ...
22
  #include <net/net_namespace.h>
ae5e1b22f   Pavel Emelyanov   namespaces: move ...
23
  #include <linux/ipc_namespace.h>
0bb80f240   David Howells   proc: Split the n...
24
  #include <linux/proc_ns.h>
0663c6f8f   Eric W. Biederman   ns: Introduce the...
25
26
  #include <linux/file.h>
  #include <linux/syscalls.h>
a79a908fd   Aditya Kali   cgroup: introduce...
27
  #include <linux/cgroup.h>
0437eb594   Serge E. Hallyn   [PATCH] nsproxy: ...
28

98c0d07cb   Cedric Le Goater   add a kmem_cache ...
29
  static struct kmem_cache *nsproxy_cachep;
8467005da   Alexey Dobriyan   nsproxy: remove I...
30
  struct nsproxy init_nsproxy = {
c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
31
32
  	.count			= ATOMIC_INIT(1),
  	.uts_ns			= &init_uts_ns,
8467005da   Alexey Dobriyan   nsproxy: remove I...
33
  #if defined(CONFIG_POSIX_MQUEUE) || defined(CONFIG_SYSVIPC)
c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
34
  	.ipc_ns			= &init_ipc_ns,
8467005da   Alexey Dobriyan   nsproxy: remove I...
35
  #endif
c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
36
37
  	.mnt_ns			= NULL,
  	.pid_ns_for_children	= &init_pid_ns,
8467005da   Alexey Dobriyan   nsproxy: remove I...
38
  #ifdef CONFIG_NET
c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
39
  	.net_ns			= &init_net,
8467005da   Alexey Dobriyan   nsproxy: remove I...
40
  #endif
a79a908fd   Aditya Kali   cgroup: introduce...
41
42
43
  #ifdef CONFIG_CGROUPS
  	.cgroup_ns		= &init_cgroup_ns,
  #endif
8467005da   Alexey Dobriyan   nsproxy: remove I...
44
  };
ab516013a   Serge E. Hallyn   [PATCH] namespace...
45

90af90d7d   Alexey Dobriyan   nsproxy: extract ...
46
  static inline struct nsproxy *create_nsproxy(void)
ab516013a   Serge E. Hallyn   [PATCH] namespace...
47
  {
90af90d7d   Alexey Dobriyan   nsproxy: extract ...
48
  	struct nsproxy *nsproxy;
ab516013a   Serge E. Hallyn   [PATCH] namespace...
49

90af90d7d   Alexey Dobriyan   nsproxy: extract ...
50
51
52
53
  	nsproxy = kmem_cache_alloc(nsproxy_cachep, GFP_KERNEL);
  	if (nsproxy)
  		atomic_set(&nsproxy->count, 1);
  	return nsproxy;
ab516013a   Serge E. Hallyn   [PATCH] namespace...
54
55
56
  }
  
  /*
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
57
58
59
   * Create new nsproxy and all of its the associated namespaces.
   * Return the newly created nsproxy.  Do not attach this to the task,
   * leave it to the caller to do proper locking and attach it to task.
ab516013a   Serge E. Hallyn   [PATCH] namespace...
60
   */
213dd266d   Eric W. Biederman   namespace: ensure...
61
  static struct nsproxy *create_new_namespaces(unsigned long flags,
bcf58e725   Eric W. Biederman   userns: Make crea...
62
63
  	struct task_struct *tsk, struct user_namespace *user_ns,
  	struct fs_struct *new_fs)
ab516013a   Serge E. Hallyn   [PATCH] namespace...
64
  {
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
65
  	struct nsproxy *new_nsp;
467e9f4b5   Cedric Le Goater   fix create_new_na...
66
  	int err;
ab516013a   Serge E. Hallyn   [PATCH] namespace...
67

90af90d7d   Alexey Dobriyan   nsproxy: extract ...
68
  	new_nsp = create_nsproxy();
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
69
70
  	if (!new_nsp)
  		return ERR_PTR(-ENOMEM);
1651e14e2   Serge E. Hallyn   [PATCH] namespace...
71

bcf58e725   Eric W. Biederman   userns: Make crea...
72
  	new_nsp->mnt_ns = copy_mnt_ns(flags, tsk->nsproxy->mnt_ns, user_ns, new_fs);
467e9f4b5   Cedric Le Goater   fix create_new_na...
73
74
  	if (IS_ERR(new_nsp->mnt_ns)) {
  		err = PTR_ERR(new_nsp->mnt_ns);
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
75
  		goto out_ns;
467e9f4b5   Cedric Le Goater   fix create_new_na...
76
  	}
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
77

bcf58e725   Eric W. Biederman   userns: Make crea...
78
  	new_nsp->uts_ns = copy_utsname(flags, user_ns, tsk->nsproxy->uts_ns);
467e9f4b5   Cedric Le Goater   fix create_new_na...
79
80
  	if (IS_ERR(new_nsp->uts_ns)) {
  		err = PTR_ERR(new_nsp->uts_ns);
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
81
  		goto out_uts;
467e9f4b5   Cedric Le Goater   fix create_new_na...
82
  	}
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
83

bcf58e725   Eric W. Biederman   userns: Make crea...
84
  	new_nsp->ipc_ns = copy_ipcs(flags, user_ns, tsk->nsproxy->ipc_ns);
467e9f4b5   Cedric Le Goater   fix create_new_na...
85
86
  	if (IS_ERR(new_nsp->ipc_ns)) {
  		err = PTR_ERR(new_nsp->ipc_ns);
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
87
  		goto out_ipc;
467e9f4b5   Cedric Le Goater   fix create_new_na...
88
  	}
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
89

c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
90
91
92
93
  	new_nsp->pid_ns_for_children =
  		copy_pid_ns(flags, user_ns, tsk->nsproxy->pid_ns_for_children);
  	if (IS_ERR(new_nsp->pid_ns_for_children)) {
  		err = PTR_ERR(new_nsp->pid_ns_for_children);
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
94
  		goto out_pid;
467e9f4b5   Cedric Le Goater   fix create_new_na...
95
  	}
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
96

a79a908fd   Aditya Kali   cgroup: introduce...
97
98
99
100
101
102
  	new_nsp->cgroup_ns = copy_cgroup_ns(flags, user_ns,
  					    tsk->nsproxy->cgroup_ns);
  	if (IS_ERR(new_nsp->cgroup_ns)) {
  		err = PTR_ERR(new_nsp->cgroup_ns);
  		goto out_cgroup;
  	}
bcf58e725   Eric W. Biederman   userns: Make crea...
103
  	new_nsp->net_ns = copy_net_ns(flags, user_ns, tsk->nsproxy->net_ns);
9dd776b6d   Eric W. Biederman   [NET]: Add networ...
104
105
106
107
  	if (IS_ERR(new_nsp->net_ns)) {
  		err = PTR_ERR(new_nsp->net_ns);
  		goto out_net;
  	}
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
108
  	return new_nsp;
9dd776b6d   Eric W. Biederman   [NET]: Add networ...
109
  out_net:
a79a908fd   Aditya Kali   cgroup: introduce...
110
111
  	put_cgroup_ns(new_nsp->cgroup_ns);
  out_cgroup:
c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
112
113
  	if (new_nsp->pid_ns_for_children)
  		put_pid_ns(new_nsp->pid_ns_for_children);
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
114
115
116
117
118
119
120
121
122
123
  out_pid:
  	if (new_nsp->ipc_ns)
  		put_ipc_ns(new_nsp->ipc_ns);
  out_ipc:
  	if (new_nsp->uts_ns)
  		put_uts_ns(new_nsp->uts_ns);
  out_uts:
  	if (new_nsp->mnt_ns)
  		put_mnt_ns(new_nsp->mnt_ns);
  out_ns:
98c0d07cb   Cedric Le Goater   add a kmem_cache ...
124
  	kmem_cache_free(nsproxy_cachep, new_nsp);
467e9f4b5   Cedric Le Goater   fix create_new_na...
125
  	return ERR_PTR(err);
ab516013a   Serge E. Hallyn   [PATCH] namespace...
126
127
128
129
130
131
  }
  
  /*
   * called from clone.  This now handles copy for nsproxy and all
   * namespaces therein.
   */
213dd266d   Eric W. Biederman   namespace: ensure...
132
  int copy_namespaces(unsigned long flags, struct task_struct *tsk)
ab516013a   Serge E. Hallyn   [PATCH] namespace...
133
134
  {
  	struct nsproxy *old_ns = tsk->nsproxy;
b33c77ef2   Eric W. Biederman   userns: Allow unp...
135
  	struct user_namespace *user_ns = task_cred_xxx(tsk, user_ns);
1651e14e2   Serge E. Hallyn   [PATCH] namespace...
136
  	struct nsproxy *new_ns;
ab516013a   Serge E. Hallyn   [PATCH] namespace...
137

dbef0c1c4   Eric W. Biederman   namespaces: Simpl...
138
  	if (likely(!(flags & (CLONE_NEWNS | CLONE_NEWUTS | CLONE_NEWIPC |
a79a908fd   Aditya Kali   cgroup: introduce...
139
140
  			      CLONE_NEWPID | CLONE_NEWNET |
  			      CLONE_NEWCGROUP)))) {
dbef0c1c4   Eric W. Biederman   namespaces: Simpl...
141
  		get_nsproxy(old_ns);
ab516013a   Serge E. Hallyn   [PATCH] namespace...
142
  		return 0;
1651e14e2   Serge E. Hallyn   [PATCH] namespace...
143
  	}
dbef0c1c4   Eric W. Biederman   namespaces: Simpl...
144
145
  	if (!ns_capable(user_ns, CAP_SYS_ADMIN))
  		return -EPERM;
02fdb36ae   Serge E. Hallyn   ipc: sysvsem: ref...
146
147
148
149
150
151
152
  	/*
  	 * CLONE_NEWIPC must detach from the undolist: after switching
  	 * to a new ipc namespace, the semaphore arrays from the old
  	 * namespace are unreachable.  In clone parlance, CLONE_SYSVSEM
  	 * means share undolist with parent, so we must forbid using
  	 * it along with CLONE_NEWIPC.
  	 */
21e851943   Raphael S.Carvalho   kernel/nsproxy.c:...
153
  	if ((flags & (CLONE_NEWIPC | CLONE_SYSVSEM)) ==
dbef0c1c4   Eric W. Biederman   namespaces: Simpl...
154
155
  		(CLONE_NEWIPC | CLONE_SYSVSEM)) 
  		return -EINVAL;
02fdb36ae   Serge E. Hallyn   ipc: sysvsem: ref...
156

d7d48f621   Yuanhan Liu   kernel/nsproxy.c:...
157
  	new_ns = create_new_namespaces(flags, tsk, user_ns, tsk->fs);
dbef0c1c4   Eric W. Biederman   namespaces: Simpl...
158
159
  	if (IS_ERR(new_ns))
  		return  PTR_ERR(new_ns);
9a575a92d   Cedric Le Goater   [PATCH] to nsproxy
160

e3222c4ec   Badari Pulavarty   Merge sys_clone()...
161
  	tsk->nsproxy = new_ns;
dbef0c1c4   Eric W. Biederman   namespaces: Simpl...
162
  	return 0;
ab516013a   Serge E. Hallyn   [PATCH] namespace...
163
164
165
166
  }
  
  void free_nsproxy(struct nsproxy *ns)
  {
9a575a92d   Cedric Le Goater   [PATCH] to nsproxy
167
168
169
170
171
172
  	if (ns->mnt_ns)
  		put_mnt_ns(ns->mnt_ns);
  	if (ns->uts_ns)
  		put_uts_ns(ns->uts_ns);
  	if (ns->ipc_ns)
  		put_ipc_ns(ns->ipc_ns);
c2b1df2eb   Andy Lutomirski   Rename nsproxy.pi...
173
174
  	if (ns->pid_ns_for_children)
  		put_pid_ns(ns->pid_ns_for_children);
a79a908fd   Aditya Kali   cgroup: introduce...
175
  	put_cgroup_ns(ns->cgroup_ns);
9dd776b6d   Eric W. Biederman   [NET]: Add networ...
176
  	put_net(ns->net_ns);
98c0d07cb   Cedric Le Goater   add a kmem_cache ...
177
  	kmem_cache_free(nsproxy_cachep, ns);
ab516013a   Serge E. Hallyn   [PATCH] namespace...
178
  }
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
179
180
181
  
  /*
   * Called from unshare. Unshare all the namespaces part of nsproxy.
4e71e474c   Cedric Le Goater   fix refcounting o...
182
   * On success, returns the new nsproxy.
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
183
184
   */
  int unshare_nsproxy_namespaces(unsigned long unshare_flags,
b2e0d9870   Eric W. Biederman   userns: Implement...
185
  	struct nsproxy **new_nsp, struct cred *new_cred, struct fs_struct *new_fs)
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
186
  {
bcf58e725   Eric W. Biederman   userns: Make crea...
187
  	struct user_namespace *user_ns;
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
188
  	int err = 0;
77ec739d8   Serge E. Hallyn   user namespace: a...
189
  	if (!(unshare_flags & (CLONE_NEWNS | CLONE_NEWUTS | CLONE_NEWIPC |
a79a908fd   Aditya Kali   cgroup: introduce...
190
  			       CLONE_NEWNET | CLONE_NEWPID | CLONE_NEWCGROUP)))
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
191
  		return 0;
b2e0d9870   Eric W. Biederman   userns: Implement...
192
193
  	user_ns = new_cred ? new_cred->user_ns : current_user_ns();
  	if (!ns_capable(user_ns, CAP_SYS_ADMIN))
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
194
  		return -EPERM;
bcf58e725   Eric W. Biederman   userns: Make crea...
195
  	*new_nsp = create_new_namespaces(unshare_flags, current, user_ns,
b2e0d9870   Eric W. Biederman   userns: Implement...
196
  					 new_fs ? new_fs : current->fs);
858d72ead   Serge E. Hallyn   cgroups: implemen...
197
  	if (IS_ERR(*new_nsp)) {
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
198
  		err = PTR_ERR(*new_nsp);
858d72ead   Serge E. Hallyn   cgroups: implemen...
199
200
  		goto out;
  	}
858d72ead   Serge E. Hallyn   cgroups: implemen...
201
  out:
e3222c4ec   Badari Pulavarty   Merge sys_clone()...
202
203
  	return err;
  }
98c0d07cb   Cedric Le Goater   add a kmem_cache ...
204

cf7b708c8   Pavel Emelyanov   Make access to ta...
205
206
207
208
209
  void switch_task_namespaces(struct task_struct *p, struct nsproxy *new)
  {
  	struct nsproxy *ns;
  
  	might_sleep();
728dba3a3   Eric W. Biederman   namespaces: Use t...
210
  	task_lock(p);
cf7b708c8   Pavel Emelyanov   Make access to ta...
211
  	ns = p->nsproxy;
728dba3a3   Eric W. Biederman   namespaces: Use t...
212
213
  	p->nsproxy = new;
  	task_unlock(p);
cf7b708c8   Pavel Emelyanov   Make access to ta...
214

728dba3a3   Eric W. Biederman   namespaces: Use t...
215
  	if (ns && atomic_dec_and_test(&ns->count))
cf7b708c8   Pavel Emelyanov   Make access to ta...
216
  		free_nsproxy(ns);
cf7b708c8   Pavel Emelyanov   Make access to ta...
217
218
219
220
221
222
  }
  
  void exit_task_namespaces(struct task_struct *p)
  {
  	switch_task_namespaces(p, NULL);
  }
0663c6f8f   Eric W. Biederman   ns: Introduce the...
223
224
  SYSCALL_DEFINE2(setns, int, fd, int, nstype)
  {
0663c6f8f   Eric W. Biederman   ns: Introduce the...
225
226
  	struct task_struct *tsk = current;
  	struct nsproxy *new_nsproxy;
0663c6f8f   Eric W. Biederman   ns: Introduce the...
227
  	struct file *file;
33c429405   Al Viro   copy address of p...
228
  	struct ns_common *ns;
0663c6f8f   Eric W. Biederman   ns: Introduce the...
229
  	int err;
0663c6f8f   Eric W. Biederman   ns: Introduce the...
230
231
232
233
234
  	file = proc_ns_fget(fd);
  	if (IS_ERR(file))
  		return PTR_ERR(file);
  
  	err = -EINVAL;
f77c80142   Al Viro   bury struct proc_...
235
  	ns = get_proc_ns(file_inode(file));
33c429405   Al Viro   copy address of p...
236
  	if (nstype && (ns->ops->type != nstype))
0663c6f8f   Eric W. Biederman   ns: Introduce the...
237
  		goto out;
bcf58e725   Eric W. Biederman   userns: Make crea...
238
  	new_nsproxy = create_new_namespaces(0, tsk, current_user_ns(), tsk->fs);
0663c6f8f   Eric W. Biederman   ns: Introduce the...
239
240
241
242
  	if (IS_ERR(new_nsproxy)) {
  		err = PTR_ERR(new_nsproxy);
  		goto out;
  	}
33c429405   Al Viro   copy address of p...
243
  	err = ns->ops->install(new_nsproxy, ns);
0663c6f8f   Eric W. Biederman   ns: Introduce the...
244
245
246
247
248
249
250
251
252
  	if (err) {
  		free_nsproxy(new_nsproxy);
  		goto out;
  	}
  	switch_task_namespaces(tsk, new_nsproxy);
  out:
  	fput(file);
  	return err;
  }
665771939   Al Viro   make sure that ns...
253
  int __init nsproxy_cache_init(void)
98c0d07cb   Cedric Le Goater   add a kmem_cache ...
254
  {
db8906da5   Pavel Emelyanov   Use KMEM_CACHE ma...
255
  	nsproxy_cachep = KMEM_CACHE(nsproxy, SLAB_PANIC);
98c0d07cb   Cedric Le Goater   add a kmem_cache ...
256
257
  	return 0;
  }