diff options
author | Frederic Weisbecker <fweisbec@gmail.com> | 2011-12-21 14:03:19 -0500 |
---|---|---|
committer | Tejun Heo <tj@kernel.org> | 2011-12-21 14:15:00 -0500 |
commit | 7e381b0eb1e1a9805c37335562e8dc02e7d7848c (patch) | |
tree | 766a7e7dad168f93a0159114795277dc22920e01 /kernel | |
parent | 29e21368b9baf9c4b25060d65062da2dda926c70 (diff) |
cgroup: Drop task_lock(parent) on cgroup_fork()
We don't need to hold the parent task_lock() on the
parent in cgroup_fork() because we are already synchronized
against the two places that may change the parent css_set
concurrently:
- cgroup_exit(), but the parent obviously can't exit concurrently
- cgroup migration: we are synchronized against threadgroup_lock()
So we can safely remove the task_lock() there.
Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
Reviewed-by: Li Zefan <lizf@cn.fujitsu.com>
Signed-off-by: Tejun Heo <tj@kernel.org>
Cc: Containers <containers@lists.linux-foundation.org>
Cc: Cgroups <cgroups@vger.kernel.org>
Cc: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Cc: Oleg Nesterov <oleg@redhat.com>
Cc: Andrew Morton <akpm@linux-foundation.org>
Cc: Paul Menage <paul@paulmenage.org>
Cc: Mandeep Singh Baines <msb@chromium.org>
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/cgroup.c | 23 |
1 files changed, 17 insertions, 6 deletions
diff --git a/kernel/cgroup.c b/kernel/cgroup.c index bc3caff138d8..dae50d0d8e4b 100644 --- a/kernel/cgroup.c +++ b/kernel/cgroup.c | |||
@@ -4556,20 +4556,31 @@ static const struct file_operations proc_cgroupstats_operations = { | |||
4556 | * | 4556 | * |
4557 | * A pointer to the shared css_set was automatically copied in | 4557 | * A pointer to the shared css_set was automatically copied in |
4558 | * fork.c by dup_task_struct(). However, we ignore that copy, since | 4558 | * fork.c by dup_task_struct(). However, we ignore that copy, since |
4559 | * it was not made under the protection of RCU or cgroup_mutex, so | 4559 | * it was not made under the protection of RCU, cgroup_mutex or |
4560 | * might no longer be a valid cgroup pointer. cgroup_attach_task() might | 4560 | * threadgroup_change_begin(), so it might no longer be a valid |
4561 | * have already changed current->cgroups, allowing the previously | 4561 | * cgroup pointer. cgroup_attach_task() might have already changed |
4562 | * referenced cgroup group to be removed and freed. | 4562 | * current->cgroups, allowing the previously referenced cgroup |
4563 | * group to be removed and freed. | ||
4564 | * | ||
4565 | * Outside the pointer validity we also need to process the css_set | ||
4566 | * inheritance between threadgoup_change_begin() and | ||
4567 | * threadgoup_change_end(), this way there is no leak in any process | ||
4568 | * wide migration performed by cgroup_attach_proc() that could otherwise | ||
4569 | * miss a thread because it is too early or too late in the fork stage. | ||
4563 | * | 4570 | * |
4564 | * At the point that cgroup_fork() is called, 'current' is the parent | 4571 | * At the point that cgroup_fork() is called, 'current' is the parent |
4565 | * task, and the passed argument 'child' points to the child task. | 4572 | * task, and the passed argument 'child' points to the child task. |
4566 | */ | 4573 | */ |
4567 | void cgroup_fork(struct task_struct *child) | 4574 | void cgroup_fork(struct task_struct *child) |
4568 | { | 4575 | { |
4569 | task_lock(current); | 4576 | /* |
4577 | * We don't need to task_lock() current because current->cgroups | ||
4578 | * can't be changed concurrently here. The parent obviously hasn't | ||
4579 | * exited and called cgroup_exit(), and we are synchronized against | ||
4580 | * cgroup migration through threadgroup_change_begin(). | ||
4581 | */ | ||
4570 | child->cgroups = current->cgroups; | 4582 | child->cgroups = current->cgroups; |
4571 | get_css_set(child->cgroups); | 4583 | get_css_set(child->cgroups); |
4572 | task_unlock(current); | ||
4573 | INIT_LIST_HEAD(&child->cg_list); | 4584 | INIT_LIST_HEAD(&child->cg_list); |
4574 | } | 4585 | } |
4575 | 4586 | ||