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 | ||
