btrfs: abort transaction during log replay if walk_log_tree() failed
authorFilipe Manana <fdmanana@suse.com>
Wed, 21 May 2025 16:41:18 +0000 (17:41 +0100)
committerDavid Sterba <dsterba@suse.com>
Mon, 21 Jul 2025 21:50:28 +0000 (23:50 +0200)
If we failed walking a log tree during replay, we have a missing
transaction abort to prevent committing a transaction where we didn't
fully replay all the changes from a log tree and therefore can leave the
respective subvolume tree in some inconsistent state. So add the missing
transaction abort.

CC: stable@vger.kernel.org # 6.1+
Reviewed-by: Qu Wenruo <wqu@suse.com>
Signed-off-by: Filipe Manana <fdmanana@suse.com>
Reviewed-by: David Sterba <dsterba@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
fs/btrfs/tree-log.c

index fea978ffadf659a1adc1dd200fc35d41146eb460..6a60f072c36596e975f35eb5744923261b67af93 100644 (file)
@@ -7283,11 +7283,14 @@ again:
 
                wc.replay_dest->log_root = log;
                ret = btrfs_record_root_in_trans(trans, wc.replay_dest);
-               if (ret)
+               if (ret) {
                        /* The loop needs to continue due to the root refs */
                        btrfs_abort_transaction(trans, ret);
-               else
+               } else {
                        ret = walk_log_tree(trans, log, &wc);
+                       if (ret)
+                               btrfs_abort_transaction(trans, ret);
+               }
 
                if (!ret && wc.stage == LOG_WALK_REPLAY_ALL) {
                        ret = fixup_inode_link_counts(trans, wc.replay_dest,