aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--kernel/cgroup.c40
1 files changed, 25 insertions, 15 deletions
diff --git a/kernel/cgroup.c b/kernel/cgroup.c
index 5679cb1ce43f..4412d9694f13 100644
--- a/kernel/cgroup.c
+++ b/kernel/cgroup.c
@@ -242,6 +242,8 @@ static DEFINE_SPINLOCK(hierarchy_id_lock);
242 */ 242 */
243static int need_forkexit_callback __read_mostly; 243static int need_forkexit_callback __read_mostly;
244 244
245static int cgroup_destroy_locked(struct cgroup *cgrp);
246
245#ifdef CONFIG_PROVE_LOCKING 247#ifdef CONFIG_PROVE_LOCKING
246int cgroup_lock_is_held(void) 248int cgroup_lock_is_held(void)
247{ 249{
@@ -4209,22 +4211,20 @@ static int cgroup_has_css_refs(struct cgroup *cgrp)
4209 return 0; 4211 return 0;
4210} 4212}
4211 4213
4212static int cgroup_rmdir(struct inode *unused_dir, struct dentry *dentry) 4214static int cgroup_destroy_locked(struct cgroup *cgrp)
4215 __releases(&cgroup_mutex) __acquires(&cgroup_mutex)
4213{ 4216{
4214 struct cgroup *cgrp = dentry->d_fsdata; 4217 struct dentry *d = cgrp->dentry;
4215 struct dentry *d; 4218 struct cgroup *parent = cgrp->parent;
4216 struct cgroup *parent;
4217 DEFINE_WAIT(wait); 4219 DEFINE_WAIT(wait);
4218 struct cgroup_event *event, *tmp; 4220 struct cgroup_event *event, *tmp;
4219 struct cgroup_subsys *ss; 4221 struct cgroup_subsys *ss;
4220 4222
4221 /* the vfs holds both inode->i_mutex already */ 4223 lockdep_assert_held(&d->d_inode->i_mutex);
4222 mutex_lock(&cgroup_mutex); 4224 lockdep_assert_held(&cgroup_mutex);
4223 parent = cgrp->parent; 4225
4224 if (atomic_read(&cgrp->count) || !list_empty(&cgrp->children)) { 4226 if (atomic_read(&cgrp->count) || !list_empty(&cgrp->children))
4225 mutex_unlock(&cgroup_mutex);
4226 return -EBUSY; 4227 return -EBUSY;
4227 }
4228 4228
4229 /* 4229 /*
4230 * Block new css_tryget() by deactivating refcnt and mark @cgrp 4230 * Block new css_tryget() by deactivating refcnt and mark @cgrp
@@ -4243,7 +4243,9 @@ static int cgroup_rmdir(struct inode *unused_dir, struct dentry *dentry)
4243 /* 4243 /*
4244 * Tell subsystems to initate destruction. pre_destroy() should be 4244 * Tell subsystems to initate destruction. pre_destroy() should be
4245 * called with cgroup_mutex unlocked. See 3fa59dfbc3 ("cgroup: fix 4245 * called with cgroup_mutex unlocked. See 3fa59dfbc3 ("cgroup: fix
4246 * potential deadlock in pre_destroy") for details. 4246 * potential deadlock in pre_destroy") for details. This temporary
4247 * unlocking should go away once cgroup_mutex is unexported from
4248 * controllers.
4247 */ 4249 */
4248 mutex_unlock(&cgroup_mutex); 4250 mutex_unlock(&cgroup_mutex);
4249 for_each_subsys(cgrp->root, ss) 4251 for_each_subsys(cgrp->root, ss)
@@ -4268,11 +4270,9 @@ static int cgroup_rmdir(struct inode *unused_dir, struct dentry *dentry)
4268 4270
4269 /* delete this cgroup from parent->children */ 4271 /* delete this cgroup from parent->children */
4270 list_del_rcu(&cgrp->sibling); 4272 list_del_rcu(&cgrp->sibling);
4271
4272 list_del_init(&cgrp->allcg_node); 4273 list_del_init(&cgrp->allcg_node);
4273 4274
4274 d = dget(cgrp->dentry); 4275 dget(d);
4275
4276 cgroup_d_remove_dir(d); 4276 cgroup_d_remove_dir(d);
4277 dput(d); 4277 dput(d);
4278 4278
@@ -4293,10 +4293,20 @@ static int cgroup_rmdir(struct inode *unused_dir, struct dentry *dentry)
4293 } 4293 }
4294 spin_unlock(&cgrp->event_list_lock); 4294 spin_unlock(&cgrp->event_list_lock);
4295 4295
4296 mutex_unlock(&cgroup_mutex);
4297 return 0; 4296 return 0;
4298} 4297}
4299 4298
4299static int cgroup_rmdir(struct inode *unused_dir, struct dentry *dentry)
4300{
4301 int ret;
4302
4303 mutex_lock(&cgroup_mutex);
4304 ret = cgroup_destroy_locked(dentry->d_fsdata);
4305 mutex_unlock(&cgroup_mutex);
4306
4307 return ret;
4308}
4309
4300static void __init_or_module cgroup_init_cftsets(struct cgroup_subsys *ss) 4310static void __init_or_module cgroup_init_cftsets(struct cgroup_subsys *ss)
4301{ 4311{
4302 INIT_LIST_HEAD(&ss->cftsets); 4312 INIT_LIST_HEAD(&ss->cftsets);