diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2012-10-24 19:35:13 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-10-24 19:35:13 -0400 |
commit | cbb525b447996f5d5702f7f2d7073f395fadcfe1 (patch) | |
tree | a4a8db1a3223b7410a25c52535ceaa4c52d443c1 /kernel | |
parent | d579a35d0e488770ec6fbe0ce5efe8a797cada2f (diff) | |
parent | d87838321124061f6c935069d97f37010fa417e6 (diff) |
Merge branch 'for-3.7-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/tj/cgroup
Pull cgroup fixes from Tejun Heo:
"This pull request contains three fixes.
Two are reverts of task_lock() removal in cgroup fork path. The
optimizations incorrectly assumed that threadgroup_lock can protect
process forks (as opposed to thread creations) too. Further cleanup
of cgroup fork path is scheduled.
The third fixes cgroup emptiness notification loss."
* 'for-3.7-fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/tj/cgroup:
Revert "cgroup: Remove task_lock() from cgroup_post_fork()"
Revert "cgroup: Drop task_lock(parent) on cgroup_fork()"
cgroup: notify_on_release may not be triggered in some cases
Diffstat (limited to 'kernel')
-rw-r--r-- | kernel/cgroup.c | 41 |
1 files changed, 10 insertions, 31 deletions
diff --git a/kernel/cgroup.c b/kernel/cgroup.c index 13774b3b39aa..f24f724620dd 100644 --- a/kernel/cgroup.c +++ b/kernel/cgroup.c | |||
@@ -1962,9 +1962,8 @@ static void cgroup_task_migrate(struct cgroup *cgrp, struct cgroup *oldcgrp, | |||
1962 | * trading it for newcg is protected by cgroup_mutex, we're safe to drop | 1962 | * trading it for newcg is protected by cgroup_mutex, we're safe to drop |
1963 | * it here; it will be freed under RCU. | 1963 | * it here; it will be freed under RCU. |
1964 | */ | 1964 | */ |
1965 | put_css_set(oldcg); | ||
1966 | |||
1967 | set_bit(CGRP_RELEASABLE, &oldcgrp->flags); | 1965 | set_bit(CGRP_RELEASABLE, &oldcgrp->flags); |
1966 | put_css_set(oldcg); | ||
1968 | } | 1967 | } |
1969 | 1968 | ||
1970 | /** | 1969 | /** |
@@ -4815,31 +4814,20 @@ static const struct file_operations proc_cgroupstats_operations = { | |||
4815 | * | 4814 | * |
4816 | * A pointer to the shared css_set was automatically copied in | 4815 | * A pointer to the shared css_set was automatically copied in |
4817 | * fork.c by dup_task_struct(). However, we ignore that copy, since | 4816 | * fork.c by dup_task_struct(). However, we ignore that copy, since |
4818 | * it was not made under the protection of RCU, cgroup_mutex or | 4817 | * it was not made under the protection of RCU or cgroup_mutex, so |
4819 | * threadgroup_change_begin(), so it might no longer be a valid | 4818 | * might no longer be a valid cgroup pointer. cgroup_attach_task() might |
4820 | * cgroup pointer. cgroup_attach_task() might have already changed | 4819 | * have already changed current->cgroups, allowing the previously |
4821 | * current->cgroups, allowing the previously referenced cgroup | 4820 | * referenced cgroup group to be removed and freed. |
4822 | * group to be removed and freed. | ||
4823 | * | ||
4824 | * Outside the pointer validity we also need to process the css_set | ||
4825 | * inheritance between threadgoup_change_begin() and | ||
4826 | * threadgoup_change_end(), this way there is no leak in any process | ||
4827 | * wide migration performed by cgroup_attach_proc() that could otherwise | ||
4828 | * miss a thread because it is too early or too late in the fork stage. | ||
4829 | * | 4821 | * |
4830 | * At the point that cgroup_fork() is called, 'current' is the parent | 4822 | * At the point that cgroup_fork() is called, 'current' is the parent |
4831 | * task, and the passed argument 'child' points to the child task. | 4823 | * task, and the passed argument 'child' points to the child task. |
4832 | */ | 4824 | */ |
4833 | void cgroup_fork(struct task_struct *child) | 4825 | void cgroup_fork(struct task_struct *child) |
4834 | { | 4826 | { |
4835 | /* | 4827 | task_lock(current); |
4836 | * We don't need to task_lock() current because current->cgroups | ||
4837 | * can't be changed concurrently here. The parent obviously hasn't | ||
4838 | * exited and called cgroup_exit(), and we are synchronized against | ||
4839 | * cgroup migration through threadgroup_change_begin(). | ||
4840 | */ | ||
4841 | child->cgroups = current->cgroups; | 4828 | child->cgroups = current->cgroups; |
4842 | get_css_set(child->cgroups); | 4829 | get_css_set(child->cgroups); |
4830 | task_unlock(current); | ||
4843 | INIT_LIST_HEAD(&child->cg_list); | 4831 | INIT_LIST_HEAD(&child->cg_list); |
4844 | } | 4832 | } |
4845 | 4833 | ||
@@ -4895,19 +4883,10 @@ void cgroup_post_fork(struct task_struct *child) | |||
4895 | */ | 4883 | */ |
4896 | if (use_task_css_set_links) { | 4884 | if (use_task_css_set_links) { |
4897 | write_lock(&css_set_lock); | 4885 | write_lock(&css_set_lock); |
4898 | if (list_empty(&child->cg_list)) { | 4886 | task_lock(child); |
4899 | /* | 4887 | if (list_empty(&child->cg_list)) |
4900 | * It's safe to use child->cgroups without task_lock() | ||
4901 | * here because we are protected through | ||
4902 | * threadgroup_change_begin() against concurrent | ||
4903 | * css_set change in cgroup_task_migrate(). Also | ||
4904 | * the task can't exit at that point until | ||
4905 | * wake_up_new_task() is called, so we are protected | ||
4906 | * against cgroup_exit() setting child->cgroup to | ||
4907 | * init_css_set. | ||
4908 | */ | ||
4909 | list_add(&child->cg_list, &child->cgroups->tasks); | 4888 | list_add(&child->cg_list, &child->cgroups->tasks); |
4910 | } | 4889 | task_unlock(child); |
4911 | write_unlock(&css_set_lock); | 4890 | write_unlock(&css_set_lock); |
4912 | } | 4891 | } |
4913 | } | 4892 | } |