Commit f63070d3 authored by Tejun Heo's avatar Tejun Heo

cgroup: make interface files visible iff enabled on cgroup->subtree_control

cgroup is implementing support for subsystem dependency which would
require a way to enable a subsystem even when it's not directly
configured through "cgroup.subtree_control".

The preceding patch distinguished cgroup->subtree_control and
->child_subsys_mask where the former is the subsystems explicitly
configured by the userland and the latter is all enabled subsystems
currently is equal to the former but will include subsystems
implicitly enabled through dependency.

Subsystems which are enabled due to dependency shouldn't be visible to
userland.  This patch updates cgroup_subtree_control_write() and
create_css() such that interface files are not created for implicitly
enabled subsytems.

* @visible paramter is added to create_css().  Interface files are
  created only when true.

* If an already implicitly enabled subsystem is turned on through
  "cgroup.subtree_control", the existing css should be used.  css
  draining is skipped.

* cgroup_subtree_control_write() computes the new target
  cgroup->child_subsys_mask and create/kill or show/hide csses
  accordingly.

As the two subsystem masks are still kept identical, this patch
doesn't introduce any behavior changes.
Signed-off-by: default avatarTejun Heo <tj@kernel.org>
Acked-by: default avatarLi Zefan <lizefan@huawei.com>
Acked-by: default avatarJohannes Weiner <hannes@cmpxchg.org>
parent 667c2491
...@@ -208,6 +208,8 @@ struct cgroup { ...@@ -208,6 +208,8 @@ struct cgroup {
* ->subtree_control is the one configured through * ->subtree_control is the one configured through
* "cgroup.subtree_control" while ->child_subsys_mask is the * "cgroup.subtree_control" while ->child_subsys_mask is the
* effective one which may have more subsystems enabled. * effective one which may have more subsystems enabled.
* Controller knobs are made available iff it's enabled in
* ->subtree_control.
*/ */
unsigned int subtree_control; unsigned int subtree_control;
unsigned int child_subsys_mask; unsigned int child_subsys_mask;
......
...@@ -186,7 +186,8 @@ static void cgroup_put(struct cgroup *cgrp); ...@@ -186,7 +186,8 @@ static void cgroup_put(struct cgroup *cgrp);
static int rebind_subsystems(struct cgroup_root *dst_root, static int rebind_subsystems(struct cgroup_root *dst_root,
unsigned int ss_mask); unsigned int ss_mask);
static int cgroup_destroy_locked(struct cgroup *cgrp); static int cgroup_destroy_locked(struct cgroup *cgrp);
static int create_css(struct cgroup *cgrp, struct cgroup_subsys *ss); static int create_css(struct cgroup *cgrp, struct cgroup_subsys *ss,
bool visible);
static void css_release(struct percpu_ref *ref); static void css_release(struct percpu_ref *ref);
static void kill_css(struct cgroup_subsys_state *css); static void kill_css(struct cgroup_subsys_state *css);
static int cgroup_addrm_files(struct cgroup *cgrp, struct cftype cfts[], static int cgroup_addrm_files(struct cgroup *cgrp, struct cftype cfts[],
...@@ -2577,6 +2578,7 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of, ...@@ -2577,6 +2578,7 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of,
loff_t off) loff_t off)
{ {
unsigned int enable = 0, disable = 0; unsigned int enable = 0, disable = 0;
unsigned int css_enable, css_disable, old_ctrl, new_ctrl;
struct cgroup *cgrp, *child; struct cgroup *cgrp, *child;
struct cgroup_subsys *ss; struct cgroup_subsys *ss;
char *tok; char *tok;
...@@ -2629,6 +2631,13 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of, ...@@ -2629,6 +2631,13 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of,
goto out_unlock; goto out_unlock;
} }
/*
* @ss is already enabled through dependency and
* we'll just make it visible. Skip draining.
*/
if (cgrp->child_subsys_mask & (1 << ssid))
continue;
/* /*
* Because css offlining is asynchronous, userland * Because css offlining is asynchronous, userland
* might try to re-enable the same controller while * might try to re-enable the same controller while
...@@ -2681,17 +2690,39 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of, ...@@ -2681,17 +2690,39 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of,
goto out_unlock; goto out_unlock;
} }
/*
* Update subsys masks and calculate what needs to be done. More
* subsystems than specified may need to be enabled or disabled
* depending on subsystem dependencies.
*/
cgrp->subtree_control |= enable; cgrp->subtree_control |= enable;
cgrp->subtree_control &= ~disable; cgrp->subtree_control &= ~disable;
old_ctrl = cgrp->child_subsys_mask;
cgroup_refresh_child_subsys_mask(cgrp); cgroup_refresh_child_subsys_mask(cgrp);
new_ctrl = cgrp->child_subsys_mask;
css_enable = ~old_ctrl & new_ctrl;
css_disable = old_ctrl & ~new_ctrl;
enable |= css_enable;
disable |= css_disable;
/* create new csses */ /*
* Create new csses or make the existing ones visible. A css is
* created invisible if it's being implicitly enabled through
* dependency. An invisible css is made visible when the userland
* explicitly enables it.
*/
for_each_subsys(ss, ssid) { for_each_subsys(ss, ssid) {
if (!(enable & (1 << ssid))) if (!(enable & (1 << ssid)))
continue; continue;
cgroup_for_each_live_child(child, cgrp) { cgroup_for_each_live_child(child, cgrp) {
ret = create_css(child, ss); if (css_enable & (1 << ssid))
ret = create_css(child, ss,
cgrp->subtree_control & (1 << ssid));
else
ret = cgroup_populate_dir(child, 1 << ssid);
if (ret) if (ret)
goto err_undo_css; goto err_undo_css;
} }
...@@ -2706,13 +2737,21 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of, ...@@ -2706,13 +2737,21 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of,
if (ret) if (ret)
goto err_undo_css; goto err_undo_css;
/* all tasks are now migrated away from the old csses, kill them */ /*
* All tasks are migrated out of disabled csses. Kill or hide
* them. A css is hidden when the userland requests it to be
* disabled while other subsystems are still depending on it.
*/
for_each_subsys(ss, ssid) { for_each_subsys(ss, ssid) {
if (!(disable & (1 << ssid))) if (!(disable & (1 << ssid)))
continue; continue;
cgroup_for_each_live_child(child, cgrp) cgroup_for_each_live_child(child, cgrp) {
if (css_disable & (1 << ssid))
kill_css(cgroup_css(child, ss)); kill_css(cgroup_css(child, ss));
else
cgroup_clear_dir(child, 1 << ssid);
}
} }
kernfs_activate(cgrp->kn); kernfs_activate(cgrp->kn);
...@@ -2732,8 +2771,14 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of, ...@@ -2732,8 +2771,14 @@ static ssize_t cgroup_subtree_control_write(struct kernfs_open_file *of,
cgroup_for_each_live_child(child, cgrp) { cgroup_for_each_live_child(child, cgrp) {
struct cgroup_subsys_state *css = cgroup_css(child, ss); struct cgroup_subsys_state *css = cgroup_css(child, ss);
if (css)
if (!css)
continue;
if (css_enable & (1 << ssid))
kill_css(css); kill_css(css);
else
cgroup_clear_dir(child, 1 << ssid);
} }
} }
goto out_unlock; goto out_unlock;
...@@ -4282,12 +4327,14 @@ static void offline_css(struct cgroup_subsys_state *css) ...@@ -4282,12 +4327,14 @@ static void offline_css(struct cgroup_subsys_state *css)
* create_css - create a cgroup_subsys_state * create_css - create a cgroup_subsys_state
* @cgrp: the cgroup new css will be associated with * @cgrp: the cgroup new css will be associated with
* @ss: the subsys of new css * @ss: the subsys of new css
* @visible: whether to create control knobs for the new css or not
* *
* Create a new css associated with @cgrp - @ss pair. On success, the new * Create a new css associated with @cgrp - @ss pair. On success, the new
* css is online and installed in @cgrp with all interface files created. * css is online and installed in @cgrp with all interface files created if
* Returns 0 on success, -errno on failure. * @visible. Returns 0 on success, -errno on failure.
*/ */
static int create_css(struct cgroup *cgrp, struct cgroup_subsys *ss) static int create_css(struct cgroup *cgrp, struct cgroup_subsys *ss,
bool visible)
{ {
struct cgroup *parent = cgroup_parent(cgrp); struct cgroup *parent = cgroup_parent(cgrp);
struct cgroup_subsys_state *parent_css = cgroup_css(parent, ss); struct cgroup_subsys_state *parent_css = cgroup_css(parent, ss);
...@@ -4311,9 +4358,11 @@ static int create_css(struct cgroup *cgrp, struct cgroup_subsys *ss) ...@@ -4311,9 +4358,11 @@ static int create_css(struct cgroup *cgrp, struct cgroup_subsys *ss)
goto err_free_percpu_ref; goto err_free_percpu_ref;
css->id = err; css->id = err;
if (visible) {
err = cgroup_populate_dir(cgrp, 1 << ss->id); err = cgroup_populate_dir(cgrp, 1 << ss->id);
if (err) if (err)
goto err_free_id; goto err_free_id;
}
/* @css is ready to be brought online now, make it visible */ /* @css is ready to be brought online now, make it visible */
list_add_tail_rcu(&css->sibling, &parent_css->children); list_add_tail_rcu(&css->sibling, &parent_css->children);
...@@ -4430,7 +4479,8 @@ static int cgroup_mkdir(struct kernfs_node *parent_kn, const char *name, ...@@ -4430,7 +4479,8 @@ static int cgroup_mkdir(struct kernfs_node *parent_kn, const char *name,
/* let's create and online css's */ /* let's create and online css's */
for_each_subsys(ss, ssid) { for_each_subsys(ss, ssid) {
if (parent->child_subsys_mask & (1 << ssid)) { if (parent->child_subsys_mask & (1 << ssid)) {
ret = create_css(cgrp, ss); ret = create_css(cgrp, ss,
parent->subtree_control & (1 << ssid));
if (ret) if (ret)
goto out_destroy; goto out_destroy;
} }
......
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