Commit c2931b70 authored by Tejun Heo's avatar Tejun Heo

cgroup: iterate cgroup_subsys_states directly

Currently, css_next_child() is implemented as finding the next child
cgroup which has the css enabled, which used to be the only way to do
it as only cgroups participated in sibling lists and thus could be
iteratd.  This works as long as what's required during iteration is
not missing online csses; however, it turns out that there are use
cases where offlined but not yet released csses need to be iterated.
This is difficult to implement through cgroup iteration the unified
hierarchy as there may be multiple dying csses for the same subsystem
associated with single cgroup.

After the recent changes, the cgroup self and regular csses behave
identically in how they're linked and unlinked from the sibling lists
including assertion of CSS_RELEASED and css_next_child() can simply
switch to iterating csses directly.  This both simplifies the logic
and ensures that all visible non-released csses are included in the
iteration whether there are multiple dying csses for a subsystem or
not.

As all other iterators depend on css_next_child() for sibling
iteration, this changes behaviors of all css iterators.  Add and
update explanations on the css states which are included in traversal
to all iterators.

As css iteration could always contain offlined csses, this shouldn't
break any of the current users and new usages which need iteration of
all on and offline csses can make use of the new semantics.
Signed-off-by: default avatarTejun Heo <tj@kernel.org>
Acked-by: default avatarLi Zefan <lizefan@huawei.com>
Cc: Johannes Weiner <hannes@cmpxchg.org>
parent de3f0341
...@@ -764,14 +764,14 @@ struct cgroup_subsys_state *css_from_id(int id, struct cgroup_subsys *ss); ...@@ -764,14 +764,14 @@ struct cgroup_subsys_state *css_from_id(int id, struct cgroup_subsys *ss);
* @pos: the css * to use as the loop cursor * @pos: the css * to use as the loop cursor
* @parent: css whose children to walk * @parent: css whose children to walk
* *
* Walk @parent's children. Must be called under rcu_read_lock(). A child * Walk @parent's children. Must be called under rcu_read_lock().
* css which hasn't finished ->css_online() or already has finished
* ->css_offline() may show up during traversal and it's each subsystem's
* responsibility to verify that each @pos is alive.
* *
* If a subsystem synchronizes against the parent in its ->css_online() and * If a subsystem synchronizes ->css_online() and the start of iteration, a
* before starting iterating, a css which finished ->css_online() is * css which finished ->css_online() is guaranteed to be visible in the
* guaranteed to be visible in the future iterations. * future iterations and will stay visible until the last reference is put.
* A css which hasn't finished ->css_online() or already finished
* ->css_offline() may show up during traversal. It's each subsystem's
* responsibility to synchronize against on/offlining.
* *
* It is allowed to temporarily drop RCU read lock during iteration. The * It is allowed to temporarily drop RCU read lock during iteration. The
* caller is responsible for ensuring that @pos remains accessible until * caller is responsible for ensuring that @pos remains accessible until
...@@ -794,17 +794,16 @@ css_rightmost_descendant(struct cgroup_subsys_state *pos); ...@@ -794,17 +794,16 @@ css_rightmost_descendant(struct cgroup_subsys_state *pos);
* @root: css whose descendants to walk * @root: css whose descendants to walk
* *
* Walk @root's descendants. @root is included in the iteration and the * Walk @root's descendants. @root is included in the iteration and the
* first node to be visited. Must be called under rcu_read_lock(). A * first node to be visited. Must be called under rcu_read_lock().
* descendant css which hasn't finished ->css_online() or already has
* finished ->css_offline() may show up during traversal and it's each
* subsystem's responsibility to verify that each @pos is alive.
* *
* If a subsystem synchronizes against the parent in its ->css_online() and * If a subsystem synchronizes ->css_online() and the start of iteration, a
* before starting iterating, and synchronizes against @pos on each * css which finished ->css_online() is guaranteed to be visible in the
* iteration, any descendant css which finished ->css_online() is * future iterations and will stay visible until the last reference is put.
* guaranteed to be visible in the future iterations. * A css which hasn't finished ->css_online() or already finished
* ->css_offline() may show up during traversal. It's each subsystem's
* responsibility to synchronize against on/offlining.
* *
* In other words, the following guarantees that a descendant can't escape * For example, the following guarantees that a descendant can't escape
* state updates of its ancestors. * state updates of its ancestors.
* *
* my_online(@css) * my_online(@css)
...@@ -860,8 +859,17 @@ css_next_descendant_post(struct cgroup_subsys_state *pos, ...@@ -860,8 +859,17 @@ css_next_descendant_post(struct cgroup_subsys_state *pos,
* *
* Similar to css_for_each_descendant_pre() but performs post-order * Similar to css_for_each_descendant_pre() but performs post-order
* traversal instead. @root is included in the iteration and the last * traversal instead. @root is included in the iteration and the last
* node to be visited. Note that the walk visibility guarantee described * node to be visited.
* in pre-order walk doesn't apply the same to post-order walks. *
* If a subsystem synchronizes ->css_online() and the start of iteration, a
* css which finished ->css_online() is guaranteed to be visible in the
* future iterations and will stay visible until the last reference is put.
* A css which hasn't finished ->css_online() or already finished
* ->css_offline() may show up during traversal. It's each subsystem's
* responsibility to synchronize against on/offlining.
*
* Note that the walk visibility guarantee example described in pre-order
* walk doesn't apply the same to post-order walks.
*/ */
#define css_for_each_descendant_post(pos, css) \ #define css_for_each_descendant_post(pos, css) \
for ((pos) = css_next_descendant_post(NULL, (css)); (pos); \ for ((pos) = css_next_descendant_post(NULL, (css)); (pos); \
......
...@@ -3089,21 +3089,25 @@ static int cgroup_task_count(const struct cgroup *cgrp) ...@@ -3089,21 +3089,25 @@ static int cgroup_task_count(const struct cgroup *cgrp)
/** /**
* css_next_child - find the next child of a given css * css_next_child - find the next child of a given css
* @pos_css: the current position (%NULL to initiate traversal) * @pos: the current position (%NULL to initiate traversal)
* @parent_css: css whose children to walk * @parent: css whose children to walk
* *
* This function returns the next child of @parent_css and should be called * This function returns the next child of @parent and should be called
* under either cgroup_mutex or RCU read lock. The only requirement is * under either cgroup_mutex or RCU read lock. The only requirement is
* that @parent_css and @pos_css are accessible. The next sibling is * that @parent and @pos are accessible. The next sibling is guaranteed to
* guaranteed to be returned regardless of their states. * be returned regardless of their states.
*
* If a subsystem synchronizes ->css_online() and the start of iteration, a
* css which finished ->css_online() is guaranteed to be visible in the
* future iterations and will stay visible until the last reference is put.
* A css which hasn't finished ->css_online() or already finished
* ->css_offline() may show up during traversal. It's each subsystem's
* responsibility to synchronize against on/offlining.
*/ */
struct cgroup_subsys_state * struct cgroup_subsys_state *css_next_child(struct cgroup_subsys_state *pos,
css_next_child(struct cgroup_subsys_state *pos_css, struct cgroup_subsys_state *parent)
struct cgroup_subsys_state *parent_css)
{ {
struct cgroup *pos = pos_css ? pos_css->cgroup : NULL; struct cgroup_subsys_state *next;
struct cgroup *cgrp = parent_css->cgroup;
struct cgroup *next;
cgroup_assert_mutex_or_rcu_locked(); cgroup_assert_mutex_or_rcu_locked();
...@@ -3128,27 +3132,21 @@ css_next_child(struct cgroup_subsys_state *pos_css, ...@@ -3128,27 +3132,21 @@ css_next_child(struct cgroup_subsys_state *pos_css,
* races against release and the race window is very small. * races against release and the race window is very small.
*/ */
if (!pos) { if (!pos) {
next = list_entry_rcu(cgrp->self.children.next, struct cgroup, self.sibling); next = list_entry_rcu(parent->children.next, struct cgroup_subsys_state, sibling);
} else if (likely(!(pos->self.flags & CSS_RELEASED))) { } else if (likely(!(pos->flags & CSS_RELEASED))) {
next = list_entry_rcu(pos->self.sibling.next, struct cgroup, self.sibling); next = list_entry_rcu(pos->sibling.next, struct cgroup_subsys_state, sibling);
} else { } else {
list_for_each_entry_rcu(next, &cgrp->self.children, self.sibling) list_for_each_entry_rcu(next, &parent->children, sibling)
if (next->self.serial_nr > pos->self.serial_nr) if (next->serial_nr > pos->serial_nr)
break; break;
} }
/* /*
* @next, if not pointing to the head, can be dereferenced and is * @next, if not pointing to the head, can be dereferenced and is
* the next sibling; however, it might have @ss disabled. If so, * the next sibling.
* fast-forward to the next enabled one.
*/ */
while (&next->self.sibling != &cgrp->self.children) { if (&next->sibling != &parent->children)
struct cgroup_subsys_state *next_css = cgroup_css(next, parent_css->ss); return next;
if (next_css)
return next_css;
next = list_entry_rcu(next->self.sibling.next, struct cgroup, self.sibling);
}
return NULL; return NULL;
} }
...@@ -3165,6 +3163,13 @@ css_next_child(struct cgroup_subsys_state *pos_css, ...@@ -3165,6 +3163,13 @@ css_next_child(struct cgroup_subsys_state *pos_css,
* doesn't require the whole traversal to be contained in a single critical * doesn't require the whole traversal to be contained in a single critical
* section. This function will return the correct next descendant as long * section. This function will return the correct next descendant as long
* as both @pos and @root are accessible and @pos is a descendant of @root. * as both @pos and @root are accessible and @pos is a descendant of @root.
*
* If a subsystem synchronizes ->css_online() and the start of iteration, a
* css which finished ->css_online() is guaranteed to be visible in the
* future iterations and will stay visible until the last reference is put.
* A css which hasn't finished ->css_online() or already finished
* ->css_offline() may show up during traversal. It's each subsystem's
* responsibility to synchronize against on/offlining.
*/ */
struct cgroup_subsys_state * struct cgroup_subsys_state *
css_next_descendant_pre(struct cgroup_subsys_state *pos, css_next_descendant_pre(struct cgroup_subsys_state *pos,
...@@ -3252,6 +3257,13 @@ css_leftmost_descendant(struct cgroup_subsys_state *pos) ...@@ -3252,6 +3257,13 @@ css_leftmost_descendant(struct cgroup_subsys_state *pos)
* section. This function will return the correct next descendant as long * section. This function will return the correct next descendant as long
* as both @pos and @cgroup are accessible and @pos is a descendant of * as both @pos and @cgroup are accessible and @pos is a descendant of
* @cgroup. * @cgroup.
*
* If a subsystem synchronizes ->css_online() and the start of iteration, a
* css which finished ->css_online() is guaranteed to be visible in the
* future iterations and will stay visible until the last reference is put.
* A css which hasn't finished ->css_online() or already finished
* ->css_offline() may show up during traversal. It's each subsystem's
* responsibility to synchronize against on/offlining.
*/ */
struct cgroup_subsys_state * struct cgroup_subsys_state *
css_next_descendant_post(struct cgroup_subsys_state *pos, css_next_descendant_post(struct cgroup_subsys_state *pos,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment