]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
btrfs: reloc: clean dirty subvols if we fail to start a transaction
authorJosef Bacik <josef@toxicpanda.com>
Wed, 4 Mar 2020 16:18:27 +0000 (11:18 -0500)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 17 Apr 2020 14:12:08 +0000 (16:12 +0200)
commit 6217b0fadd4473a16fabc6aecd7527a9f71af534 upstream.

If we do merge_reloc_roots() we could insert a few roots onto the dirty
subvol roots list, where we hold a ref on them.  If we fail to start the
transaction we need to run clean_dirty_subvols() in order to cleanup the
refs.

CC: stable@vger.kernel.org # 5.4+
Signed-off-by: Josef Bacik <josef@toxicpanda.com>
Reviewed-by: David Sterba <dsterba@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
fs/btrfs/relocation.c

index ddaefecaa7c423c046850a7f6927a19461af74a0..b976ae4c5fe037e7a38f746462e9af72dd8ca673 100644 (file)
@@ -4221,10 +4221,10 @@ restart:
                goto out_free;
        }
        btrfs_commit_transaction(trans);
+out_free:
        ret = clean_dirty_subvols(rc);
        if (ret < 0 && !err)
                err = ret;
-out_free:
        btrfs_free_block_rsv(fs_info, rc->block_rsv);
        btrfs_free_path(path);
        return err;
@@ -4622,10 +4622,10 @@ int btrfs_recover_relocation(struct btrfs_root *root)
        trans = btrfs_join_transaction(rc->extent_root);
        if (IS_ERR(trans)) {
                err = PTR_ERR(trans);
-               goto out_free;
+               goto out_clean;
        }
        err = btrfs_commit_transaction(trans);
-
+out_clean:
        ret = clean_dirty_subvols(rc);
        if (ret < 0 && !err)
                err = ret;