btrfs: handle errors in reference count manipulation in replace_path
authorJosef Bacik <josef@toxicpanda.com>
Fri, 12 Mar 2021 20:25:24 +0000 (15:25 -0500)
committerDavid Sterba <dsterba@suse.com>
Mon, 19 Apr 2021 15:25:21 +0000 (17:25 +0200)
If any of the reference count manipulation stuff fails in replace_path
we need to abort the transaction, as we've modified the blocks already.
We can simply break at this point and everything will be cleaned up.

Reviewed-by: Qu Wenruo <wqu@suse.com>
Signed-off-by: Josef Bacik <josef@toxicpanda.com>
Reviewed-by: David Sterba <dsterba@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
fs/btrfs/relocation.c

index b62f5e4..2663c4b 100644 (file)
@@ -1367,27 +1367,39 @@ again:
                ref.skip_qgroup = true;
                btrfs_init_tree_ref(&ref, level - 1, src->root_key.objectid);
                ret = btrfs_inc_extent_ref(trans, &ref);
-               BUG_ON(ret);
+               if (ret) {
+                       btrfs_abort_transaction(trans, ret);
+                       break;
+               }
                btrfs_init_generic_ref(&ref, BTRFS_ADD_DELAYED_REF, new_bytenr,
                                       blocksize, 0);
                ref.skip_qgroup = true;
                btrfs_init_tree_ref(&ref, level - 1, dest->root_key.objectid);
                ret = btrfs_inc_extent_ref(trans, &ref);
-               BUG_ON(ret);
+               if (ret) {
+                       btrfs_abort_transaction(trans, ret);
+                       break;
+               }
 
                btrfs_init_generic_ref(&ref, BTRFS_DROP_DELAYED_REF, new_bytenr,
                                       blocksize, path->nodes[level]->start);
                btrfs_init_tree_ref(&ref, level - 1, src->root_key.objectid);
                ref.skip_qgroup = true;
                ret = btrfs_free_extent(trans, &ref);
-               BUG_ON(ret);
+               if (ret) {
+                       btrfs_abort_transaction(trans, ret);
+                       break;
+               }
 
                btrfs_init_generic_ref(&ref, BTRFS_DROP_DELAYED_REF, old_bytenr,
                                       blocksize, 0);
                btrfs_init_tree_ref(&ref, level - 1, dest->root_key.objectid);
                ref.skip_qgroup = true;
                ret = btrfs_free_extent(trans, &ref);
-               BUG_ON(ret);
+               if (ret) {
+                       btrfs_abort_transaction(trans, ret);
+                       break;
+               }
 
                btrfs_unlock_up_safe(path, 0);