|
@@ -613,6 +613,19 @@ static noinline int check_leaf(struct btrfs_root *root,
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+static int check_node(struct btrfs_root *root, struct extent_buffer *node)
|
|
|
+{
|
|
|
+ unsigned long nr = btrfs_header_nritems(node);
|
|
|
+
|
|
|
+ if (nr == 0 || nr > BTRFS_NODEPTRS_PER_BLOCK(root)) {
|
|
|
+ btrfs_crit(root->fs_info,
|
|
|
+ "corrupt node: block %llu root %llu nritems %lu",
|
|
|
+ node->start, root->objectid, nr);
|
|
|
+ return -EIO;
|
|
|
+ }
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
static int btree_readpage_end_io_hook(struct btrfs_io_bio *io_bio,
|
|
|
u64 phy_offset, struct page *page,
|
|
|
u64 start, u64 end, int mirror)
|
|
@@ -683,6 +696,9 @@ static int btree_readpage_end_io_hook(struct btrfs_io_bio *io_bio,
|
|
|
ret = -EIO;
|
|
|
}
|
|
|
|
|
|
+ if (found_level > 0 && check_node(root, eb))
|
|
|
+ ret = -EIO;
|
|
|
+
|
|
|
if (!ret)
|
|
|
set_extent_buffer_uptodate(eb);
|
|
|
err:
|