btrfs: handle errors returned from unpin_extent_cache()
authorDavid Sterba <dsterba@suse.com>
Fri, 12 Jan 2024 17:31:40 +0000 (18:31 +0100)
committerDavid Sterba <dsterba@suse.com>
Mon, 4 Mar 2024 15:24:46 +0000 (16:24 +0100)
We've had numerous attempts to let function unpin_extent_cache() return
void as it only returns 0. There are still error cases to handle so do
that, in addition to the verbose messages. The only caller
btrfs_finish_one_ordered() will now abort the transaction, previously it
let it continue which could lead to further problems.

Signed-off-by: David Sterba <dsterba@suse.com>
fs/btrfs/extent_map.c
fs/btrfs/inode.c

index e9b20fbbdfcad7817d5bb8bdaf48235ac8f012e0..1bf4d0319f6cbec1cc74ff20bbca298255880c09 100644 (file)
@@ -290,6 +290,10 @@ static void try_merge_map(struct extent_map_tree *tree, struct extent_map *em)
  * Called after an extent has been written to disk properly.  Set the generation
  * to the generation that actually added the file item to the inode so we know
  * we need to sync this extent when we call fsync().
+ *
+ * Returns: 0       on success
+ *         -ENOENT  when the extent is not found in the tree
+ *         -EUCLEAN if the found extent does not match the expected start
  */
 int unpin_extent_cache(struct btrfs_inode *inode, u64 start, u64 len, u64 gen)
 {
@@ -307,14 +311,18 @@ int unpin_extent_cache(struct btrfs_inode *inode, u64 start, u64 len, u64 gen)
 "no extent map found for inode %llu (root %lld) when unpinning extent range [%llu, %llu), generation %llu",
                           btrfs_ino(inode), btrfs_root_id(inode->root),
                           start, len, gen);
+               ret = -ENOENT;
                goto out;
        }
 
-       if (WARN_ON(em->start != start))
+       if (WARN_ON(em->start != start)) {
                btrfs_warn(fs_info,
 "found extent map for inode %llu (root %lld) with unexpected start offset %llu when unpinning extent range [%llu, %llu), generation %llu",
                           btrfs_ino(inode), btrfs_root_id(inode->root),
                           em->start, start, len, gen);
+               ret = -EUCLEAN;
+               goto out;
+       }
 
        em->generation = gen;
        em->flags &= ~EXTENT_FLAG_PINNED;
index f8d98a03caaceced0c67e215a30385c7aab762f4..469963c89c67b8c207c9c933b6c7507cf69cd860 100644 (file)
@@ -3125,8 +3125,13 @@ int btrfs_finish_one_ordered(struct btrfs_ordered_extent *ordered_extent)
                                                ordered_extent->disk_num_bytes);
                }
        }
-       unpin_extent_cache(inode, ordered_extent->file_offset,
-                          ordered_extent->num_bytes, trans->transid);
+       if (ret < 0) {
+               btrfs_abort_transaction(trans, ret);
+               goto out;
+       }
+
+       ret = unpin_extent_cache(inode, ordered_extent->file_offset,
+                                ordered_extent->num_bytes, trans->transid);
        if (ret < 0) {
                btrfs_abort_transaction(trans, ret);
                goto out;