btrfs: validate system chunk array at btrfs_validate_super()

Currently btrfs_validate_super() only does a very basic check on the
array chunk size (not too large than the available space, but not too
small to contain no chunk).

The more comprehensive checks (the regular chunk checks and size check
inside the system chunk array) are all done inside btrfs_read_sys_array().

It's not a big deal, but it also means we do not do any validation on
the system chunk array at super block writeback time either.

Do the following modification to centralize the system chunk array
checks into btrfs_validate_super():

- Make chunk_err() helper accept stack chunk pointer
  If @leaf parameter is NULL, then the @chunk pointer will be a pointer
  to the chunk item, other than the offset inside the leaf.

  And since @leaf can be NULL, add a new @fs_info parameter for that
  case.

- Make btrfs_check_chunk_valid() handle stack chunk pointer
  The same as chunk_err(), a new @fs_info parameter, and if @leaf is
  NULL, then @chunk will be a pointer to a stack chunk.

  If @chunk is NULL, then all needed btrfs_chunk members will be read
  using the stack helper instead of the leaf helper.
  This means we need to read out all the needed member at the beginning
  of the function.

  Furthermore, at super block read time, fs_info->sectorsize is not yet
  initialized, we need one extra @sectorsize parameter to grab the
  correct sectorsize.

- Introduce a helper validate_sys_chunk_array()
  * Validate the disk key.
  * Validate the size before we access the full chunk items.
  * Do the full chunk item validation.

- Call validate_sys_chunk_array() at btrfs_validate_super()

- Simplify the checks inside btrfs_read_sys_array()
  Now the checks will be converted to an ASSERT().

- Simplify the checks inside read_one_chunk()
  Now that all chunk items inside system chunk array and chunk tree are
  verified, there is no need to verify them again inside read_one_chunk().

This change has the following advantages:

- More comprehensive checks at write time
  And unlike the sys_chunk_array read routine, this time we do not need
  to allocate a dummy extent buffer to do the check.
  All the checks done here require no new memory allocation.

- Slightly improved readability when iterating the system chunk array

Signed-off-by: Qu Wenruo <wqu@suse.com>
Reviewed-by: David Sterba <dsterba@suse.com>
Signed-off-by: David Sterba <dsterba@suse.com>
This commit is contained in:
Qu Wenruo 2024-12-21 16:15:19 +10:30 committed by David Sterba
parent 4e4d058e21
commit 2a9bb78cfd
4 changed files with 139 additions and 104 deletions

View File

@ -2327,6 +2327,71 @@ out:
return ret; return ret;
} }
static int validate_sys_chunk_array(const struct btrfs_fs_info *fs_info,
const struct btrfs_super_block *sb)
{
unsigned int cur = 0; /* Offset inside the sys chunk array */
/*
* At sb read time, fs_info is not fully initialized. Thus we have
* to use super block sectorsize, which should have been validated.
*/
const u32 sectorsize = btrfs_super_sectorsize(sb);
u32 sys_array_size = btrfs_super_sys_array_size(sb);
if (sys_array_size > BTRFS_SYSTEM_CHUNK_ARRAY_SIZE) {
btrfs_err(fs_info, "system chunk array too big %u > %u",
sys_array_size, BTRFS_SYSTEM_CHUNK_ARRAY_SIZE);
return -EUCLEAN;
}
while (cur < sys_array_size) {
struct btrfs_disk_key *disk_key;
struct btrfs_chunk *chunk;
struct btrfs_key key;
u64 type;
u16 num_stripes;
u32 len;
int ret;
disk_key = (struct btrfs_disk_key *)(sb->sys_chunk_array + cur);
len = sizeof(*disk_key);
if (cur + len > sys_array_size)
goto short_read;
cur += len;
btrfs_disk_key_to_cpu(&key, disk_key);
if (key.type != BTRFS_CHUNK_ITEM_KEY) {
btrfs_err(fs_info,
"unexpected item type %u in sys_array at offset %u",
key.type, cur);
return -EUCLEAN;
}
chunk = (struct btrfs_chunk *)(sb->sys_chunk_array + cur);
num_stripes = btrfs_stack_chunk_num_stripes(chunk);
if (cur + btrfs_chunk_item_size(num_stripes) > sys_array_size)
goto short_read;
type = btrfs_stack_chunk_type(chunk);
if (!(type & BTRFS_BLOCK_GROUP_SYSTEM)) {
btrfs_err(fs_info,
"invalid chunk type %llu in sys_array at offset %u",
type, cur);
return -EUCLEAN;
}
ret = btrfs_check_chunk_valid(fs_info, NULL, chunk, key.offset,
sectorsize);
if (ret < 0)
return ret;
cur += btrfs_chunk_item_size(num_stripes);
}
return 0;
short_read:
btrfs_err(fs_info,
"super block sys chunk array short read, cur=%u sys_array_size=%u",
cur, sys_array_size);
return -EUCLEAN;
}
/* /*
* Real super block validation * Real super block validation
* NOTE: super csum type and incompat features will not be checked here. * NOTE: super csum type and incompat features will not be checked here.
@ -2495,6 +2560,8 @@ int btrfs_validate_super(const struct btrfs_fs_info *fs_info,
ret = -EINVAL; ret = -EINVAL;
} }
ret = validate_sys_chunk_array(fs_info, sb);
/* /*
* Obvious sys_chunk_array corruptions, it must hold at least one key * Obvious sys_chunk_array corruptions, it must hold at least one key
* and one chunk * and one chunk

View File

@ -764,22 +764,19 @@ static int check_block_group_item(struct extent_buffer *leaf,
return 0; return 0;
} }
__printf(4, 5) __printf(5, 6)
__cold __cold
static void chunk_err(const struct extent_buffer *leaf, static void chunk_err(const struct btrfs_fs_info *fs_info,
const struct extent_buffer *leaf,
const struct btrfs_chunk *chunk, u64 logical, const struct btrfs_chunk *chunk, u64 logical,
const char *fmt, ...) const char *fmt, ...)
{ {
const struct btrfs_fs_info *fs_info = leaf->fs_info; bool is_sb = !leaf;
bool is_sb;
struct va_format vaf; struct va_format vaf;
va_list args; va_list args;
int i; int i;
int slot = -1; int slot = -1;
/* Only superblock eb is able to have such small offset */
is_sb = (leaf->start == BTRFS_SUPER_INFO_OFFSET);
if (!is_sb) { if (!is_sb) {
/* /*
* Get the slot number by iterating through all slots, this * Get the slot number by iterating through all slots, this
@ -812,13 +809,17 @@ static void chunk_err(const struct extent_buffer *leaf,
/* /*
* The common chunk check which could also work on super block sys chunk array. * The common chunk check which could also work on super block sys chunk array.
* *
* If @leaf is NULL, then @chunk must be an on-stack chunk item.
* (For superblock sys_chunk array, and fs_info->sectorsize is unreliable)
*
* Return -EUCLEAN if anything is corrupted. * Return -EUCLEAN if anything is corrupted.
* Return 0 if everything is OK. * Return 0 if everything is OK.
*/ */
int btrfs_check_chunk_valid(struct extent_buffer *leaf, int btrfs_check_chunk_valid(const struct btrfs_fs_info *fs_info,
struct btrfs_chunk *chunk, u64 logical) const struct extent_buffer *leaf,
const struct btrfs_chunk *chunk, u64 logical,
u32 sectorsize)
{ {
struct btrfs_fs_info *fs_info = leaf->fs_info;
u64 length; u64 length;
u64 chunk_end; u64 chunk_end;
u64 stripe_len; u64 stripe_len;
@ -826,63 +827,73 @@ int btrfs_check_chunk_valid(struct extent_buffer *leaf,
u16 sub_stripes; u16 sub_stripes;
u64 type; u64 type;
u64 features; u64 features;
u32 chunk_sector_size;
bool mixed = false; bool mixed = false;
int raid_index; int raid_index;
int nparity; int nparity;
int ncopies; int ncopies;
length = btrfs_chunk_length(leaf, chunk); if (leaf) {
stripe_len = btrfs_chunk_stripe_len(leaf, chunk); length = btrfs_chunk_length(leaf, chunk);
num_stripes = btrfs_chunk_num_stripes(leaf, chunk); stripe_len = btrfs_chunk_stripe_len(leaf, chunk);
sub_stripes = btrfs_chunk_sub_stripes(leaf, chunk); num_stripes = btrfs_chunk_num_stripes(leaf, chunk);
type = btrfs_chunk_type(leaf, chunk); sub_stripes = btrfs_chunk_sub_stripes(leaf, chunk);
type = btrfs_chunk_type(leaf, chunk);
chunk_sector_size = btrfs_chunk_sector_size(leaf, chunk);
} else {
length = btrfs_stack_chunk_length(chunk);
stripe_len = btrfs_stack_chunk_stripe_len(chunk);
num_stripes = btrfs_stack_chunk_num_stripes(chunk);
sub_stripes = btrfs_stack_chunk_sub_stripes(chunk);
type = btrfs_stack_chunk_type(chunk);
chunk_sector_size = btrfs_stack_chunk_sector_size(chunk);
}
raid_index = btrfs_bg_flags_to_raid_index(type); raid_index = btrfs_bg_flags_to_raid_index(type);
ncopies = btrfs_raid_array[raid_index].ncopies; ncopies = btrfs_raid_array[raid_index].ncopies;
nparity = btrfs_raid_array[raid_index].nparity; nparity = btrfs_raid_array[raid_index].nparity;
if (unlikely(!num_stripes)) { if (unlikely(!num_stripes)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk num_stripes, have %u", num_stripes); "invalid chunk num_stripes, have %u", num_stripes);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(num_stripes < ncopies)) { if (unlikely(num_stripes < ncopies)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk num_stripes < ncopies, have %u < %d", "invalid chunk num_stripes < ncopies, have %u < %d",
num_stripes, ncopies); num_stripes, ncopies);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(nparity && num_stripes == nparity)) { if (unlikely(nparity && num_stripes == nparity)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk num_stripes == nparity, have %u == %d", "invalid chunk num_stripes == nparity, have %u == %d",
num_stripes, nparity); num_stripes, nparity);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(!IS_ALIGNED(logical, fs_info->sectorsize))) { if (unlikely(!IS_ALIGNED(logical, sectorsize))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk logical, have %llu should aligned to %u", "invalid chunk logical, have %llu should aligned to %u",
logical, fs_info->sectorsize); logical, sectorsize);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(btrfs_chunk_sector_size(leaf, chunk) != fs_info->sectorsize)) { if (unlikely(chunk_sector_size != sectorsize)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk sectorsize, have %u expect %u", "invalid chunk sectorsize, have %u expect %u",
btrfs_chunk_sector_size(leaf, chunk), chunk_sector_size, sectorsize);
fs_info->sectorsize);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(!length || !IS_ALIGNED(length, fs_info->sectorsize))) { if (unlikely(!length || !IS_ALIGNED(length, sectorsize))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk length, have %llu", length); "invalid chunk length, have %llu", length);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(check_add_overflow(logical, length, &chunk_end))) { if (unlikely(check_add_overflow(logical, length, &chunk_end))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk logical start and length, have logical start %llu length %llu", "invalid chunk logical start and length, have logical start %llu length %llu",
logical, length); logical, length);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(!is_power_of_2(stripe_len) || stripe_len != BTRFS_STRIPE_LEN)) { if (unlikely(!is_power_of_2(stripe_len) || stripe_len != BTRFS_STRIPE_LEN)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk stripe length: %llu", "invalid chunk stripe length: %llu",
stripe_len); stripe_len);
return -EUCLEAN; return -EUCLEAN;
@ -896,30 +907,29 @@ int btrfs_check_chunk_valid(struct extent_buffer *leaf,
* Thus it should be a good way to catch obvious bitflips. * Thus it should be a good way to catch obvious bitflips.
*/ */
if (unlikely(length >= btrfs_stripe_nr_to_offset(U32_MAX))) { if (unlikely(length >= btrfs_stripe_nr_to_offset(U32_MAX))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"chunk length too large: have %llu limit %llu", "chunk length too large: have %llu limit %llu",
length, btrfs_stripe_nr_to_offset(U32_MAX)); length, btrfs_stripe_nr_to_offset(U32_MAX));
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(type & ~(BTRFS_BLOCK_GROUP_TYPE_MASK | if (unlikely(type & ~(BTRFS_BLOCK_GROUP_TYPE_MASK |
BTRFS_BLOCK_GROUP_PROFILE_MASK))) { BTRFS_BLOCK_GROUP_PROFILE_MASK))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"unrecognized chunk type: 0x%llx", "unrecognized chunk type: 0x%llx",
~(BTRFS_BLOCK_GROUP_TYPE_MASK | ~(BTRFS_BLOCK_GROUP_TYPE_MASK |
BTRFS_BLOCK_GROUP_PROFILE_MASK) & BTRFS_BLOCK_GROUP_PROFILE_MASK) & type);
btrfs_chunk_type(leaf, chunk));
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely(!has_single_bit_set(type & BTRFS_BLOCK_GROUP_PROFILE_MASK) && if (unlikely(!has_single_bit_set(type & BTRFS_BLOCK_GROUP_PROFILE_MASK) &&
(type & BTRFS_BLOCK_GROUP_PROFILE_MASK) != 0)) { (type & BTRFS_BLOCK_GROUP_PROFILE_MASK) != 0)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid chunk profile flag: 0x%llx, expect 0 or 1 bit set", "invalid chunk profile flag: 0x%llx, expect 0 or 1 bit set",
type & BTRFS_BLOCK_GROUP_PROFILE_MASK); type & BTRFS_BLOCK_GROUP_PROFILE_MASK);
return -EUCLEAN; return -EUCLEAN;
} }
if (unlikely((type & BTRFS_BLOCK_GROUP_TYPE_MASK) == 0)) { if (unlikely((type & BTRFS_BLOCK_GROUP_TYPE_MASK) == 0)) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"missing chunk type flag, have 0x%llx one bit must be set in 0x%llx", "missing chunk type flag, have 0x%llx one bit must be set in 0x%llx",
type, BTRFS_BLOCK_GROUP_TYPE_MASK); type, BTRFS_BLOCK_GROUP_TYPE_MASK);
return -EUCLEAN; return -EUCLEAN;
@ -928,7 +938,7 @@ int btrfs_check_chunk_valid(struct extent_buffer *leaf,
if (unlikely((type & BTRFS_BLOCK_GROUP_SYSTEM) && if (unlikely((type & BTRFS_BLOCK_GROUP_SYSTEM) &&
(type & (BTRFS_BLOCK_GROUP_METADATA | (type & (BTRFS_BLOCK_GROUP_METADATA |
BTRFS_BLOCK_GROUP_DATA)))) { BTRFS_BLOCK_GROUP_DATA)))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"system chunk with data or metadata type: 0x%llx", "system chunk with data or metadata type: 0x%llx",
type); type);
return -EUCLEAN; return -EUCLEAN;
@ -941,7 +951,7 @@ int btrfs_check_chunk_valid(struct extent_buffer *leaf,
if (!mixed) { if (!mixed) {
if (unlikely((type & BTRFS_BLOCK_GROUP_METADATA) && if (unlikely((type & BTRFS_BLOCK_GROUP_METADATA) &&
(type & BTRFS_BLOCK_GROUP_DATA))) { (type & BTRFS_BLOCK_GROUP_DATA))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"mixed chunk type in non-mixed mode: 0x%llx", type); "mixed chunk type in non-mixed mode: 0x%llx", type);
return -EUCLEAN; return -EUCLEAN;
} }
@ -963,7 +973,7 @@ int btrfs_check_chunk_valid(struct extent_buffer *leaf,
num_stripes != btrfs_raid_array[BTRFS_RAID_DUP].dev_stripes) || num_stripes != btrfs_raid_array[BTRFS_RAID_DUP].dev_stripes) ||
((type & BTRFS_BLOCK_GROUP_PROFILE_MASK) == 0 && ((type & BTRFS_BLOCK_GROUP_PROFILE_MASK) == 0 &&
num_stripes != btrfs_raid_array[BTRFS_RAID_SINGLE].dev_stripes))) { num_stripes != btrfs_raid_array[BTRFS_RAID_SINGLE].dev_stripes))) {
chunk_err(leaf, chunk, logical, chunk_err(fs_info, leaf, chunk, logical,
"invalid num_stripes:sub_stripes %u:%u for profile %llu", "invalid num_stripes:sub_stripes %u:%u for profile %llu",
num_stripes, sub_stripes, num_stripes, sub_stripes,
type & BTRFS_BLOCK_GROUP_PROFILE_MASK); type & BTRFS_BLOCK_GROUP_PROFILE_MASK);
@ -983,14 +993,15 @@ static int check_leaf_chunk_item(struct extent_buffer *leaf,
struct btrfs_chunk *chunk, struct btrfs_chunk *chunk,
struct btrfs_key *key, int slot) struct btrfs_key *key, int slot)
{ {
struct btrfs_fs_info *fs_info = leaf->fs_info;
int num_stripes; int num_stripes;
if (unlikely(btrfs_item_size(leaf, slot) < sizeof(struct btrfs_chunk))) { if (unlikely(btrfs_item_size(leaf, slot) < sizeof(struct btrfs_chunk))) {
chunk_err(leaf, chunk, key->offset, chunk_err(fs_info, leaf, chunk, key->offset,
"invalid chunk item size: have %u expect [%zu, %u)", "invalid chunk item size: have %u expect [%zu, %u)",
btrfs_item_size(leaf, slot), btrfs_item_size(leaf, slot),
sizeof(struct btrfs_chunk), sizeof(struct btrfs_chunk),
BTRFS_LEAF_DATA_SIZE(leaf->fs_info)); BTRFS_LEAF_DATA_SIZE(fs_info));
return -EUCLEAN; return -EUCLEAN;
} }
@ -1001,14 +1012,15 @@ static int check_leaf_chunk_item(struct extent_buffer *leaf,
if (unlikely(btrfs_chunk_item_size(num_stripes) != if (unlikely(btrfs_chunk_item_size(num_stripes) !=
btrfs_item_size(leaf, slot))) { btrfs_item_size(leaf, slot))) {
chunk_err(leaf, chunk, key->offset, chunk_err(fs_info, leaf, chunk, key->offset,
"invalid chunk item size: have %u expect %lu", "invalid chunk item size: have %u expect %lu",
btrfs_item_size(leaf, slot), btrfs_item_size(leaf, slot),
btrfs_chunk_item_size(num_stripes)); btrfs_chunk_item_size(num_stripes));
return -EUCLEAN; return -EUCLEAN;
} }
out: out:
return btrfs_check_chunk_valid(leaf, chunk, key->offset); return btrfs_check_chunk_valid(fs_info, leaf, chunk, key->offset,
fs_info->sectorsize);
} }
__printf(3, 4) __printf(3, 4)

View File

@ -10,6 +10,7 @@
#include <uapi/linux/btrfs_tree.h> #include <uapi/linux/btrfs_tree.h>
struct extent_buffer; struct extent_buffer;
struct btrfs_fs_info;
struct btrfs_chunk; struct btrfs_chunk;
struct btrfs_key; struct btrfs_key;
@ -66,8 +67,10 @@ enum btrfs_tree_block_status __btrfs_check_node(struct extent_buffer *node);
int btrfs_check_leaf(struct extent_buffer *leaf); int btrfs_check_leaf(struct extent_buffer *leaf);
int btrfs_check_node(struct extent_buffer *node); int btrfs_check_node(struct extent_buffer *node);
int btrfs_check_chunk_valid(struct extent_buffer *leaf, int btrfs_check_chunk_valid(const struct btrfs_fs_info *fs_info,
struct btrfs_chunk *chunk, u64 logical); const struct extent_buffer *leaf,
const struct btrfs_chunk *chunk, u64 logical,
u32 sectorsize);
int btrfs_check_eb_owner(const struct extent_buffer *eb, u64 root_owner); int btrfs_check_eb_owner(const struct extent_buffer *eb, u64 root_owner);
int btrfs_verify_level_key(struct extent_buffer *eb, int btrfs_verify_level_key(struct extent_buffer *eb,
const struct btrfs_tree_parent_check *check); const struct btrfs_tree_parent_check *check);

View File

@ -7004,16 +7004,6 @@ static int read_one_chunk(struct btrfs_key *key, struct extent_buffer *leaf,
warn_32bit_meta_chunk(fs_info, logical, length, type); warn_32bit_meta_chunk(fs_info, logical, length, type);
#endif #endif
/*
* Only need to verify chunk item if we're reading from sys chunk array,
* as chunk item in tree block is already verified by tree-checker.
*/
if (leaf->start == BTRFS_SUPER_INFO_OFFSET) {
ret = btrfs_check_chunk_valid(leaf, chunk, logical);
if (ret)
return ret;
}
map = btrfs_find_chunk_map(fs_info, logical, 1); map = btrfs_find_chunk_map(fs_info, logical, 1);
/* already mapped? */ /* already mapped? */
@ -7271,16 +7261,11 @@ int btrfs_read_sys_array(struct btrfs_fs_info *fs_info)
{ {
struct btrfs_super_block *super_copy = fs_info->super_copy; struct btrfs_super_block *super_copy = fs_info->super_copy;
struct extent_buffer *sb; struct extent_buffer *sb;
struct btrfs_disk_key *disk_key;
struct btrfs_chunk *chunk;
u8 *array_ptr; u8 *array_ptr;
unsigned long sb_array_offset; unsigned long sb_array_offset;
int ret = 0; int ret = 0;
u32 num_stripes;
u32 array_size; u32 array_size;
u32 len = 0;
u32 cur_offset; u32 cur_offset;
u64 type;
struct btrfs_key key; struct btrfs_key key;
ASSERT(BTRFS_SUPER_INFO_SIZE <= fs_info->nodesize); ASSERT(BTRFS_SUPER_INFO_SIZE <= fs_info->nodesize);
@ -7303,10 +7288,15 @@ int btrfs_read_sys_array(struct btrfs_fs_info *fs_info)
cur_offset = 0; cur_offset = 0;
while (cur_offset < array_size) { while (cur_offset < array_size) {
disk_key = (struct btrfs_disk_key *)array_ptr; struct btrfs_chunk *chunk;
len = sizeof(*disk_key); struct btrfs_disk_key *disk_key = (struct btrfs_disk_key *)array_ptr;
if (cur_offset + len > array_size) u32 len = sizeof(*disk_key);
goto out_short_read;
/*
* The sys_chunk_array has been already verified at super block
* read time. Only do ASSERT()s for basic checks.
*/
ASSERT(cur_offset + len <= array_size);
btrfs_disk_key_to_cpu(&key, disk_key); btrfs_disk_key_to_cpu(&key, disk_key);
@ -7314,44 +7304,14 @@ int btrfs_read_sys_array(struct btrfs_fs_info *fs_info)
sb_array_offset += len; sb_array_offset += len;
cur_offset += len; cur_offset += len;
if (key.type != BTRFS_CHUNK_ITEM_KEY) { ASSERT(key.type == BTRFS_CHUNK_ITEM_KEY);
btrfs_err(fs_info,
"unexpected item type %u in sys_array at offset %u",
(u32)key.type, cur_offset);
ret = -EIO;
break;
}
chunk = (struct btrfs_chunk *)sb_array_offset; chunk = (struct btrfs_chunk *)sb_array_offset;
/* ASSERT(btrfs_chunk_type(sb, chunk) & BTRFS_BLOCK_GROUP_SYSTEM);
* At least one btrfs_chunk with one stripe must be present,
* exact stripe count check comes afterwards
*/
len = btrfs_chunk_item_size(1);
if (cur_offset + len > array_size)
goto out_short_read;
num_stripes = btrfs_chunk_num_stripes(sb, chunk); len = btrfs_chunk_item_size(btrfs_chunk_num_stripes(sb, chunk));
if (!num_stripes) {
btrfs_err(fs_info,
"invalid number of stripes %u in sys_array at offset %u",
num_stripes, cur_offset);
ret = -EIO;
break;
}
type = btrfs_chunk_type(sb, chunk); ASSERT(cur_offset + len <= array_size);
if ((type & BTRFS_BLOCK_GROUP_SYSTEM) == 0) {
btrfs_err(fs_info,
"invalid chunk type %llu in sys_array at offset %u",
type, cur_offset);
ret = -EIO;
break;
}
len = btrfs_chunk_item_size(num_stripes);
if (cur_offset + len > array_size)
goto out_short_read;
ret = read_one_chunk(&key, sb, chunk); ret = read_one_chunk(&key, sb, chunk);
if (ret) if (ret)
@ -7364,13 +7324,6 @@ int btrfs_read_sys_array(struct btrfs_fs_info *fs_info)
clear_extent_buffer_uptodate(sb); clear_extent_buffer_uptodate(sb);
free_extent_buffer_stale(sb); free_extent_buffer_stale(sb);
return ret; return ret;
out_short_read:
btrfs_err(fs_info, "sys_array too short to read %u bytes at offset %u",
len, cur_offset);
clear_extent_buffer_uptodate(sb);
free_extent_buffer_stale(sb);
return -EIO;
} }
/* /*