Commit 6bb3f7f4 authored by Guoyu Ou's avatar Guoyu Ou Committed by Kent Overstreet

bcachefs: unlock parent dir if entry is not found in subvolume deletion

Parent dir is locked by user_path_locked_at() before validating the
required dentry. It should be unlocked if we can not perform the
deletion.

This fixes the problem:

$ bcachefs subvolume delete not-exist-entry
BCH_IOCTL_SUBVOLUME_DESTROY ioctl error: No such file or directory
$ bcachefs subvolume delete not-exist-entry

the second will stuck because the parent dir is locked in the previous
deletion.
Signed-off-by: default avatarGuoyu Ou <benogy@gmail.com>
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent eba38cc7
...@@ -455,6 +455,7 @@ static long bch2_ioctl_subvolume_destroy(struct bch_fs *c, struct file *filp, ...@@ -455,6 +455,7 @@ static long bch2_ioctl_subvolume_destroy(struct bch_fs *c, struct file *filp,
if (IS_ERR(victim)) if (IS_ERR(victim))
return PTR_ERR(victim); return PTR_ERR(victim);
dir = d_inode(path.dentry);
if (victim->d_sb->s_fs_info != c) { if (victim->d_sb->s_fs_info != c) {
ret = -EXDEV; ret = -EXDEV;
goto err; goto err;
...@@ -463,14 +464,13 @@ static long bch2_ioctl_subvolume_destroy(struct bch_fs *c, struct file *filp, ...@@ -463,14 +464,13 @@ static long bch2_ioctl_subvolume_destroy(struct bch_fs *c, struct file *filp,
ret = -ENOENT; ret = -ENOENT;
goto err; goto err;
} }
dir = d_inode(path.dentry);
ret = __bch2_unlink(dir, victim, true); ret = __bch2_unlink(dir, victim, true);
if (!ret) { if (!ret) {
fsnotify_rmdir(dir, victim); fsnotify_rmdir(dir, victim);
d_delete(victim); d_delete(victim);
} }
inode_unlock(dir);
err: err:
inode_unlock(dir);
dput(victim); dput(victim);
path_put(&path); path_put(&path);
return ret; return ret;
......
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