mirror of
https://github.com/edk2-porting/linux-next.git
synced 2025-01-22 04:24:02 +08:00
btrfs: relocation: Use btrfs_find_all_leafs to locate data extent parent tree leaves
In relocation, we need to locate all parent tree leaves referring to one data extent, thus we have a complex mechanism to iterate throught extent tree and subvolume trees to locate the related leaves. However this is already done in backref.c, we have btrfs_find_all_leafs(), which can return a ulist containing all leaves referring to that data extent. Use btrfs_find_all_leafs() to replace find_data_references(). There is a special handling for v1 space cache data extents, where we need to delete the v1 space cache data extents, to avoid those data extents to hang the data relocation. In this patch, the special handling is done by re-iterating the root tree leaf. Although it's a little less efficient than the old handling, considering we can reuse a lot of code, it should be acceptable. 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:
parent
b39c8f5a39
commit
19b546d7a1
@ -1409,10 +1409,10 @@ static void free_leaf_list(struct ulist *blocks)
|
|||||||
*
|
*
|
||||||
* returns 0 on success, <0 on error
|
* returns 0 on success, <0 on error
|
||||||
*/
|
*/
|
||||||
static int btrfs_find_all_leafs(struct btrfs_trans_handle *trans,
|
int btrfs_find_all_leafs(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_fs_info *fs_info, u64 bytenr,
|
struct btrfs_fs_info *fs_info, u64 bytenr,
|
||||||
u64 time_seq, struct ulist **leafs,
|
u64 time_seq, struct ulist **leafs,
|
||||||
const u64 *extent_item_pos, bool ignore_offset)
|
const u64 *extent_item_pos, bool ignore_offset)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
@ -40,6 +40,10 @@ int iterate_inodes_from_logical(u64 logical, struct btrfs_fs_info *fs_info,
|
|||||||
|
|
||||||
int paths_from_inode(u64 inum, struct inode_fs_paths *ipath);
|
int paths_from_inode(u64 inum, struct inode_fs_paths *ipath);
|
||||||
|
|
||||||
|
int btrfs_find_all_leafs(struct btrfs_trans_handle *trans,
|
||||||
|
struct btrfs_fs_info *fs_info, u64 bytenr,
|
||||||
|
u64 time_seq, struct ulist **leafs,
|
||||||
|
const u64 *extent_item_pos, bool ignore_offset);
|
||||||
int btrfs_find_all_roots(struct btrfs_trans_handle *trans,
|
int btrfs_find_all_roots(struct btrfs_trans_handle *trans,
|
||||||
struct btrfs_fs_info *fs_info, u64 bytenr,
|
struct btrfs_fs_info *fs_info, u64 bytenr,
|
||||||
u64 time_seq, struct ulist **roots, bool ignore_offset);
|
u64 time_seq, struct ulist **roots, bool ignore_offset);
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include "print-tree.h"
|
#include "print-tree.h"
|
||||||
#include "delalloc-space.h"
|
#include "delalloc-space.h"
|
||||||
#include "block-group.h"
|
#include "block-group.h"
|
||||||
|
#include "backref.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Relocation overview
|
* Relocation overview
|
||||||
@ -3620,31 +3621,6 @@ out:
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* helper to check if the block use full backrefs for pointers in it
|
|
||||||
*/
|
|
||||||
static int block_use_full_backref(struct reloc_control *rc,
|
|
||||||
struct extent_buffer *eb)
|
|
||||||
{
|
|
||||||
u64 flags;
|
|
||||||
int ret;
|
|
||||||
|
|
||||||
if (btrfs_header_flag(eb, BTRFS_HEADER_FLAG_RELOC) ||
|
|
||||||
btrfs_header_backref_rev(eb) < BTRFS_MIXED_BACKREF_REV)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
ret = btrfs_lookup_extent_info(NULL, rc->extent_root->fs_info,
|
|
||||||
eb->start, btrfs_header_level(eb), 1,
|
|
||||||
NULL, &flags);
|
|
||||||
BUG_ON(ret);
|
|
||||||
|
|
||||||
if (flags & BTRFS_BLOCK_FLAG_FULL_BACKREF)
|
|
||||||
ret = 1;
|
|
||||||
else
|
|
||||||
ret = 0;
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int delete_block_group_cache(struct btrfs_fs_info *fs_info,
|
static int delete_block_group_cache(struct btrfs_fs_info *fs_info,
|
||||||
struct btrfs_block_group *block_group,
|
struct btrfs_block_group *block_group,
|
||||||
struct inode *inode,
|
struct inode *inode,
|
||||||
@ -3688,174 +3664,40 @@ out:
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* helper to add tree blocks for backref of type BTRFS_EXTENT_DATA_REF_KEY
|
* Locate the free space cache EXTENT_DATA in root tree leaf and delete the
|
||||||
* this function scans fs tree to find blocks reference the data extent
|
* cache inode, to avoid free space cache data extent blocking data relocation.
|
||||||
*/
|
*/
|
||||||
static int find_data_references(struct reloc_control *rc,
|
static int delete_v1_space_cache(struct extent_buffer *leaf,
|
||||||
struct btrfs_key *extent_key,
|
struct btrfs_block_group *block_group,
|
||||||
struct extent_buffer *leaf,
|
u64 data_bytenr)
|
||||||
struct btrfs_extent_data_ref *ref,
|
|
||||||
struct rb_root *blocks)
|
|
||||||
{
|
{
|
||||||
struct btrfs_fs_info *fs_info = rc->extent_root->fs_info;
|
u64 space_cache_ino;
|
||||||
struct btrfs_path *path;
|
struct btrfs_file_extent_item *ei;
|
||||||
struct tree_block *block;
|
|
||||||
struct btrfs_root *root;
|
|
||||||
struct btrfs_file_extent_item *fi;
|
|
||||||
struct rb_node *rb_node;
|
|
||||||
struct btrfs_key key;
|
struct btrfs_key key;
|
||||||
u64 ref_root;
|
bool found = false;
|
||||||
u64 ref_objectid;
|
int i;
|
||||||
u64 ref_offset;
|
|
||||||
u32 ref_count;
|
|
||||||
u32 nritems;
|
|
||||||
int err = 0;
|
|
||||||
int added = 0;
|
|
||||||
int counted;
|
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ref_root = btrfs_extent_data_ref_root(leaf, ref);
|
if (btrfs_header_owner(leaf) != BTRFS_ROOT_TREE_OBJECTID)
|
||||||
ref_objectid = btrfs_extent_data_ref_objectid(leaf, ref);
|
return 0;
|
||||||
ref_offset = btrfs_extent_data_ref_offset(leaf, ref);
|
|
||||||
ref_count = btrfs_extent_data_ref_count(leaf, ref);
|
|
||||||
|
|
||||||
/*
|
for (i = 0; i < btrfs_header_nritems(leaf); i++) {
|
||||||
* This is an extent belonging to the free space cache, lets just delete
|
btrfs_item_key_to_cpu(leaf, &key, i);
|
||||||
* it and redo the search.
|
if (key.type != BTRFS_EXTENT_DATA_KEY)
|
||||||
*/
|
continue;
|
||||||
if (ref_root == BTRFS_ROOT_TREE_OBJECTID) {
|
ei = btrfs_item_ptr(leaf, i, struct btrfs_file_extent_item);
|
||||||
ret = delete_block_group_cache(fs_info, rc->block_group,
|
if (btrfs_file_extent_type(leaf, ei) == BTRFS_FILE_EXTENT_REG &&
|
||||||
NULL, ref_objectid);
|
btrfs_file_extent_disk_bytenr(leaf, ei) == data_bytenr) {
|
||||||
if (ret != -ENOENT)
|
found = true;
|
||||||
return ret;
|
space_cache_ino = key.objectid;
|
||||||
ret = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
path = btrfs_alloc_path();
|
|
||||||
if (!path)
|
|
||||||
return -ENOMEM;
|
|
||||||
path->reada = READA_FORWARD;
|
|
||||||
|
|
||||||
root = read_fs_root(fs_info, ref_root);
|
|
||||||
if (IS_ERR(root)) {
|
|
||||||
err = PTR_ERR(root);
|
|
||||||
goto out_free;
|
|
||||||
}
|
|
||||||
|
|
||||||
key.objectid = ref_objectid;
|
|
||||||
key.type = BTRFS_EXTENT_DATA_KEY;
|
|
||||||
if (ref_offset > ((u64)-1 << 32))
|
|
||||||
key.offset = 0;
|
|
||||||
else
|
|
||||||
key.offset = ref_offset;
|
|
||||||
|
|
||||||
path->search_commit_root = 1;
|
|
||||||
path->skip_locking = 1;
|
|
||||||
ret = btrfs_search_slot(NULL, root, &key, path, 0, 0);
|
|
||||||
if (ret < 0) {
|
|
||||||
err = ret;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
|
|
||||||
leaf = path->nodes[0];
|
|
||||||
nritems = btrfs_header_nritems(leaf);
|
|
||||||
/*
|
|
||||||
* the references in tree blocks that use full backrefs
|
|
||||||
* are not counted in
|
|
||||||
*/
|
|
||||||
if (block_use_full_backref(rc, leaf))
|
|
||||||
counted = 0;
|
|
||||||
else
|
|
||||||
counted = 1;
|
|
||||||
rb_node = tree_search(blocks, leaf->start);
|
|
||||||
if (rb_node) {
|
|
||||||
if (counted)
|
|
||||||
added = 1;
|
|
||||||
else
|
|
||||||
path->slots[0] = nritems;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (ref_count > 0) {
|
|
||||||
while (path->slots[0] >= nritems) {
|
|
||||||
ret = btrfs_next_leaf(root, path);
|
|
||||||
if (ret < 0) {
|
|
||||||
err = ret;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
if (WARN_ON(ret > 0))
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
leaf = path->nodes[0];
|
|
||||||
nritems = btrfs_header_nritems(leaf);
|
|
||||||
added = 0;
|
|
||||||
|
|
||||||
if (block_use_full_backref(rc, leaf))
|
|
||||||
counted = 0;
|
|
||||||
else
|
|
||||||
counted = 1;
|
|
||||||
rb_node = tree_search(blocks, leaf->start);
|
|
||||||
if (rb_node) {
|
|
||||||
if (counted)
|
|
||||||
added = 1;
|
|
||||||
else
|
|
||||||
path->slots[0] = nritems;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
btrfs_item_key_to_cpu(leaf, &key, path->slots[0]);
|
|
||||||
if (WARN_ON(key.objectid != ref_objectid ||
|
|
||||||
key.type != BTRFS_EXTENT_DATA_KEY))
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
fi = btrfs_item_ptr(leaf, path->slots[0],
|
|
||||||
struct btrfs_file_extent_item);
|
|
||||||
|
|
||||||
if (btrfs_file_extent_type(leaf, fi) ==
|
|
||||||
BTRFS_FILE_EXTENT_INLINE)
|
|
||||||
goto next;
|
|
||||||
|
|
||||||
if (btrfs_file_extent_disk_bytenr(leaf, fi) !=
|
|
||||||
extent_key->objectid)
|
|
||||||
goto next;
|
|
||||||
|
|
||||||
key.offset -= btrfs_file_extent_offset(leaf, fi);
|
|
||||||
if (key.offset != ref_offset)
|
|
||||||
goto next;
|
|
||||||
|
|
||||||
if (counted)
|
|
||||||
ref_count--;
|
|
||||||
if (added)
|
|
||||||
goto next;
|
|
||||||
|
|
||||||
if (!tree_block_processed(leaf->start, rc)) {
|
|
||||||
block = kmalloc(sizeof(*block), GFP_NOFS);
|
|
||||||
if (!block) {
|
|
||||||
err = -ENOMEM;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
block->bytenr = leaf->start;
|
|
||||||
btrfs_item_key_to_cpu(leaf, &block->key, 0);
|
|
||||||
block->level = 0;
|
|
||||||
block->key_ready = 1;
|
|
||||||
rb_node = tree_insert(blocks, block->bytenr,
|
|
||||||
&block->rb_node);
|
|
||||||
if (rb_node)
|
|
||||||
backref_tree_panic(rb_node, -EEXIST,
|
|
||||||
block->bytenr);
|
|
||||||
}
|
}
|
||||||
if (counted)
|
|
||||||
added = 1;
|
|
||||||
else
|
|
||||||
path->slots[0] = nritems;
|
|
||||||
next:
|
|
||||||
path->slots[0]++;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
out:
|
if (!found)
|
||||||
btrfs_put_root(root);
|
return -ENOENT;
|
||||||
out_free:
|
ret = delete_block_group_cache(leaf->fs_info, block_group, NULL,
|
||||||
btrfs_free_path(path);
|
space_cache_ino);
|
||||||
return err;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -3867,91 +3709,41 @@ int add_data_references(struct reloc_control *rc,
|
|||||||
struct btrfs_path *path,
|
struct btrfs_path *path,
|
||||||
struct rb_root *blocks)
|
struct rb_root *blocks)
|
||||||
{
|
{
|
||||||
struct btrfs_key key;
|
struct btrfs_fs_info *fs_info = rc->extent_root->fs_info;
|
||||||
struct extent_buffer *eb;
|
struct ulist *leaves = NULL;
|
||||||
struct btrfs_extent_data_ref *dref;
|
struct ulist_iterator leaf_uiter;
|
||||||
struct btrfs_extent_inline_ref *iref;
|
struct ulist_node *ref_node = NULL;
|
||||||
unsigned long ptr;
|
const u32 blocksize = fs_info->nodesize;
|
||||||
unsigned long end;
|
|
||||||
u32 blocksize = rc->extent_root->fs_info->nodesize;
|
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
int err = 0;
|
|
||||||
|
|
||||||
eb = path->nodes[0];
|
|
||||||
ptr = btrfs_item_ptr_offset(eb, path->slots[0]);
|
|
||||||
end = ptr + btrfs_item_size_nr(eb, path->slots[0]);
|
|
||||||
ptr += sizeof(struct btrfs_extent_item);
|
|
||||||
|
|
||||||
while (ptr < end) {
|
|
||||||
iref = (struct btrfs_extent_inline_ref *)ptr;
|
|
||||||
key.type = btrfs_get_extent_inline_ref_type(eb, iref,
|
|
||||||
BTRFS_REF_TYPE_DATA);
|
|
||||||
if (key.type == BTRFS_SHARED_DATA_REF_KEY) {
|
|
||||||
key.offset = btrfs_extent_inline_ref_offset(eb, iref);
|
|
||||||
ret = __add_tree_block(rc, key.offset, blocksize,
|
|
||||||
blocks);
|
|
||||||
} else if (key.type == BTRFS_EXTENT_DATA_REF_KEY) {
|
|
||||||
dref = (struct btrfs_extent_data_ref *)(&iref->offset);
|
|
||||||
ret = find_data_references(rc, extent_key,
|
|
||||||
eb, dref, blocks);
|
|
||||||
} else {
|
|
||||||
ret = -EUCLEAN;
|
|
||||||
btrfs_err(rc->extent_root->fs_info,
|
|
||||||
"extent %llu slot %d has an invalid inline ref type",
|
|
||||||
eb->start, path->slots[0]);
|
|
||||||
}
|
|
||||||
if (ret) {
|
|
||||||
err = ret;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
ptr += btrfs_extent_inline_ref_size(key.type);
|
|
||||||
}
|
|
||||||
WARN_ON(ptr > end);
|
|
||||||
|
|
||||||
while (1) {
|
|
||||||
cond_resched();
|
|
||||||
eb = path->nodes[0];
|
|
||||||
if (path->slots[0] >= btrfs_header_nritems(eb)) {
|
|
||||||
ret = btrfs_next_leaf(rc->extent_root, path);
|
|
||||||
if (ret < 0) {
|
|
||||||
err = ret;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (ret > 0)
|
|
||||||
break;
|
|
||||||
eb = path->nodes[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
btrfs_item_key_to_cpu(eb, &key, path->slots[0]);
|
|
||||||
if (key.objectid != extent_key->objectid)
|
|
||||||
break;
|
|
||||||
|
|
||||||
if (key.type == BTRFS_SHARED_DATA_REF_KEY) {
|
|
||||||
ret = __add_tree_block(rc, key.offset, blocksize,
|
|
||||||
blocks);
|
|
||||||
} else if (key.type == BTRFS_EXTENT_DATA_REF_KEY) {
|
|
||||||
dref = btrfs_item_ptr(eb, path->slots[0],
|
|
||||||
struct btrfs_extent_data_ref);
|
|
||||||
ret = find_data_references(rc, extent_key,
|
|
||||||
eb, dref, blocks);
|
|
||||||
} else if (unlikely(key.type == BTRFS_EXTENT_REF_V0_KEY)) {
|
|
||||||
btrfs_print_v0_err(eb->fs_info);
|
|
||||||
btrfs_handle_fs_error(eb->fs_info, -EINVAL, NULL);
|
|
||||||
ret = -EINVAL;
|
|
||||||
} else {
|
|
||||||
ret = 0;
|
|
||||||
}
|
|
||||||
if (ret) {
|
|
||||||
err = ret;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
path->slots[0]++;
|
|
||||||
}
|
|
||||||
out:
|
|
||||||
btrfs_release_path(path);
|
btrfs_release_path(path);
|
||||||
if (err)
|
ret = btrfs_find_all_leafs(NULL, fs_info, extent_key->objectid,
|
||||||
|
0, &leaves, NULL, true);
|
||||||
|
if (ret < 0)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
ULIST_ITER_INIT(&leaf_uiter);
|
||||||
|
while ((ref_node = ulist_next(leaves, &leaf_uiter))) {
|
||||||
|
struct extent_buffer *eb;
|
||||||
|
|
||||||
|
eb = read_tree_block(fs_info, ref_node->val, 0, 0, NULL);
|
||||||
|
if (IS_ERR(eb)) {
|
||||||
|
ret = PTR_ERR(eb);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
ret = delete_v1_space_cache(eb, rc->block_group,
|
||||||
|
extent_key->objectid);
|
||||||
|
free_extent_buffer(eb);
|
||||||
|
if (ret < 0)
|
||||||
|
break;
|
||||||
|
ret = __add_tree_block(rc, ref_node->val, blocksize, blocks);
|
||||||
|
if (ret < 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (ret < 0)
|
||||||
free_block_list(blocks);
|
free_block_list(blocks);
|
||||||
return err;
|
ulist_free(leaves);
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
Loading…
Reference in New Issue
Block a user