Commit 87ef2bb4 authored by Chris Mason's avatar Chris Mason

Btrfs: prevent looping forever in finish_current_insert and del_pending_extents

finish_current_insert and del_pending_extents process extent tree modifications
that build up while we are changing the extent tree.  It is a confusing
bit of code that prevents recursion.

Both functions run through a list of pending operations and both funcs
add to the list of pending operations.  If you have two procs in either
one of them, they can end up looping forever making more work for each other.

This patch makes them walk forward through the list of pending changes instead
of always trying to process the entire list.  At transaction commit
time, we catch any changes that were left over.
Signed-off-by: default avatarChris Mason <chris.mason@oracle.com>
parent 09fde3c9
...@@ -45,9 +45,9 @@ struct pending_extent_op { ...@@ -45,9 +45,9 @@ struct pending_extent_op {
}; };
static int finish_current_insert(struct btrfs_trans_handle *trans, struct static int finish_current_insert(struct btrfs_trans_handle *trans, struct
btrfs_root *extent_root); btrfs_root *extent_root, int all);
static int del_pending_extents(struct btrfs_trans_handle *trans, struct static int del_pending_extents(struct btrfs_trans_handle *trans, struct
btrfs_root *extent_root); btrfs_root *extent_root, int all);
static struct btrfs_block_group_cache * static struct btrfs_block_group_cache *
__btrfs_find_block_group(struct btrfs_root *root, __btrfs_find_block_group(struct btrfs_root *root,
struct btrfs_block_group_cache *hint, struct btrfs_block_group_cache *hint,
...@@ -711,8 +711,8 @@ static int __btrfs_update_extent_ref(struct btrfs_trans_handle *trans, ...@@ -711,8 +711,8 @@ static int __btrfs_update_extent_ref(struct btrfs_trans_handle *trans,
parent, ref_root, ref_generation, parent, ref_root, ref_generation,
owner_objectid); owner_objectid);
BUG_ON(ret); BUG_ON(ret);
finish_current_insert(trans, extent_root); finish_current_insert(trans, extent_root, 0);
del_pending_extents(trans, extent_root); del_pending_extents(trans, extent_root, 0);
out: out:
btrfs_free_path(path); btrfs_free_path(path);
return ret; return ret;
...@@ -784,8 +784,8 @@ static int __btrfs_inc_extent_ref(struct btrfs_trans_handle *trans, ...@@ -784,8 +784,8 @@ static int __btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
ref_root, ref_generation, ref_root, ref_generation,
owner_objectid); owner_objectid);
BUG_ON(ret); BUG_ON(ret);
finish_current_insert(trans, root->fs_info->extent_root); finish_current_insert(trans, root->fs_info->extent_root, 0);
del_pending_extents(trans, root->fs_info->extent_root); del_pending_extents(trans, root->fs_info->extent_root, 0);
btrfs_free_path(path); btrfs_free_path(path);
return 0; return 0;
...@@ -810,8 +810,8 @@ int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans, ...@@ -810,8 +810,8 @@ int btrfs_inc_extent_ref(struct btrfs_trans_handle *trans,
int btrfs_extent_post_op(struct btrfs_trans_handle *trans, int btrfs_extent_post_op(struct btrfs_trans_handle *trans,
struct btrfs_root *root) struct btrfs_root *root)
{ {
finish_current_insert(trans, root->fs_info->extent_root); finish_current_insert(trans, root->fs_info->extent_root, 1);
del_pending_extents(trans, root->fs_info->extent_root); del_pending_extents(trans, root->fs_info->extent_root, 1);
return 0; return 0;
} }
...@@ -1292,8 +1292,8 @@ static int write_one_cache_group(struct btrfs_trans_handle *trans, ...@@ -1292,8 +1292,8 @@ static int write_one_cache_group(struct btrfs_trans_handle *trans,
btrfs_mark_buffer_dirty(leaf); btrfs_mark_buffer_dirty(leaf);
btrfs_release_path(extent_root, path); btrfs_release_path(extent_root, path);
fail: fail:
finish_current_insert(trans, extent_root); finish_current_insert(trans, extent_root, 0);
pending_ret = del_pending_extents(trans, extent_root); pending_ret = del_pending_extents(trans, extent_root, 0);
if (ret) if (ret)
return ret; return ret;
if (pending_ret) if (pending_ret)
...@@ -1690,7 +1690,7 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans, ...@@ -1690,7 +1690,7 @@ int btrfs_finish_extent_commit(struct btrfs_trans_handle *trans,
} }
static int finish_current_insert(struct btrfs_trans_handle *trans, static int finish_current_insert(struct btrfs_trans_handle *trans,
struct btrfs_root *extent_root) struct btrfs_root *extent_root, int all)
{ {
u64 start; u64 start;
u64 end; u64 end;
...@@ -1714,7 +1714,7 @@ static int finish_current_insert(struct btrfs_trans_handle *trans, ...@@ -1714,7 +1714,7 @@ static int finish_current_insert(struct btrfs_trans_handle *trans,
&end, EXTENT_WRITEBACK); &end, EXTENT_WRITEBACK);
if (ret) { if (ret) {
mutex_unlock(&info->extent_ins_mutex); mutex_unlock(&info->extent_ins_mutex);
if (search) { if (search && all) {
search = 0; search = 0;
continue; continue;
} }
...@@ -1723,7 +1723,7 @@ static int finish_current_insert(struct btrfs_trans_handle *trans, ...@@ -1723,7 +1723,7 @@ static int finish_current_insert(struct btrfs_trans_handle *trans,
ret = try_lock_extent(&info->extent_ins, start, end, GFP_NOFS); ret = try_lock_extent(&info->extent_ins, start, end, GFP_NOFS);
if (!ret) { if (!ret) {
search = end+1; search = end + 1;
mutex_unlock(&info->extent_ins_mutex); mutex_unlock(&info->extent_ins_mutex);
cond_resched(); cond_resched();
continue; continue;
...@@ -1785,7 +1785,10 @@ static int finish_current_insert(struct btrfs_trans_handle *trans, ...@@ -1785,7 +1785,10 @@ static int finish_current_insert(struct btrfs_trans_handle *trans,
} }
kfree(extent_op); kfree(extent_op);
unlock_extent(&info->extent_ins, start, end, GFP_NOFS); unlock_extent(&info->extent_ins, start, end, GFP_NOFS);
search = 0; if (all)
search = 0;
else
search = end + 1;
cond_resched(); cond_resched();
} }
...@@ -1992,7 +1995,7 @@ static int __free_extent(struct btrfs_trans_handle *trans, ...@@ -1992,7 +1995,7 @@ static int __free_extent(struct btrfs_trans_handle *trans,
#endif #endif
} }
btrfs_free_path(path); btrfs_free_path(path);
finish_current_insert(trans, extent_root); finish_current_insert(trans, extent_root, 0);
return ret; return ret;
} }
...@@ -2001,7 +2004,7 @@ static int __free_extent(struct btrfs_trans_handle *trans, ...@@ -2001,7 +2004,7 @@ static int __free_extent(struct btrfs_trans_handle *trans,
* them from the extent map * them from the extent map
*/ */
static int del_pending_extents(struct btrfs_trans_handle *trans, struct static int del_pending_extents(struct btrfs_trans_handle *trans, struct
btrfs_root *extent_root) btrfs_root *extent_root, int all)
{ {
int ret; int ret;
int err = 0; int err = 0;
...@@ -2023,7 +2026,7 @@ static int del_pending_extents(struct btrfs_trans_handle *trans, struct ...@@ -2023,7 +2026,7 @@ static int del_pending_extents(struct btrfs_trans_handle *trans, struct
EXTENT_WRITEBACK); EXTENT_WRITEBACK);
if (ret) { if (ret) {
mutex_unlock(&info->extent_ins_mutex); mutex_unlock(&info->extent_ins_mutex);
if (search) { if (all && search) {
search = 0; search = 0;
continue; continue;
} }
...@@ -2088,7 +2091,10 @@ static int del_pending_extents(struct btrfs_trans_handle *trans, struct ...@@ -2088,7 +2091,10 @@ static int del_pending_extents(struct btrfs_trans_handle *trans, struct
err = ret; err = ret;
unlock_extent(extent_ins, start, end, GFP_NOFS); unlock_extent(extent_ins, start, end, GFP_NOFS);
search = 0; if (all)
search = 0;
else
search = end + 1;
cond_resched(); cond_resched();
} }
return err; return err;
...@@ -2155,8 +2161,8 @@ static int __btrfs_free_extent(struct btrfs_trans_handle *trans, ...@@ -2155,8 +2161,8 @@ static int __btrfs_free_extent(struct btrfs_trans_handle *trans,
root_objectid, ref_generation, root_objectid, ref_generation,
owner_objectid, pin, pin == 0); owner_objectid, pin, pin == 0);
finish_current_insert(trans, root->fs_info->extent_root); finish_current_insert(trans, root->fs_info->extent_root, 0);
pending_ret = del_pending_extents(trans, root->fs_info->extent_root); pending_ret = del_pending_extents(trans, root->fs_info->extent_root, 0);
return ret ? ret : pending_ret; return ret ? ret : pending_ret;
} }
...@@ -2580,8 +2586,8 @@ static int __btrfs_alloc_reserved_extent(struct btrfs_trans_handle *trans, ...@@ -2580,8 +2586,8 @@ static int __btrfs_alloc_reserved_extent(struct btrfs_trans_handle *trans,
trans->alloc_exclude_start = 0; trans->alloc_exclude_start = 0;
trans->alloc_exclude_nr = 0; trans->alloc_exclude_nr = 0;
btrfs_free_path(path); btrfs_free_path(path);
finish_current_insert(trans, extent_root); finish_current_insert(trans, extent_root, 0);
pending_ret = del_pending_extents(trans, extent_root); pending_ret = del_pending_extents(trans, extent_root, 0);
if (ret) if (ret)
goto out; goto out;
...@@ -5229,8 +5235,8 @@ int btrfs_make_block_group(struct btrfs_trans_handle *trans, ...@@ -5229,8 +5235,8 @@ int btrfs_make_block_group(struct btrfs_trans_handle *trans,
sizeof(cache->item)); sizeof(cache->item));
BUG_ON(ret); BUG_ON(ret);
finish_current_insert(trans, extent_root); finish_current_insert(trans, extent_root, 0);
ret = del_pending_extents(trans, extent_root); ret = del_pending_extents(trans, extent_root, 0);
BUG_ON(ret); BUG_ON(ret);
set_avail_alloc_bits(extent_root->fs_info, type); set_avail_alloc_bits(extent_root->fs_info, type);
......
...@@ -430,7 +430,10 @@ static int update_cowonly_root(struct btrfs_trans_handle *trans, ...@@ -430,7 +430,10 @@ static int update_cowonly_root(struct btrfs_trans_handle *trans,
u64 old_root_bytenr; u64 old_root_bytenr;
struct btrfs_root *tree_root = root->fs_info->tree_root; struct btrfs_root *tree_root = root->fs_info->tree_root;
btrfs_extent_post_op(trans, root);
btrfs_write_dirty_block_groups(trans, root); btrfs_write_dirty_block_groups(trans, root);
btrfs_extent_post_op(trans, root);
while(1) { while(1) {
old_root_bytenr = btrfs_root_bytenr(&root->root_item); old_root_bytenr = btrfs_root_bytenr(&root->root_item);
if (old_root_bytenr == root->node->start) if (old_root_bytenr == root->node->start)
...@@ -440,11 +443,15 @@ static int update_cowonly_root(struct btrfs_trans_handle *trans, ...@@ -440,11 +443,15 @@ static int update_cowonly_root(struct btrfs_trans_handle *trans,
btrfs_set_root_level(&root->root_item, btrfs_set_root_level(&root->root_item,
btrfs_header_level(root->node)); btrfs_header_level(root->node));
btrfs_set_root_generation(&root->root_item, trans->transid); btrfs_set_root_generation(&root->root_item, trans->transid);
btrfs_extent_post_op(trans, root);
ret = btrfs_update_root(trans, tree_root, ret = btrfs_update_root(trans, tree_root,
&root->root_key, &root->root_key,
&root->root_item); &root->root_item);
BUG_ON(ret); BUG_ON(ret);
btrfs_write_dirty_block_groups(trans, root); btrfs_write_dirty_block_groups(trans, root);
btrfs_extent_post_op(trans, root);
} }
return 0; return 0;
} }
...@@ -459,15 +466,20 @@ int btrfs_commit_tree_roots(struct btrfs_trans_handle *trans, ...@@ -459,15 +466,20 @@ int btrfs_commit_tree_roots(struct btrfs_trans_handle *trans,
struct list_head *next; struct list_head *next;
struct extent_buffer *eb; struct extent_buffer *eb;
btrfs_extent_post_op(trans, fs_info->tree_root);
eb = btrfs_lock_root_node(fs_info->tree_root); eb = btrfs_lock_root_node(fs_info->tree_root);
btrfs_cow_block(trans, fs_info->tree_root, eb, NULL, 0, &eb, 0); btrfs_cow_block(trans, fs_info->tree_root, eb, NULL, 0, &eb, 0);
btrfs_tree_unlock(eb); btrfs_tree_unlock(eb);
free_extent_buffer(eb); free_extent_buffer(eb);
btrfs_extent_post_op(trans, fs_info->tree_root);
while(!list_empty(&fs_info->dirty_cowonly_roots)) { while(!list_empty(&fs_info->dirty_cowonly_roots)) {
next = fs_info->dirty_cowonly_roots.next; next = fs_info->dirty_cowonly_roots.next;
list_del_init(next); list_del_init(next);
root = list_entry(next, struct btrfs_root, dirty_list); root = list_entry(next, struct btrfs_root, dirty_list);
update_cowonly_root(trans, root); update_cowonly_root(trans, root);
} }
return 0; return 0;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment