diff options
author | Tejun Heo <tj@kernel.org> | 2013-08-13 11:01:54 -0400 |
---|---|---|
committer | Tejun Heo <tj@kernel.org> | 2013-08-13 11:01:54 -0400 |
commit | 0ae78e0bf10ac38ab53548e18383afc9997eca22 (patch) | |
tree | 8f3e828b1751966af779becd60a0ff432cfefc8d | |
parent | 35ef10da65d43211f4cd7e7822cbb3becdfc0ae1 (diff) |
cgroup: add cgroup_subsys_state->parent
With the planned unified hierarchy, css's (cgroup_subsys_state) will
be RCU protected and allowed to be attached and detached dynamically
over the course of a cgroup's lifetime. This means that css's will
stay accessible after being detached from its cgroup - the matching
pointer in cgroup->subsys[] cleared - for ref draining and RCU grace
period.
cgroup core still wants to guarantee that the parent css is never
destroyed before its children and css_parent() always returns the
parent regardless of the state of the child css as long as it's
accessible.
This patch makes css's hold onto their parents and adds css->parent so
that the parent css is never detroyed before its children and can be
determined without consulting the cgroups.
cgroup->dummy_css is also updated to point to the parent dummy_css;
however, it doesn't need to worry about object lifetime as the parent
cgroup is already pinned by the child.
Signed-off-by: Tejun Heo <tj@kernel.org>
Acked-by: Li Zefan <lizefan@huawei.com>
-rw-r--r-- | include/linux/cgroup.h | 13 | ||||
-rw-r--r-- | kernel/cgroup.c | 18 |
2 files changed, 19 insertions, 12 deletions
diff --git a/include/linux/cgroup.h b/include/linux/cgroup.h index 12d66fee26f8..8a5dc91fbaad 100644 --- a/include/linux/cgroup.h +++ b/include/linux/cgroup.h | |||
@@ -75,6 +75,9 @@ struct cgroup_subsys_state { | |||
75 | /* reference count - access via css_[try]get() and css_put() */ | 75 | /* reference count - access via css_[try]get() and css_put() */ |
76 | struct percpu_ref refcnt; | 76 | struct percpu_ref refcnt; |
77 | 77 | ||
78 | /* the parent css */ | ||
79 | struct cgroup_subsys_state *parent; | ||
80 | |||
78 | unsigned long flags; | 81 | unsigned long flags; |
79 | /* ID for this css, if possible */ | 82 | /* ID for this css, if possible */ |
80 | struct css_id __rcu *id; | 83 | struct css_id __rcu *id; |
@@ -666,15 +669,7 @@ struct cgroup_subsys { | |||
666 | static inline | 669 | static inline |
667 | struct cgroup_subsys_state *css_parent(struct cgroup_subsys_state *css) | 670 | struct cgroup_subsys_state *css_parent(struct cgroup_subsys_state *css) |
668 | { | 671 | { |
669 | struct cgroup *parent_cgrp = css->cgroup->parent; | 672 | return css->parent; |
670 | |||
671 | if (!parent_cgrp) | ||
672 | return NULL; | ||
673 | |||
674 | if (css->ss) | ||
675 | return parent_cgrp->subsys[css->ss->subsys_id]; | ||
676 | else | ||
677 | return &parent_cgrp->dummy_css; | ||
678 | } | 673 | } |
679 | 674 | ||
680 | /** | 675 | /** |
diff --git a/kernel/cgroup.c b/kernel/cgroup.c index 0b280978f097..5c6dd7ed26a7 100644 --- a/kernel/cgroup.c +++ b/kernel/cgroup.c | |||
@@ -4264,6 +4264,9 @@ static void css_free_work_fn(struct work_struct *work) | |||
4264 | struct cgroup_subsys_state *css = | 4264 | struct cgroup_subsys_state *css = |
4265 | container_of(work, struct cgroup_subsys_state, destroy_work); | 4265 | container_of(work, struct cgroup_subsys_state, destroy_work); |
4266 | 4266 | ||
4267 | if (css->parent) | ||
4268 | css_put(css->parent); | ||
4269 | |||
4267 | cgroup_dput(css->cgroup); | 4270 | cgroup_dput(css->cgroup); |
4268 | } | 4271 | } |
4269 | 4272 | ||
@@ -4290,8 +4293,12 @@ static void init_cgroup_css(struct cgroup_subsys_state *css, | |||
4290 | css->ss = ss; | 4293 | css->ss = ss; |
4291 | css->flags = 0; | 4294 | css->flags = 0; |
4292 | css->id = NULL; | 4295 | css->id = NULL; |
4293 | if (cgrp == cgroup_dummy_top) | 4296 | |
4297 | if (cgrp->parent) | ||
4298 | css->parent = cgroup_css(cgrp->parent, ss->subsys_id); | ||
4299 | else | ||
4294 | css->flags |= CSS_ROOT; | 4300 | css->flags |= CSS_ROOT; |
4301 | |||
4295 | BUG_ON(cgroup_css(cgrp, ss->subsys_id)); | 4302 | BUG_ON(cgroup_css(cgrp, ss->subsys_id)); |
4296 | cgrp->subsys[ss->subsys_id] = css; | 4303 | cgrp->subsys[ss->subsys_id] = css; |
4297 | } | 4304 | } |
@@ -4388,6 +4395,7 @@ static long cgroup_create(struct cgroup *parent, struct dentry *dentry, | |||
4388 | cgrp->dentry = dentry; | 4395 | cgrp->dentry = dentry; |
4389 | 4396 | ||
4390 | cgrp->parent = parent; | 4397 | cgrp->parent = parent; |
4398 | cgrp->dummy_css.parent = &parent->dummy_css; | ||
4391 | cgrp->root = parent->root; | 4399 | cgrp->root = parent->root; |
4392 | 4400 | ||
4393 | if (notify_on_release(parent)) | 4401 | if (notify_on_release(parent)) |
@@ -4436,9 +4444,13 @@ static long cgroup_create(struct cgroup *parent, struct dentry *dentry, | |||
4436 | list_add_tail_rcu(&cgrp->sibling, &cgrp->parent->children); | 4444 | list_add_tail_rcu(&cgrp->sibling, &cgrp->parent->children); |
4437 | root->number_of_cgroups++; | 4445 | root->number_of_cgroups++; |
4438 | 4446 | ||
4439 | /* each css holds a ref to the cgroup's dentry */ | 4447 | /* each css holds a ref to the cgroup's dentry and the parent css */ |
4440 | for_each_root_subsys(root, ss) | 4448 | for_each_root_subsys(root, ss) { |
4449 | struct cgroup_subsys_state *css = cgroup_css(cgrp, ss->subsys_id); | ||
4450 | |||
4441 | dget(dentry); | 4451 | dget(dentry); |
4452 | percpu_ref_get(&css->parent->refcnt); | ||
4453 | } | ||
4442 | 4454 | ||
4443 | /* hold a ref to the parent's dentry */ | 4455 | /* hold a ref to the parent's dentry */ |
4444 | dget(parent->dentry); | 4456 | dget(parent->dentry); |