Commit b0e8c75e authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: Fix subvol deletion deadlock

d_prune_aliases() may call bch2_evict_inode(), which needs
c->vfs_inodes_list_lock.

Fix this by always calling igrab() before putting the inodes onto our
disposal list, and then calling d_prune_aliases() with
c->vfs_inodes_lock dropped.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 5bc74082
...@@ -1481,22 +1481,14 @@ void bch2_evict_subvolume_inodes(struct bch_fs *c, snapshot_id_list *s) ...@@ -1481,22 +1481,14 @@ void bch2_evict_subvolume_inodes(struct bch_fs *c, snapshot_id_list *s)
continue; continue;
if (!(inode->v.i_state & I_DONTCACHE) && if (!(inode->v.i_state & I_DONTCACHE) &&
!(inode->v.i_state & I_FREEING)) { !(inode->v.i_state & I_FREEING) &&
igrab(&inode->v)) {
this_pass_clean = false; this_pass_clean = false;
d_mark_dontcache(&inode->v); if (darray_push_gfp(&grabbed, inode, GFP_ATOMIC|__GFP_NOWARN)) {
d_prune_aliases(&inode->v); iput(&inode->v);
/*
* If i_count was zero, we have to take and release a
* ref in order for I_DONTCACHE to be noticed and the
* inode to be dropped;
*/
if (!atomic_read(&inode->v.i_count) &&
igrab(&inode->v) &&
darray_push_gfp(&grabbed, inode, GFP_ATOMIC|__GFP_NOWARN))
break; break;
}
} else if (clean_pass && this_pass_clean) { } else if (clean_pass && this_pass_clean) {
wait_queue_head_t *wq = bit_waitqueue(&inode->v.i_state, __I_NEW); wait_queue_head_t *wq = bit_waitqueue(&inode->v.i_state, __I_NEW);
DEFINE_WAIT_BIT(wait, &inode->v.i_state, __I_NEW); DEFINE_WAIT_BIT(wait, &inode->v.i_state, __I_NEW);
...@@ -1511,8 +1503,12 @@ void bch2_evict_subvolume_inodes(struct bch_fs *c, snapshot_id_list *s) ...@@ -1511,8 +1503,12 @@ void bch2_evict_subvolume_inodes(struct bch_fs *c, snapshot_id_list *s)
} }
mutex_unlock(&c->vfs_inodes_lock); mutex_unlock(&c->vfs_inodes_lock);
darray_for_each(grabbed, i) darray_for_each(grabbed, i) {
iput(&(*i)->v); inode = *i;
d_mark_dontcache(&inode->v);
d_prune_aliases(&inode->v);
iput(&inode->v);
}
grabbed.nr = 0; grabbed.nr = 0;
if (!clean_pass || !this_pass_clean) { if (!clean_pass || !this_pass_clean) {
......
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