aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/cgroup.c
diff options
context:
space:
mode:
authorLi Zefan <lizf@cn.fujitsu.com>2009-01-29 17:25:21 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2009-01-29 21:04:45 -0500
commit1404f06565ee89e0ce04d4a5859c00b0e3a0dc8d (patch)
treee2f294ed420461e0032e03143f06634b883dc06e /kernel/cgroup.c
parent945048ca36173315afa2f0c53bed21ba01a588c1 (diff)
cgroups: fix lock inconsistency in cgroup_clone()
I fixed a bug in cgroup_clone() in Linus' tree in commit 7b574b7 ("cgroups: fix a race between cgroup_clone and umount") without noticing there was a cleanup patch in -mm tree that should be rebased (now commit 104cbd5, "cgroups: use task_lock() for access tsk->cgroups safe in cgroup_clone()"), thus resulted in lock inconsistency. Signed-off-by: Li Zefan <lizf@cn.fujitsu.com> Acked-by: Paul Menage <menage@google.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'kernel/cgroup.c')
-rw-r--r--kernel/cgroup.c13
1 files changed, 7 insertions, 6 deletions
diff --git a/kernel/cgroup.c b/kernel/cgroup.c
index 2ae7cb47dbfa..0066092de19a 100644
--- a/kernel/cgroup.c
+++ b/kernel/cgroup.c
@@ -2993,20 +2993,21 @@ int cgroup_clone(struct task_struct *tsk, struct cgroup_subsys *subsys,
2993 mutex_unlock(&cgroup_mutex); 2993 mutex_unlock(&cgroup_mutex);
2994 return 0; 2994 return 0;
2995 } 2995 }
2996 task_lock(tsk);
2997 cg = tsk->cgroups;
2998 parent = task_cgroup(tsk, subsys->subsys_id);
2999 2996
3000 /* Pin the hierarchy */ 2997 /* Pin the hierarchy */
3001 if (!atomic_inc_not_zero(&parent->root->sb->s_active)) { 2998 if (!atomic_inc_not_zero(&root->sb->s_active)) {
3002 /* We race with the final deactivate_super() */ 2999 /* We race with the final deactivate_super() */
3003 mutex_unlock(&cgroup_mutex); 3000 mutex_unlock(&cgroup_mutex);
3004 return 0; 3001 return 0;
3005 } 3002 }
3006 3003
3007 /* Keep the cgroup alive */ 3004 /* Keep the cgroup alive */
3005 task_lock(tsk);
3006 parent = task_cgroup(tsk, subsys->subsys_id);
3007 cg = tsk->cgroups;
3008 get_css_set(cg); 3008 get_css_set(cg);
3009 task_unlock(tsk); 3009 task_unlock(tsk);
3010
3010 mutex_unlock(&cgroup_mutex); 3011 mutex_unlock(&cgroup_mutex);
3011 3012
3012 /* Now do the VFS work to create a cgroup */ 3013 /* Now do the VFS work to create a cgroup */
@@ -3045,7 +3046,7 @@ int cgroup_clone(struct task_struct *tsk, struct cgroup_subsys *subsys,
3045 mutex_unlock(&inode->i_mutex); 3046 mutex_unlock(&inode->i_mutex);
3046 put_css_set(cg); 3047 put_css_set(cg);
3047 3048
3048 deactivate_super(parent->root->sb); 3049 deactivate_super(root->sb);
3049 /* The cgroup is still accessible in the VFS, but 3050 /* The cgroup is still accessible in the VFS, but
3050 * we're not going to try to rmdir() it at this 3051 * we're not going to try to rmdir() it at this
3051 * point. */ 3052 * point. */
@@ -3071,7 +3072,7 @@ int cgroup_clone(struct task_struct *tsk, struct cgroup_subsys *subsys,
3071 mutex_lock(&cgroup_mutex); 3072 mutex_lock(&cgroup_mutex);
3072 put_css_set(cg); 3073 put_css_set(cg);
3073 mutex_unlock(&cgroup_mutex); 3074 mutex_unlock(&cgroup_mutex);
3074 deactivate_super(parent->root->sb); 3075 deactivate_super(root->sb);
3075 return ret; 3076 return ret;
3076} 3077}
3077 3078