diff options
-rw-r--r-- | include/linux/init_task.h | 2 | ||||
-rw-r--r-- | include/linux/nsproxy.h | 2 | ||||
-rw-r--r-- | include/linux/pid_namespace.h | 20 | ||||
-rw-r--r-- | kernel/nsproxy.c | 26 | ||||
-rw-r--r-- | kernel/pid.c | 23 |
5 files changed, 64 insertions, 9 deletions
diff --git a/include/linux/init_task.h b/include/linux/init_task.h index 90c5f9a07730..7272ff9ee77c 100644 --- a/include/linux/init_task.h +++ b/include/linux/init_task.h | |||
@@ -7,6 +7,7 @@ | |||
7 | #include <linux/utsname.h> | 7 | #include <linux/utsname.h> |
8 | #include <linux/lockdep.h> | 8 | #include <linux/lockdep.h> |
9 | #include <linux/ipc.h> | 9 | #include <linux/ipc.h> |
10 | #include <linux/pid_namespace.h> | ||
10 | 11 | ||
11 | #define INIT_FDTABLE \ | 12 | #define INIT_FDTABLE \ |
12 | { \ | 13 | { \ |
@@ -73,6 +74,7 @@ | |||
73 | 74 | ||
74 | extern struct nsproxy init_nsproxy; | 75 | extern struct nsproxy init_nsproxy; |
75 | #define INIT_NSPROXY(nsproxy) { \ | 76 | #define INIT_NSPROXY(nsproxy) { \ |
77 | .pid_ns = &init_pid_ns, \ | ||
76 | .count = ATOMIC_INIT(1), \ | 78 | .count = ATOMIC_INIT(1), \ |
77 | .nslock = __SPIN_LOCK_UNLOCKED(nsproxy.nslock), \ | 79 | .nslock = __SPIN_LOCK_UNLOCKED(nsproxy.nslock), \ |
78 | .id = 0, \ | 80 | .id = 0, \ |
diff --git a/include/linux/nsproxy.h b/include/linux/nsproxy.h index 27f37c1ec1d9..fdfb0e44912f 100644 --- a/include/linux/nsproxy.h +++ b/include/linux/nsproxy.h | |||
@@ -7,6 +7,7 @@ | |||
7 | struct mnt_namespace; | 7 | struct mnt_namespace; |
8 | struct uts_namespace; | 8 | struct uts_namespace; |
9 | struct ipc_namespace; | 9 | struct ipc_namespace; |
10 | struct pid_namespace; | ||
10 | 11 | ||
11 | /* | 12 | /* |
12 | * A structure to contain pointers to all per-process | 13 | * A structure to contain pointers to all per-process |
@@ -27,6 +28,7 @@ struct nsproxy { | |||
27 | struct uts_namespace *uts_ns; | 28 | struct uts_namespace *uts_ns; |
28 | struct ipc_namespace *ipc_ns; | 29 | struct ipc_namespace *ipc_ns; |
29 | struct mnt_namespace *mnt_ns; | 30 | struct mnt_namespace *mnt_ns; |
31 | struct pid_namespace *pid_ns; | ||
30 | }; | 32 | }; |
31 | extern struct nsproxy init_nsproxy; | 33 | extern struct nsproxy init_nsproxy; |
32 | 34 | ||
diff --git a/include/linux/pid_namespace.h b/include/linux/pid_namespace.h index 54d79095e295..76e7c6b2cf33 100644 --- a/include/linux/pid_namespace.h +++ b/include/linux/pid_namespace.h | |||
@@ -5,6 +5,8 @@ | |||
5 | #include <linux/mm.h> | 5 | #include <linux/mm.h> |
6 | #include <linux/threads.h> | 6 | #include <linux/threads.h> |
7 | #include <linux/pid.h> | 7 | #include <linux/pid.h> |
8 | #include <linux/nsproxy.h> | ||
9 | #include <linux/kref.h> | ||
8 | 10 | ||
9 | struct pidmap { | 11 | struct pidmap { |
10 | atomic_t nr_free; | 12 | atomic_t nr_free; |
@@ -14,10 +16,24 @@ struct pidmap { | |||
14 | #define PIDMAP_ENTRIES ((PID_MAX_LIMIT + 8*PAGE_SIZE - 1)/PAGE_SIZE/8) | 16 | #define PIDMAP_ENTRIES ((PID_MAX_LIMIT + 8*PAGE_SIZE - 1)/PAGE_SIZE/8) |
15 | 17 | ||
16 | struct pid_namespace { | 18 | struct pid_namespace { |
17 | struct pidmap pidmap[PIDMAP_ENTRIES]; | 19 | struct kref kref; |
18 | int last_pid; | 20 | struct pidmap pidmap[PIDMAP_ENTRIES]; |
21 | int last_pid; | ||
19 | }; | 22 | }; |
20 | 23 | ||
21 | extern struct pid_namespace init_pid_ns; | 24 | extern struct pid_namespace init_pid_ns; |
22 | 25 | ||
26 | static inline void get_pid_ns(struct pid_namespace *ns) | ||
27 | { | ||
28 | kref_get(&ns->kref); | ||
29 | } | ||
30 | |||
31 | extern int copy_pid_ns(int flags, struct task_struct *tsk); | ||
32 | extern void free_pid_ns(struct kref *kref); | ||
33 | |||
34 | static inline void put_pid_ns(struct pid_namespace *ns) | ||
35 | { | ||
36 | kref_put(&ns->kref, free_pid_ns); | ||
37 | } | ||
38 | |||
23 | #endif /* _LINUX_PID_NS_H */ | 39 | #endif /* _LINUX_PID_NS_H */ |
diff --git a/kernel/nsproxy.c b/kernel/nsproxy.c index f223c15c18e9..e2ce748e96af 100644 --- a/kernel/nsproxy.c +++ b/kernel/nsproxy.c | |||
@@ -19,6 +19,7 @@ | |||
19 | #include <linux/init_task.h> | 19 | #include <linux/init_task.h> |
20 | #include <linux/mnt_namespace.h> | 20 | #include <linux/mnt_namespace.h> |
21 | #include <linux/utsname.h> | 21 | #include <linux/utsname.h> |
22 | #include <linux/pid_namespace.h> | ||
22 | 23 | ||
23 | struct nsproxy init_nsproxy = INIT_NSPROXY(init_nsproxy); | 24 | struct nsproxy init_nsproxy = INIT_NSPROXY(init_nsproxy); |
24 | 25 | ||
@@ -68,6 +69,8 @@ struct nsproxy *dup_namespaces(struct nsproxy *orig) | |||
68 | get_uts_ns(ns->uts_ns); | 69 | get_uts_ns(ns->uts_ns); |
69 | if (ns->ipc_ns) | 70 | if (ns->ipc_ns) |
70 | get_ipc_ns(ns->ipc_ns); | 71 | get_ipc_ns(ns->ipc_ns); |
72 | if (ns->pid_ns) | ||
73 | get_pid_ns(ns->pid_ns); | ||
71 | } | 74 | } |
72 | 75 | ||
73 | return ns; | 76 | return ns; |
@@ -111,10 +114,17 @@ int copy_namespaces(int flags, struct task_struct *tsk) | |||
111 | if (err) | 114 | if (err) |
112 | goto out_ipc; | 115 | goto out_ipc; |
113 | 116 | ||
117 | err = copy_pid_ns(flags, tsk); | ||
118 | if (err) | ||
119 | goto out_pid; | ||
120 | |||
114 | out: | 121 | out: |
115 | put_nsproxy(old_ns); | 122 | put_nsproxy(old_ns); |
116 | return err; | 123 | return err; |
117 | 124 | ||
125 | out_pid: | ||
126 | if (new_ns->ipc_ns) | ||
127 | put_ipc_ns(new_ns->ipc_ns); | ||
118 | out_ipc: | 128 | out_ipc: |
119 | if (new_ns->uts_ns) | 129 | if (new_ns->uts_ns) |
120 | put_uts_ns(new_ns->uts_ns); | 130 | put_uts_ns(new_ns->uts_ns); |
@@ -129,11 +139,13 @@ out_ns: | |||
129 | 139 | ||
130 | void free_nsproxy(struct nsproxy *ns) | 140 | void free_nsproxy(struct nsproxy *ns) |
131 | { | 141 | { |
132 | if (ns->mnt_ns) | 142 | if (ns->mnt_ns) |
133 | put_mnt_ns(ns->mnt_ns); | 143 | put_mnt_ns(ns->mnt_ns); |
134 | if (ns->uts_ns) | 144 | if (ns->uts_ns) |
135 | put_uts_ns(ns->uts_ns); | 145 | put_uts_ns(ns->uts_ns); |
136 | if (ns->ipc_ns) | 146 | if (ns->ipc_ns) |
137 | put_ipc_ns(ns->ipc_ns); | 147 | put_ipc_ns(ns->ipc_ns); |
138 | kfree(ns); | 148 | if (ns->pid_ns) |
149 | put_pid_ns(ns->pid_ns); | ||
150 | kfree(ns); | ||
139 | } | 151 | } |
diff --git a/kernel/pid.c b/kernel/pid.c index 25807e1b98dd..5319b9f2fc5e 100644 --- a/kernel/pid.c +++ b/kernel/pid.c | |||
@@ -59,6 +59,9 @@ static inline int mk_pid(struct pid_namespace *pid_ns, | |||
59 | * the scheme scales to up to 4 million PIDs, runtime. | 59 | * the scheme scales to up to 4 million PIDs, runtime. |
60 | */ | 60 | */ |
61 | struct pid_namespace init_pid_ns = { | 61 | struct pid_namespace init_pid_ns = { |
62 | .kref = { | ||
63 | .refcount = ATOMIC_INIT(2), | ||
64 | }, | ||
62 | .pidmap = { | 65 | .pidmap = { |
63 | [ 0 ... PIDMAP_ENTRIES-1] = { ATOMIC_INIT(BITS_PER_PAGE), NULL } | 66 | [ 0 ... PIDMAP_ENTRIES-1] = { ATOMIC_INIT(BITS_PER_PAGE), NULL } |
64 | }, | 67 | }, |
@@ -356,6 +359,26 @@ struct pid *find_ge_pid(int nr) | |||
356 | } | 359 | } |
357 | EXPORT_SYMBOL_GPL(find_get_pid); | 360 | EXPORT_SYMBOL_GPL(find_get_pid); |
358 | 361 | ||
362 | int copy_pid_ns(int flags, struct task_struct *tsk) | ||
363 | { | ||
364 | struct pid_namespace *old_ns = tsk->nsproxy->pid_ns; | ||
365 | int err = 0; | ||
366 | |||
367 | if (!old_ns) | ||
368 | return 0; | ||
369 | |||
370 | get_pid_ns(old_ns); | ||
371 | return err; | ||
372 | } | ||
373 | |||
374 | void free_pid_ns(struct kref *kref) | ||
375 | { | ||
376 | struct pid_namespace *ns; | ||
377 | |||
378 | ns = container_of(kref, struct pid_namespace, kref); | ||
379 | kfree(ns); | ||
380 | } | ||
381 | |||
359 | /* | 382 | /* |
360 | * The pid hash table is scaled according to the amount of memory in the | 383 | * The pid hash table is scaled according to the amount of memory in the |
361 | * machine. From a minimum of 16 slots up to 4096 slots at one gigabyte or | 384 | * machine. From a minimum of 16 slots up to 4096 slots at one gigabyte or |