diff options
Diffstat (limited to 'fs/btrfs/dev-replace.c')
| -rw-r--r-- | fs/btrfs/dev-replace.c | 59 | 
1 files changed, 19 insertions, 40 deletions
| diff --git a/fs/btrfs/dev-replace.c b/fs/btrfs/dev-replace.c index 71fd99b48283..a7dd6ba25e99 100644 --- a/fs/btrfs/dev-replace.c +++ b/fs/btrfs/dev-replace.c @@ -474,6 +474,7 @@ static int mark_block_group_to_copy(struct btrfs_fs_info *fs_info,  	struct btrfs_dev_extent *dev_extent = NULL;  	struct btrfs_block_group *cache;  	struct btrfs_trans_handle *trans; +	int iter_ret = 0;  	int ret = 0;  	u64 chunk_offset; @@ -524,29 +525,8 @@ static int mark_block_group_to_copy(struct btrfs_fs_info *fs_info,  	key.type = BTRFS_DEV_EXTENT_KEY;  	key.offset = 0; -	ret = btrfs_search_slot(NULL, root, &key, path, 0, 0); -	if (ret < 0) -		goto free_path; -	if (ret > 0) { -		if (path->slots[0] >= -		    btrfs_header_nritems(path->nodes[0])) { -			ret = btrfs_next_leaf(root, path); -			if (ret < 0) -				goto free_path; -			if (ret > 0) { -				ret = 0; -				goto free_path; -			} -		} else { -			ret = 0; -		} -	} - -	while (1) { +	btrfs_for_each_slot(root, &key, &found_key, path, iter_ret) {  		struct extent_buffer *leaf = path->nodes[0]; -		int slot = path->slots[0]; - -		btrfs_item_key_to_cpu(leaf, &found_key, slot);  		if (found_key.objectid != src_dev->devid)  			break; @@ -557,30 +537,23 @@ static int mark_block_group_to_copy(struct btrfs_fs_info *fs_info,  		if (found_key.offset < key.offset)  			break; -		dev_extent = btrfs_item_ptr(leaf, slot, struct btrfs_dev_extent); +		dev_extent = btrfs_item_ptr(leaf, path->slots[0], struct btrfs_dev_extent);  		chunk_offset = btrfs_dev_extent_chunk_offset(leaf, dev_extent);  		cache = btrfs_lookup_block_group(fs_info, chunk_offset);  		if (!cache) -			goto skip; +			continue;  		spin_lock(&cache->lock);  		cache->to_copy = 1;  		spin_unlock(&cache->lock);  		btrfs_put_block_group(cache); - -skip: -		ret = btrfs_next_item(root, path); -		if (ret != 0) { -			if (ret > 0) -				ret = 0; -			break; -		}  	} +	if (iter_ret < 0) +		ret = iter_ret; -free_path:  	btrfs_free_path(path);  unlock:  	mutex_unlock(&fs_info->chunk_mutex); @@ -734,7 +707,12 @@ static int btrfs_dev_replace_start(struct btrfs_fs_info *fs_info,  	btrfs_wait_ordered_roots(fs_info, U64_MAX, 0, (u64)-1); -	/* Commit dev_replace state and reserve 1 item for it. */ +	/* +	 * Commit dev_replace state and reserve 1 item for it. +	 * This is crucial to ensure we won't miss copying extents for new block +	 * groups that are allocated after we started the device replace, and +	 * must be done after setting up the device replace state. +	 */  	trans = btrfs_start_transaction(root, 1);  	if (IS_ERR(trans)) {  		ret = PTR_ERR(trans); @@ -876,6 +854,7 @@ static int btrfs_dev_replace_finishing(struct btrfs_fs_info *fs_info,  				       int scrub_ret)  {  	struct btrfs_dev_replace *dev_replace = &fs_info->dev_replace; +	struct btrfs_fs_devices *fs_devices = fs_info->fs_devices;  	struct btrfs_device *tgt_device;  	struct btrfs_device *src_device;  	struct btrfs_root *root = fs_info->tree_root; @@ -925,12 +904,12 @@ static int btrfs_dev_replace_finishing(struct btrfs_fs_info *fs_info,  		WARN_ON(ret);  		/* Prevent write_all_supers() during the finishing procedure */ -		mutex_lock(&fs_info->fs_devices->device_list_mutex); +		mutex_lock(&fs_devices->device_list_mutex);  		/* Prevent new chunks being allocated on the source device */  		mutex_lock(&fs_info->chunk_mutex);  		if (!list_empty(&src_device->post_commit_list)) { -			mutex_unlock(&fs_info->fs_devices->device_list_mutex); +			mutex_unlock(&fs_devices->device_list_mutex);  			mutex_unlock(&fs_info->chunk_mutex);  		} else {  			break; @@ -967,7 +946,7 @@ static int btrfs_dev_replace_finishing(struct btrfs_fs_info *fs_info,  error:  		up_write(&dev_replace->rwsem);  		mutex_unlock(&fs_info->chunk_mutex); -		mutex_unlock(&fs_info->fs_devices->device_list_mutex); +		mutex_unlock(&fs_devices->device_list_mutex);  		btrfs_rm_dev_replace_blocked(fs_info);  		if (tgt_device)  			btrfs_destroy_dev_replace_tgtdev(tgt_device); @@ -996,8 +975,8 @@ error:  	btrfs_assign_next_active_device(src_device, tgt_device); -	list_add(&tgt_device->dev_alloc_list, &fs_info->fs_devices->alloc_list); -	fs_info->fs_devices->rw_devices++; +	list_add(&tgt_device->dev_alloc_list, &fs_devices->alloc_list); +	fs_devices->rw_devices++;  	up_write(&dev_replace->rwsem);  	btrfs_rm_dev_replace_blocked(fs_info); @@ -1020,7 +999,7 @@ error:  	 * belong to this filesystem.  	 */  	mutex_unlock(&fs_info->chunk_mutex); -	mutex_unlock(&fs_info->fs_devices->device_list_mutex); +	mutex_unlock(&fs_devices->device_list_mutex);  	/* replace the sysfs entry */  	btrfs_sysfs_remove_device(src_device); | 
