mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-26 00:44:03 +00:00 
			
		
		
		
	 e885dcde75
			
		
	
	
		e885dcde75
		
	
	
	
	
		
			
			cgroup_clone creates a new cgroup with the pid of the task. This works correctly for unshare, but for clone cgroup_clone is called from copy_namespaces inside copy_process, which happens before the new pid is created. As a result, the new cgroup was created with current's pid. This patch: 1. Moves the call inside copy_process to after the new pid is created 2. Passes the struct pid into ns_cgroup_clone (as it is not yet attached to the task) 3. Passes a name from ns_cgroup_clone() into cgroup_clone() so as to keep cgroup_clone() itself simpler 4. Uses pid_vnr() to get the process id value, so that the pid used to name the new cgroup is always the pid as it would be known to the task which did the cloning or unsharing. I think that is the most intuitive thing to do. This way, task t1 does clone(CLONE_NEWPID) to get t2, which does clone(CLONE_NEWPID) to get t3, then the cgroup for t3 will be named for the pid by which t2 knows t3. (Thanks to Dan Smith for finding the main bug) Changelog: June 11: Incorporate Paul Menage's feedback: don't pass NULL to ns_cgroup_clone from unshare, and reduce patch size by using 'nodename' in cgroup_clone. June 10: Original version [akpm@linux-foundation.org: build fix] [akpm@linux-foundation.org: coding-style fixes] Signed-off-by: Serge Hallyn <serge@us.ibm.com> Acked-by: Paul Menage <menage@google.com> Tested-by: Dan Smith <danms@us.ibm.com> Cc: Balbir Singh <balbir@in.ibm.com> Cc: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
		
			
				
	
	
		
			94 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			94 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef _LINUX_NSPROXY_H
 | |
| #define _LINUX_NSPROXY_H
 | |
| 
 | |
| #include <linux/spinlock.h>
 | |
| #include <linux/sched.h>
 | |
| 
 | |
| struct mnt_namespace;
 | |
| struct uts_namespace;
 | |
| struct ipc_namespace;
 | |
| struct pid_namespace;
 | |
| 
 | |
| /*
 | |
|  * A structure to contain pointers to all per-process
 | |
|  * namespaces - fs (mount), uts, network, sysvipc, etc.
 | |
|  *
 | |
|  * 'count' is the number of tasks holding a reference.
 | |
|  * The count for each namespace, then, will be the number
 | |
|  * of nsproxies pointing to it, not the number of tasks.
 | |
|  *
 | |
|  * The nsproxy is shared by tasks which share all namespaces.
 | |
|  * As soon as a single namespace is cloned or unshared, the
 | |
|  * nsproxy is copied.
 | |
|  */
 | |
| struct nsproxy {
 | |
| 	atomic_t count;
 | |
| 	struct uts_namespace *uts_ns;
 | |
| 	struct ipc_namespace *ipc_ns;
 | |
| 	struct mnt_namespace *mnt_ns;
 | |
| 	struct pid_namespace *pid_ns;
 | |
| 	struct user_namespace *user_ns;
 | |
| 	struct net 	     *net_ns;
 | |
| };
 | |
| extern struct nsproxy init_nsproxy;
 | |
| 
 | |
| /*
 | |
|  * the namespaces access rules are:
 | |
|  *
 | |
|  *  1. only current task is allowed to change tsk->nsproxy pointer or
 | |
|  *     any pointer on the nsproxy itself
 | |
|  *
 | |
|  *  2. when accessing (i.e. reading) current task's namespaces - no
 | |
|  *     precautions should be taken - just dereference the pointers
 | |
|  *
 | |
|  *  3. the access to other task namespaces is performed like this
 | |
|  *     rcu_read_lock();
 | |
|  *     nsproxy = task_nsproxy(tsk);
 | |
|  *     if (nsproxy != NULL) {
 | |
|  *             / *
 | |
|  *               * work with the namespaces here
 | |
|  *               * e.g. get the reference on one of them
 | |
|  *               * /
 | |
|  *     } / *
 | |
|  *         * NULL task_nsproxy() means that this task is
 | |
|  *         * almost dead (zombie)
 | |
|  *         * /
 | |
|  *     rcu_read_unlock();
 | |
|  *
 | |
|  */
 | |
| 
 | |
| static inline struct nsproxy *task_nsproxy(struct task_struct *tsk)
 | |
| {
 | |
| 	return rcu_dereference(tsk->nsproxy);
 | |
| }
 | |
| 
 | |
| int copy_namespaces(unsigned long flags, struct task_struct *tsk);
 | |
| void exit_task_namespaces(struct task_struct *tsk);
 | |
| void switch_task_namespaces(struct task_struct *tsk, struct nsproxy *new);
 | |
| void free_nsproxy(struct nsproxy *ns);
 | |
| int unshare_nsproxy_namespaces(unsigned long, struct nsproxy **,
 | |
| 	struct fs_struct *);
 | |
| 
 | |
| static inline void put_nsproxy(struct nsproxy *ns)
 | |
| {
 | |
| 	if (atomic_dec_and_test(&ns->count)) {
 | |
| 		free_nsproxy(ns);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static inline void get_nsproxy(struct nsproxy *ns)
 | |
| {
 | |
| 	atomic_inc(&ns->count);
 | |
| }
 | |
| 
 | |
| #ifdef CONFIG_CGROUP_NS
 | |
| int ns_cgroup_clone(struct task_struct *tsk, struct pid *pid);
 | |
| #else
 | |
| static inline int ns_cgroup_clone(struct task_struct *tsk, struct pid *pid)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| #endif
 |