diff options
| -rw-r--r-- | drivers/md/bcache/btree.c | 11 | ||||
| -rw-r--r-- | drivers/md/bcache/btree.h | 1 | ||||
| -rw-r--r-- | drivers/md/bcache/writeback.c | 10 | 
3 files changed, 21 insertions, 1 deletions
| diff --git a/drivers/md/bcache/btree.c b/drivers/md/bcache/btree.c index 0ddf91204782..68b9d7ca864e 100644 --- a/drivers/md/bcache/btree.c +++ b/drivers/md/bcache/btree.c @@ -885,7 +885,7 @@ static struct btree *mca_cannibalize(struct cache_set *c, struct btree_op *op,   * cannibalize_bucket() will take. This means every time we unlock the root of   * the btree, we need to release this lock if we have it held.   */ -static void bch_cannibalize_unlock(struct cache_set *c) +void bch_cannibalize_unlock(struct cache_set *c)  {  	spin_lock(&c->btree_cannibalize_lock);  	if (c->btree_cache_alloc_lock == current) { @@ -1970,6 +1970,15 @@ static int bch_btree_check_thread(void *arg)  			c->gc_stats.nodes++;  			bch_btree_op_init(&op, 0);  			ret = bcache_btree(check_recurse, p, c->root, &op); +			/* +			 * The op may be added to cache_set's btree_cache_wait +			 * in mca_cannibalize(), must ensure it is removed from +			 * the list and release btree_cache_alloc_lock before +			 * free op memory. +			 * Otherwise, the btree_cache_wait will be damaged. +			 */ +			bch_cannibalize_unlock(c); +			finish_wait(&c->btree_cache_wait, &(&op)->wait);  			if (ret)  				goto out;  		} diff --git a/drivers/md/bcache/btree.h b/drivers/md/bcache/btree.h index 1b5fdbc0d83e..a2920bbfcad5 100644 --- a/drivers/md/bcache/btree.h +++ b/drivers/md/bcache/btree.h @@ -282,6 +282,7 @@ void bch_initial_gc_finish(struct cache_set *c);  void bch_moving_gc(struct cache_set *c);  int bch_btree_check(struct cache_set *c);  void bch_initial_mark_key(struct cache_set *c, int level, struct bkey *k); +void bch_cannibalize_unlock(struct cache_set *c);  static inline void wake_up_gc(struct cache_set *c)  { diff --git a/drivers/md/bcache/writeback.c b/drivers/md/bcache/writeback.c index d4a5fc0650bb..24c049067f61 100644 --- a/drivers/md/bcache/writeback.c +++ b/drivers/md/bcache/writeback.c @@ -890,6 +890,16 @@ static int bch_root_node_dirty_init(struct cache_set *c,  	if (ret < 0)  		pr_warn("sectors dirty init failed, ret=%d!\n", ret); +	/* +	 * The op may be added to cache_set's btree_cache_wait +	 * in mca_cannibalize(), must ensure it is removed from +	 * the list and release btree_cache_alloc_lock before +	 * free op memory. +	 * Otherwise, the btree_cache_wait will be damaged. +	 */ +	bch_cannibalize_unlock(c); +	finish_wait(&c->btree_cache_wait, &(&op.op)->wait); +  	return ret;  } | 
