Commit 2d33036c authored by Kent Overstreet's avatar Kent Overstreet

bcachefs: Fix for 'missing subvolume' error

Subvolumes, including their root inodes, get deleted asynchronously
after an unlink. But we still need to ensure that we tell the VFS the
inode has been deleted, otherwise VFS writeback could fire after
asynchronous deletion has finished, and try to write to an
inode/subvolume that no longer exists.
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 56cc033d
...@@ -442,19 +442,27 @@ int __bch2_unlink(struct inode *vdir, struct dentry *dentry, ...@@ -442,19 +442,27 @@ int __bch2_unlink(struct inode *vdir, struct dentry *dentry,
bch2_trans_init(&trans, c, 4, 1024); bch2_trans_init(&trans, c, 4, 1024);
ret = commit_do(&trans, NULL, NULL, ret = commit_do(&trans, NULL, NULL,
BTREE_INSERT_NOFAIL, BTREE_INSERT_NOFAIL,
bch2_unlink_trans(&trans, bch2_unlink_trans(&trans,
inode_inum(dir), &dir_u, inode_inum(dir), &dir_u,
&inode_u, &dentry->d_name, &inode_u, &dentry->d_name,
deleting_snapshot)); deleting_snapshot));
if (unlikely(ret))
goto err;
if (likely(!ret)) { bch2_inode_update_after_write(&trans, dir, &dir_u,
bch2_inode_update_after_write(&trans, dir, &dir_u, ATTR_MTIME|ATTR_CTIME);
ATTR_MTIME|ATTR_CTIME); bch2_inode_update_after_write(&trans, inode, &inode_u,
bch2_inode_update_after_write(&trans, inode, &inode_u, ATTR_MTIME);
ATTR_MTIME);
}
if (inode_u.bi_subvol) {
/*
* Subvolume deletion is asynchronous, but we still want to tell
* the VFS that it's been deleted here:
*/
set_nlink(&inode->v, 0);
}
err:
bch2_trans_exit(&trans); bch2_trans_exit(&trans);
bch2_unlock_inodes(INODE_UPDATE_LOCK, dir, inode); bch2_unlock_inodes(INODE_UPDATE_LOCK, dir, inode);
......
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