|
@@ -4357,6 +4357,20 @@ int btrfs_read_sys_array(struct btrfs_root *root)
|
|
return -ENOMEM;
|
|
return -ENOMEM;
|
|
btrfs_set_buffer_uptodate(sb);
|
|
btrfs_set_buffer_uptodate(sb);
|
|
btrfs_set_buffer_lockdep_class(root->root_key.objectid, sb, 0);
|
|
btrfs_set_buffer_lockdep_class(root->root_key.objectid, sb, 0);
|
|
|
|
+ /*
|
|
|
|
+ * The sb extent buffer is artifical and just used to read the system array.
|
|
|
|
+ * btrfs_set_buffer_uptodate() call does not properly mark all it's
|
|
|
|
+ * pages up-to-date when the page is larger: extent does not cover the
|
|
|
|
+ * whole page and consequently check_page_uptodate does not find all
|
|
|
|
+ * the page's extents up-to-date (the hole beyond sb),
|
|
|
|
+ * write_extent_buffer then triggers a WARN_ON.
|
|
|
|
+ *
|
|
|
|
+ * Regular short extents go through mark_extent_buffer_dirty/writeback cycle,
|
|
|
|
+ * but sb spans only this function. Add an explicit SetPageUptodate call
|
|
|
|
+ * to silence the warning eg. on PowerPC 64.
|
|
|
|
+ */
|
|
|
|
+ if (PAGE_CACHE_SIZE > BTRFS_SUPER_INFO_SIZE)
|
|
|
|
+ SetPageUptodate(sb->first_page);
|
|
|
|
|
|
write_extent_buffer(sb, super_copy, 0, BTRFS_SUPER_INFO_SIZE);
|
|
write_extent_buffer(sb, super_copy, 0, BTRFS_SUPER_INFO_SIZE);
|
|
array_size = btrfs_super_sys_array_size(super_copy);
|
|
array_size = btrfs_super_sys_array_size(super_copy);
|