|
@@ -97,6 +97,12 @@ static DEFINE_SPINLOCK(css_set_lock);
|
|
|
*/
|
|
|
static DEFINE_SPINLOCK(cgroup_idr_lock);
|
|
|
|
|
|
+/*
|
|
|
+ * Protects cgroup_file->kn for !self csses. It synchronizes notifications
|
|
|
+ * against file removal/re-creation across css hiding.
|
|
|
+ */
|
|
|
+static DEFINE_SPINLOCK(cgroup_file_kn_lock);
|
|
|
+
|
|
|
/*
|
|
|
* Protects cgroup_subsys->release_agent_path. Modifying it also requires
|
|
|
* cgroup_mutex. Reading requires either cgroup_mutex or this spinlock.
|
|
@@ -1393,6 +1399,16 @@ static void cgroup_rm_file(struct cgroup *cgrp, const struct cftype *cft)
|
|
|
char name[CGROUP_FILE_NAME_MAX];
|
|
|
|
|
|
lockdep_assert_held(&cgroup_mutex);
|
|
|
+
|
|
|
+ if (cft->file_offset) {
|
|
|
+ struct cgroup_subsys_state *css = cgroup_css(cgrp, cft->ss);
|
|
|
+ struct cgroup_file *cfile = (void *)css + cft->file_offset;
|
|
|
+
|
|
|
+ spin_lock_irq(&cgroup_file_kn_lock);
|
|
|
+ cfile->kn = NULL;
|
|
|
+ spin_unlock_irq(&cgroup_file_kn_lock);
|
|
|
+ }
|
|
|
+
|
|
|
kernfs_remove_by_name(cgrp->kn, cgroup_file_name(cgrp, cft, name));
|
|
|
}
|
|
|
|
|
@@ -1856,7 +1872,6 @@ static void init_cgroup_housekeeping(struct cgroup *cgrp)
|
|
|
|
|
|
INIT_LIST_HEAD(&cgrp->self.sibling);
|
|
|
INIT_LIST_HEAD(&cgrp->self.children);
|
|
|
- INIT_LIST_HEAD(&cgrp->self.files);
|
|
|
INIT_LIST_HEAD(&cgrp->cset_links);
|
|
|
INIT_LIST_HEAD(&cgrp->pidlists);
|
|
|
mutex_init(&cgrp->pidlist_mutex);
|
|
@@ -3313,9 +3328,9 @@ static int cgroup_add_file(struct cgroup_subsys_state *css, struct cgroup *cgrp,
|
|
|
if (cft->file_offset) {
|
|
|
struct cgroup_file *cfile = (void *)css + cft->file_offset;
|
|
|
|
|
|
- kernfs_get(kn);
|
|
|
+ spin_lock_irq(&cgroup_file_kn_lock);
|
|
|
cfile->kn = kn;
|
|
|
- list_add(&cfile->node, &css->files);
|
|
|
+ spin_unlock_irq(&cgroup_file_kn_lock);
|
|
|
}
|
|
|
|
|
|
return 0;
|
|
@@ -3552,6 +3567,22 @@ int cgroup_add_legacy_cftypes(struct cgroup_subsys *ss, struct cftype *cfts)
|
|
|
return cgroup_add_cftypes(ss, cfts);
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * cgroup_file_notify - generate a file modified event for a cgroup_file
|
|
|
+ * @cfile: target cgroup_file
|
|
|
+ *
|
|
|
+ * @cfile must have been obtained by setting cftype->file_offset.
|
|
|
+ */
|
|
|
+void cgroup_file_notify(struct cgroup_file *cfile)
|
|
|
+{
|
|
|
+ unsigned long flags;
|
|
|
+
|
|
|
+ spin_lock_irqsave(&cgroup_file_kn_lock, flags);
|
|
|
+ if (cfile->kn)
|
|
|
+ kernfs_notify(cfile->kn);
|
|
|
+ spin_unlock_irqrestore(&cgroup_file_kn_lock, flags);
|
|
|
+}
|
|
|
+
|
|
|
/**
|
|
|
* cgroup_task_count - count the number of tasks in a cgroup.
|
|
|
* @cgrp: the cgroup in question
|
|
@@ -4613,13 +4644,9 @@ static void css_free_work_fn(struct work_struct *work)
|
|
|
container_of(work, struct cgroup_subsys_state, destroy_work);
|
|
|
struct cgroup_subsys *ss = css->ss;
|
|
|
struct cgroup *cgrp = css->cgroup;
|
|
|
- struct cgroup_file *cfile;
|
|
|
|
|
|
percpu_ref_exit(&css->refcnt);
|
|
|
|
|
|
- list_for_each_entry(cfile, &css->files, node)
|
|
|
- kernfs_put(cfile->kn);
|
|
|
-
|
|
|
if (ss) {
|
|
|
/* css free path */
|
|
|
int id = css->id;
|
|
@@ -4724,7 +4751,6 @@ static void init_and_link_css(struct cgroup_subsys_state *css,
|
|
|
css->ss = ss;
|
|
|
INIT_LIST_HEAD(&css->sibling);
|
|
|
INIT_LIST_HEAD(&css->children);
|
|
|
- INIT_LIST_HEAD(&css->files);
|
|
|
css->serial_nr = css_serial_nr_next++;
|
|
|
|
|
|
if (cgroup_parent(cgrp)) {
|