diff options
Diffstat (limited to 'fs/btrfs/ctree.c')
| -rw-r--r-- | fs/btrfs/ctree.c | 74 | 
1 files changed, 46 insertions, 28 deletions
| diff --git a/fs/btrfs/ctree.c b/fs/btrfs/ctree.c index 531e0a8645b0..b88a79e69ddf 100644 --- a/fs/btrfs/ctree.c +++ b/fs/btrfs/ctree.c @@ -1032,14 +1032,17 @@ static noinline int update_ref_for_cow(struct btrfs_trans_handle *trans,  		     root->root_key.objectid == BTRFS_TREE_RELOC_OBJECTID) &&  		    !(flags & BTRFS_BLOCK_FLAG_FULL_BACKREF)) {  			ret = btrfs_inc_ref(trans, root, buf, 1); -			BUG_ON(ret); /* -ENOMEM */ +			if (ret) +				return ret;  			if (root->root_key.objectid ==  			    BTRFS_TREE_RELOC_OBJECTID) {  				ret = btrfs_dec_ref(trans, root, buf, 0); -				BUG_ON(ret); /* -ENOMEM */ +				if (ret) +					return ret;  				ret = btrfs_inc_ref(trans, root, cow, 1); -				BUG_ON(ret); /* -ENOMEM */ +				if (ret) +					return ret;  			}  			new_flags |= BTRFS_BLOCK_FLAG_FULL_BACKREF;  		} else { @@ -1049,7 +1052,8 @@ static noinline int update_ref_for_cow(struct btrfs_trans_handle *trans,  				ret = btrfs_inc_ref(trans, root, cow, 1);  			else  				ret = btrfs_inc_ref(trans, root, cow, 0); -			BUG_ON(ret); /* -ENOMEM */ +			if (ret) +				return ret;  		}  		if (new_flags != 0) {  			int level = btrfs_header_level(buf); @@ -1068,9 +1072,11 @@ static noinline int update_ref_for_cow(struct btrfs_trans_handle *trans,  				ret = btrfs_inc_ref(trans, root, cow, 1);  			else  				ret = btrfs_inc_ref(trans, root, cow, 0); -			BUG_ON(ret); /* -ENOMEM */ +			if (ret) +				return ret;  			ret = btrfs_dec_ref(trans, root, buf, 1); -			BUG_ON(ret); /* -ENOMEM */ +			if (ret) +				return ret;  		}  		clean_tree_block(fs_info, buf);  		*last_ref = 1; @@ -1801,8 +1807,8 @@ static noinline int generic_bin_search(struct extent_buffer *eb,   * simple bin_search frontend that does the right thing for   * leaves vs nodes   */ -static int bin_search(struct extent_buffer *eb, const struct btrfs_key *key, -		      int level, int *slot) +int btrfs_bin_search(struct extent_buffer *eb, const struct btrfs_key *key, +		     int level, int *slot)  {  	if (level == 0)  		return generic_bin_search(eb, @@ -1818,12 +1824,6 @@ static int bin_search(struct extent_buffer *eb, const struct btrfs_key *key,  					  slot);  } -int btrfs_bin_search(struct extent_buffer *eb, const struct btrfs_key *key, -		     int level, int *slot) -{ -	return bin_search(eb, key, level, slot); -} -  static void root_add_used(struct btrfs_root *root, u32 size)  {  	spin_lock(&root->accounting_lock); @@ -2608,7 +2608,7 @@ static int key_search(struct extent_buffer *b, const struct btrfs_key *key,  		      int level, int *prev_cmp, int *slot)  {  	if (*prev_cmp != 0) { -		*prev_cmp = bin_search(b, key, level, slot); +		*prev_cmp = btrfs_bin_search(b, key, level, slot);  		return *prev_cmp;  	} @@ -2654,17 +2654,29 @@ int btrfs_find_item(struct btrfs_root *fs_root, struct btrfs_path *path,  }  /* - * look for key in the tree.  path is filled in with nodes along the way - * if key is found, we return zero and you can find the item in the leaf - * level of the path (level 0) + * btrfs_search_slot - look for a key in a tree and perform necessary + * modifications to preserve tree invariants.   * - * If the key isn't found, the path points to the slot where it should - * be inserted, and 1 is returned.  If there are other errors during the - * search a negative error number is returned. + * @trans:	Handle of transaction, used when modifying the tree + * @p:		Holds all btree nodes along the search path + * @root:	The root node of the tree + * @key:	The key we are looking for + * @ins_len:	Indicates purpose of search, for inserts it is 1, for + *		deletions it's -1. 0 for plain searches + * @cow:	boolean should CoW operations be performed. Must always be 1 + *		when modifying the tree.   * - * if ins_len > 0, nodes and leaves will be split as we walk down the - * tree.  if ins_len < 0, nodes will be merged as we walk down the tree (if - * possible) + * If @ins_len > 0, nodes and leaves will be split as we walk down the tree. + * If @ins_len < 0, nodes will be merged as we walk down the tree (if possible) + * + * If @key is found, 0 is returned and you can find the item in the leaf level + * of the path (level 0) + * + * If @key isn't found, 1 is returned and the leaf level of the path (level 0) + * points to the slot where it should be inserted + * + * If an error is encountered while searching the tree a negative error number + * is returned   */  int btrfs_search_slot(struct btrfs_trans_handle *trans, struct btrfs_root *root,  		      const struct btrfs_key *key, struct btrfs_path *p, @@ -2768,6 +2780,8 @@ again:  		 * contention with the cow code  		 */  		if (cow) { +			bool last_level = (level == (BTRFS_MAX_LEVEL - 1)); +  			/*  			 * if we don't really need to cow this block  			 * then we don't want to set the path blocking, @@ -2792,9 +2806,13 @@ again:  			}  			btrfs_set_path_blocking(p); -			err = btrfs_cow_block(trans, root, b, -					      p->nodes[level + 1], -					      p->slots[level + 1], &b); +			if (last_level) +				err = btrfs_cow_block(trans, root, b, NULL, 0, +						      &b); +			else +				err = btrfs_cow_block(trans, root, b, +						      p->nodes[level + 1], +						      p->slots[level + 1], &b);  			if (err) {  				ret = err;  				goto done; @@ -5169,7 +5187,7 @@ again:  	while (1) {  		nritems = btrfs_header_nritems(cur);  		level = btrfs_header_level(cur); -		sret = bin_search(cur, min_key, level, &slot); +		sret = btrfs_bin_search(cur, min_key, level, &slot);  		/* at the lowest level, we're done, setup the path and exit */  		if (level == path->lowest_level) { | 
