diff options
Diffstat (limited to 'fs/btrfs')
| -rw-r--r-- | fs/btrfs/backref.c | 11 | ||||
| -rw-r--r-- | fs/btrfs/ctree.h | 7 | ||||
| -rw-r--r-- | fs/btrfs/delayed-ref.c | 3 | ||||
| -rw-r--r-- | fs/btrfs/extent-tree.c | 4 | ||||
| -rw-r--r-- | fs/btrfs/inode-item.c | 44 | ||||
| -rw-r--r-- | fs/btrfs/inode.c | 52 | ||||
| -rw-r--r-- | fs/btrfs/qgroup.c | 9 | ||||
| -rw-r--r-- | fs/btrfs/relocation.c | 18 | ||||
| -rw-r--r-- | fs/btrfs/send.c | 3 | ||||
| -rw-r--r-- | fs/btrfs/super.c | 2 | ||||
| -rw-r--r-- | fs/btrfs/sysfs.c | 8 | ||||
| -rw-r--r-- | fs/btrfs/transaction.c | 20 | ||||
| -rw-r--r-- | fs/btrfs/tree-log.c | 146 | ||||
| -rw-r--r-- | fs/btrfs/volumes.c | 12 | 
14 files changed, 271 insertions, 68 deletions
| diff --git a/fs/btrfs/backref.c b/fs/btrfs/backref.c index e4054e533f6d..f94b2d8c744a 100644 --- a/fs/btrfs/backref.c +++ b/fs/btrfs/backref.c @@ -1264,7 +1264,16 @@ again:  	while (node) {  		ref = rb_entry(node, struct prelim_ref, rbnode);  		node = rb_next(&ref->rbnode); -		WARN_ON(ref->count < 0); +		/* +		 * ref->count < 0 can happen here if there are delayed +		 * refs with a node->action of BTRFS_DROP_DELAYED_REF. +		 * prelim_ref_insert() relies on this when merging +		 * identical refs to keep the overall count correct. +		 * prelim_ref_insert() will merge only those refs +		 * which compare identically.  Any refs having +		 * e.g. different offsets would not be merged, +		 * and would retain their original ref->count < 0. +		 */  		if (roots && ref->count && ref->root_id && ref->parent == 0) {  			if (sc && sc->root_objectid &&  			    ref->root_id != sc->root_objectid) { diff --git a/fs/btrfs/ctree.h b/fs/btrfs/ctree.h index 1a462ab85c49..da308774b8a4 100644 --- a/fs/btrfs/ctree.h +++ b/fs/btrfs/ctree.h @@ -2974,7 +2974,7 @@ static inline void free_fs_info(struct btrfs_fs_info *fs_info)  	kfree(fs_info->super_copy);  	kfree(fs_info->super_for_commit);  	security_free_mnt_opts(&fs_info->security_opts); -	kfree(fs_info); +	kvfree(fs_info);  }  /* tree mod log functions from ctree.c */ @@ -3095,7 +3095,10 @@ btrfs_lookup_inode_extref(struct btrfs_trans_handle *trans,  			  u64 inode_objectid, u64 ref_objectid, int ins_len,  			  int cow); -int btrfs_find_name_in_ext_backref(struct btrfs_path *path, +int btrfs_find_name_in_backref(struct extent_buffer *leaf, int slot, +			       const char *name, +			       int name_len, struct btrfs_inode_ref **ref_ret); +int btrfs_find_name_in_ext_backref(struct extent_buffer *leaf, int slot,  				   u64 ref_objectid, const char *name,  				   int name_len,  				   struct btrfs_inode_extref **extref_ret); diff --git a/fs/btrfs/delayed-ref.c b/fs/btrfs/delayed-ref.c index a1a40cf382e3..7ab5e0128f0c 100644 --- a/fs/btrfs/delayed-ref.c +++ b/fs/btrfs/delayed-ref.c @@ -821,7 +821,8 @@ int btrfs_add_delayed_tree_ref(struct btrfs_fs_info *fs_info,  	spin_unlock(&delayed_refs->lock);  	if (qrecord_inserted) -		return btrfs_qgroup_trace_extent_post(fs_info, record); +		btrfs_qgroup_trace_extent_post(fs_info, record); +  	return 0;  free_head_ref: diff --git a/fs/btrfs/extent-tree.c b/fs/btrfs/extent-tree.c index 05751a677da4..c1618ab9fecf 100644 --- a/fs/btrfs/extent-tree.c +++ b/fs/btrfs/extent-tree.c @@ -2147,6 +2147,10 @@ int btrfs_discard_extent(struct btrfs_fs_info *fs_info, u64 bytenr,  			u64 bytes;  			struct request_queue *req_q; +			if (!stripe->dev->bdev) { +				ASSERT(btrfs_test_opt(fs_info, DEGRADED)); +				continue; +			}  			req_q = bdev_get_queue(stripe->dev->bdev);  			if (!blk_queue_discard(req_q))  				continue; diff --git a/fs/btrfs/inode-item.c b/fs/btrfs/inode-item.c index 39c968f80157..65e1a76bf755 100644 --- a/fs/btrfs/inode-item.c +++ b/fs/btrfs/inode-item.c @@ -22,10 +22,10 @@  #include "transaction.h"  #include "print-tree.h" -static int find_name_in_backref(struct btrfs_path *path, const char *name, -			 int name_len, struct btrfs_inode_ref **ref_ret) +int btrfs_find_name_in_backref(struct extent_buffer *leaf, int slot, +			       const char *name, +			       int name_len, struct btrfs_inode_ref **ref_ret)  { -	struct extent_buffer *leaf;  	struct btrfs_inode_ref *ref;  	unsigned long ptr;  	unsigned long name_ptr; @@ -33,9 +33,8 @@ static int find_name_in_backref(struct btrfs_path *path, const char *name,  	u32 cur_offset = 0;  	int len; -	leaf = path->nodes[0]; -	item_size = btrfs_item_size_nr(leaf, path->slots[0]); -	ptr = btrfs_item_ptr_offset(leaf, path->slots[0]); +	item_size = btrfs_item_size_nr(leaf, slot); +	ptr = btrfs_item_ptr_offset(leaf, slot);  	while (cur_offset < item_size) {  		ref = (struct btrfs_inode_ref *)(ptr + cur_offset);  		len = btrfs_inode_ref_name_len(leaf, ref); @@ -44,18 +43,19 @@ static int find_name_in_backref(struct btrfs_path *path, const char *name,  		if (len != name_len)  			continue;  		if (memcmp_extent_buffer(leaf, name, name_ptr, name_len) == 0) { -			*ref_ret = ref; +			if (ref_ret) +				*ref_ret = ref;  			return 1;  		}  	}  	return 0;  } -int btrfs_find_name_in_ext_backref(struct btrfs_path *path, u64 ref_objectid, +int btrfs_find_name_in_ext_backref(struct extent_buffer *leaf, int slot, +				   u64 ref_objectid,  				   const char *name, int name_len,  				   struct btrfs_inode_extref **extref_ret)  { -	struct extent_buffer *leaf;  	struct btrfs_inode_extref *extref;  	unsigned long ptr;  	unsigned long name_ptr; @@ -63,9 +63,8 @@ int btrfs_find_name_in_ext_backref(struct btrfs_path *path, u64 ref_objectid,  	u32 cur_offset = 0;  	int ref_name_len; -	leaf = path->nodes[0]; -	item_size = btrfs_item_size_nr(leaf, path->slots[0]); -	ptr = btrfs_item_ptr_offset(leaf, path->slots[0]); +	item_size = btrfs_item_size_nr(leaf, slot); +	ptr = btrfs_item_ptr_offset(leaf, slot);  	/*  	 * Search all extended backrefs in this item. We're only @@ -113,7 +112,9 @@ btrfs_lookup_inode_extref(struct btrfs_trans_handle *trans,  		return ERR_PTR(ret);  	if (ret > 0)  		return NULL; -	if (!btrfs_find_name_in_ext_backref(path, ref_objectid, name, name_len, &extref)) +	if (!btrfs_find_name_in_ext_backref(path->nodes[0], path->slots[0], +					    ref_objectid, name, name_len, +					    &extref))  		return NULL;  	return extref;  } @@ -155,7 +156,8 @@ static int btrfs_del_inode_extref(struct btrfs_trans_handle *trans,  	 * This should always succeed so error here will make the FS  	 * readonly.  	 */ -	if (!btrfs_find_name_in_ext_backref(path, ref_objectid, +	if (!btrfs_find_name_in_ext_backref(path->nodes[0], path->slots[0], +					    ref_objectid,  					    name, name_len, &extref)) {  		btrfs_handle_fs_error(root->fs_info, -ENOENT, NULL);  		ret = -EROFS; @@ -225,7 +227,8 @@ int btrfs_del_inode_ref(struct btrfs_trans_handle *trans,  	} else if (ret < 0) {  		goto out;  	} -	if (!find_name_in_backref(path, name, name_len, &ref)) { +	if (!btrfs_find_name_in_backref(path->nodes[0], path->slots[0], +					name, name_len, &ref)) {  		ret = -ENOENT;  		search_ext_refs = 1;  		goto out; @@ -293,7 +296,9 @@ static int btrfs_insert_inode_extref(struct btrfs_trans_handle *trans,  	ret = btrfs_insert_empty_item(trans, root, path, &key,  				      ins_len);  	if (ret == -EEXIST) { -		if (btrfs_find_name_in_ext_backref(path, ref_objectid, +		if (btrfs_find_name_in_ext_backref(path->nodes[0], +						   path->slots[0], +						   ref_objectid,  						   name, name_len, NULL))  			goto out; @@ -351,7 +356,8 @@ int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans,  	if (ret == -EEXIST) {  		u32 old_size; -		if (find_name_in_backref(path, name, name_len, &ref)) +		if (btrfs_find_name_in_backref(path->nodes[0], path->slots[0], +					       name, name_len, &ref))  			goto out;  		old_size = btrfs_item_size_nr(path->nodes[0], path->slots[0]); @@ -365,7 +371,9 @@ int btrfs_insert_inode_ref(struct btrfs_trans_handle *trans,  		ret = 0;  	} else if (ret < 0) {  		if (ret == -EOVERFLOW) { -			if (find_name_in_backref(path, name, name_len, &ref)) +			if (btrfs_find_name_in_backref(path->nodes[0], +						       path->slots[0], +						       name, name_len, &ref))  				ret = -EEXIST;  			else  				ret = -EMLINK; diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c index 53ca025655fc..f53470112670 100644 --- a/fs/btrfs/inode.c +++ b/fs/btrfs/inode.c @@ -1335,8 +1335,11 @@ next_slot:  		leaf = path->nodes[0];  		if (path->slots[0] >= btrfs_header_nritems(leaf)) {  			ret = btrfs_next_leaf(root, path); -			if (ret < 0) +			if (ret < 0) { +				if (cow_start != (u64)-1) +					cur_offset = cow_start;  				goto error; +			}  			if (ret > 0)  				break;  			leaf = path->nodes[0]; @@ -2040,12 +2043,15 @@ static noinline int add_pending_csums(struct btrfs_trans_handle *trans,  			     struct inode *inode, struct list_head *list)  {  	struct btrfs_ordered_sum *sum; +	int ret;  	list_for_each_entry(sum, list, list) {  		trans->adding_csums = true; -		btrfs_csum_file_blocks(trans, +		ret = btrfs_csum_file_blocks(trans,  		       BTRFS_I(inode)->root->fs_info->csum_root, sum);  		trans->adding_csums = false; +		if (ret) +			return ret;  	}  	return 0;  } @@ -3059,7 +3065,11 @@ static int btrfs_finish_ordered_io(struct btrfs_ordered_extent *ordered_extent)  		goto out;  	} -	add_pending_csums(trans, inode, &ordered_extent->list); +	ret = add_pending_csums(trans, inode, &ordered_extent->list); +	if (ret) { +		btrfs_abort_transaction(trans, ret); +		goto out; +	}  	btrfs_ordered_update_i_size(inode, 0, ordered_extent);  	ret = btrfs_update_inode_fallback(trans, root, inode); @@ -3385,6 +3395,11 @@ int btrfs_orphan_add(struct btrfs_trans_handle *trans,  		ret = btrfs_orphan_reserve_metadata(trans, inode);  		ASSERT(!ret);  		if (ret) { +			/* +			 * dec doesn't need spin_lock as ->orphan_block_rsv +			 * would be released only if ->orphan_inodes is +			 * zero. +			 */  			atomic_dec(&root->orphan_inodes);  			clear_bit(BTRFS_INODE_ORPHAN_META_RESERVED,  				  &inode->runtime_flags); @@ -3399,12 +3414,17 @@ int btrfs_orphan_add(struct btrfs_trans_handle *trans,  	if (insert >= 1) {  		ret = btrfs_insert_orphan_item(trans, root, btrfs_ino(inode));  		if (ret) { -			atomic_dec(&root->orphan_inodes);  			if (reserve) {  				clear_bit(BTRFS_INODE_ORPHAN_META_RESERVED,  					  &inode->runtime_flags);  				btrfs_orphan_release_metadata(inode);  			} +			/* +			 * btrfs_orphan_commit_root may race with us and set +			 * ->orphan_block_rsv to zero, in order to avoid that, +			 * decrease ->orphan_inodes after everything is done. +			 */ +			atomic_dec(&root->orphan_inodes);  			if (ret != -EEXIST) {  				clear_bit(BTRFS_INODE_HAS_ORPHAN_ITEM,  					  &inode->runtime_flags); @@ -3436,28 +3456,26 @@ static int btrfs_orphan_del(struct btrfs_trans_handle *trans,  {  	struct btrfs_root *root = inode->root;  	int delete_item = 0; -	int release_rsv = 0;  	int ret = 0; -	spin_lock(&root->orphan_lock);  	if (test_and_clear_bit(BTRFS_INODE_HAS_ORPHAN_ITEM,  			       &inode->runtime_flags))  		delete_item = 1; +	if (delete_item && trans) +		ret = btrfs_del_orphan_item(trans, root, btrfs_ino(inode)); +  	if (test_and_clear_bit(BTRFS_INODE_ORPHAN_META_RESERVED,  			       &inode->runtime_flags)) -		release_rsv = 1; -	spin_unlock(&root->orphan_lock); +		btrfs_orphan_release_metadata(inode); -	if (delete_item) { +	/* +	 * btrfs_orphan_commit_root may race with us and set ->orphan_block_rsv +	 * to zero, in order to avoid that, decrease ->orphan_inodes after +	 * everything is done. +	 */ +	if (delete_item)  		atomic_dec(&root->orphan_inodes); -		if (trans) -			ret = btrfs_del_orphan_item(trans, root, -						    btrfs_ino(inode)); -	} - -	if (release_rsv) -		btrfs_orphan_release_metadata(inode);  	return ret;  } @@ -5281,7 +5299,7 @@ void btrfs_evict_inode(struct inode *inode)  	trace_btrfs_inode_evict(inode);  	if (!root) { -		kmem_cache_free(btrfs_inode_cachep, BTRFS_I(inode)); +		clear_inode(inode);  		return;  	} diff --git a/fs/btrfs/qgroup.c b/fs/btrfs/qgroup.c index 9e61dd624f7b..aa259d6986e1 100644 --- a/fs/btrfs/qgroup.c +++ b/fs/btrfs/qgroup.c @@ -1442,8 +1442,13 @@ int btrfs_qgroup_trace_extent_post(struct btrfs_fs_info *fs_info,  	int ret;  	ret = btrfs_find_all_roots(NULL, fs_info, bytenr, 0, &old_root, false); -	if (ret < 0) -		return ret; +	if (ret < 0) { +		fs_info->qgroup_flags |= BTRFS_QGROUP_STATUS_FLAG_INCONSISTENT; +		btrfs_warn(fs_info, +"error accounting new delayed refs extent (err code: %d), quota inconsistent", +			ret); +		return 0; +	}  	/*  	 * Here we don't need to get the lock of diff --git a/fs/btrfs/relocation.c b/fs/btrfs/relocation.c index f0c3f00e97cb..cd2298d185dd 100644 --- a/fs/btrfs/relocation.c +++ b/fs/btrfs/relocation.c @@ -3268,8 +3268,22 @@ static int relocate_file_extent_cluster(struct inode *inode,  			nr++;  		} -		btrfs_set_extent_delalloc(inode, page_start, page_end, 0, NULL, -					  0); +		ret = btrfs_set_extent_delalloc(inode, page_start, page_end, 0, +						NULL, 0); +		if (ret) { +			unlock_page(page); +			put_page(page); +			btrfs_delalloc_release_metadata(BTRFS_I(inode), +							 PAGE_SIZE); +			btrfs_delalloc_release_extents(BTRFS_I(inode), +			                               PAGE_SIZE); + +			clear_extent_bits(&BTRFS_I(inode)->io_tree, +					  page_start, page_end, +					  EXTENT_LOCKED | EXTENT_BOUNDARY); +			goto out; + +		}  		set_page_dirty(page);  		unlock_extent(&BTRFS_I(inode)->io_tree, diff --git a/fs/btrfs/send.c b/fs/btrfs/send.c index f306c608dc28..484e2af793de 100644 --- a/fs/btrfs/send.c +++ b/fs/btrfs/send.c @@ -5005,6 +5005,9 @@ static int send_hole(struct send_ctx *sctx, u64 end)  	u64 len;  	int ret = 0; +	if (sctx->flags & BTRFS_SEND_FLAG_NO_FILE_DATA) +		return send_update_extent(sctx, offset, end - offset); +  	p = fs_path_alloc();  	if (!p)  		return -ENOMEM; diff --git a/fs/btrfs/super.c b/fs/btrfs/super.c index 6e71a2a78363..4b817947e00f 100644 --- a/fs/btrfs/super.c +++ b/fs/btrfs/super.c @@ -1545,7 +1545,7 @@ static struct dentry *btrfs_mount_root(struct file_system_type *fs_type,  	 * it for searching for existing supers, so this lets us do that and  	 * then open_ctree will properly initialize everything later.  	 */ -	fs_info = kzalloc(sizeof(struct btrfs_fs_info), GFP_KERNEL); +	fs_info = kvzalloc(sizeof(struct btrfs_fs_info), GFP_KERNEL);  	if (!fs_info) {  		error = -ENOMEM;  		goto error_sec_opts; diff --git a/fs/btrfs/sysfs.c b/fs/btrfs/sysfs.c index a8bafed931f4..d11c70bff5a9 100644 --- a/fs/btrfs/sysfs.c +++ b/fs/btrfs/sysfs.c @@ -423,7 +423,7 @@ static ssize_t btrfs_nodesize_show(struct kobject *kobj,  {  	struct btrfs_fs_info *fs_info = to_fs_info(kobj); -	return snprintf(buf, PAGE_SIZE, "%u\n", fs_info->super_copy->nodesize); +	return snprintf(buf, PAGE_SIZE, "%u\n", fs_info->nodesize);  }  BTRFS_ATTR(, nodesize, btrfs_nodesize_show); @@ -433,8 +433,7 @@ static ssize_t btrfs_sectorsize_show(struct kobject *kobj,  {  	struct btrfs_fs_info *fs_info = to_fs_info(kobj); -	return snprintf(buf, PAGE_SIZE, "%u\n", -			fs_info->super_copy->sectorsize); +	return snprintf(buf, PAGE_SIZE, "%u\n", fs_info->sectorsize);  }  BTRFS_ATTR(, sectorsize, btrfs_sectorsize_show); @@ -444,8 +443,7 @@ static ssize_t btrfs_clone_alignment_show(struct kobject *kobj,  {  	struct btrfs_fs_info *fs_info = to_fs_info(kobj); -	return snprintf(buf, PAGE_SIZE, "%u\n", -			fs_info->super_copy->sectorsize); +	return snprintf(buf, PAGE_SIZE, "%u\n", fs_info->sectorsize);  }  BTRFS_ATTR(, clone_alignment, btrfs_clone_alignment_show); diff --git a/fs/btrfs/transaction.c b/fs/btrfs/transaction.c index 04f07144b45c..9220f004001c 100644 --- a/fs/btrfs/transaction.c +++ b/fs/btrfs/transaction.c @@ -1722,19 +1722,23 @@ static void update_super_roots(struct btrfs_fs_info *fs_info)  	super = fs_info->super_copy; +	/* update latest btrfs_super_block::chunk_root refs */  	root_item = &fs_info->chunk_root->root_item; -	super->chunk_root = root_item->bytenr; -	super->chunk_root_generation = root_item->generation; -	super->chunk_root_level = root_item->level; +	btrfs_set_super_chunk_root(super, root_item->bytenr); +	btrfs_set_super_chunk_root_generation(super, root_item->generation); +	btrfs_set_super_chunk_root_level(super, root_item->level); +	/* update latest btrfs_super_block::root refs */  	root_item = &fs_info->tree_root->root_item; -	super->root = root_item->bytenr; -	super->generation = root_item->generation; -	super->root_level = root_item->level; +	btrfs_set_super_root(super, root_item->bytenr); +	btrfs_set_super_generation(super, root_item->generation); +	btrfs_set_super_root_level(super, root_item->level); +  	if (btrfs_test_opt(fs_info, SPACE_CACHE)) -		super->cache_generation = root_item->generation; +		btrfs_set_super_cache_generation(super, root_item->generation);  	if (test_bit(BTRFS_FS_UPDATE_UUID_TREE_GEN, &fs_info->flags)) -		super->uuid_tree_generation = root_item->generation; +		btrfs_set_super_uuid_tree_generation(super, +						     root_item->generation);  }  int btrfs_transaction_in_commit(struct btrfs_fs_info *info) diff --git a/fs/btrfs/tree-log.c b/fs/btrfs/tree-log.c index afadaadab18e..434457794c27 100644 --- a/fs/btrfs/tree-log.c +++ b/fs/btrfs/tree-log.c @@ -29,6 +29,7 @@  #include "hash.h"  #include "compression.h"  #include "qgroup.h" +#include "inode-map.h"  /* magic values for the inode_only field in btrfs_log_inode:   * @@ -966,7 +967,9 @@ static noinline int backref_in_log(struct btrfs_root *log,  	ptr = btrfs_item_ptr_offset(path->nodes[0], path->slots[0]);  	if (key->type == BTRFS_INODE_EXTREF_KEY) { -		if (btrfs_find_name_in_ext_backref(path, ref_objectid, +		if (btrfs_find_name_in_ext_backref(path->nodes[0], +						   path->slots[0], +						   ref_objectid,  						   name, namelen, NULL))  			match = 1; @@ -1190,7 +1193,8 @@ static int extref_get_fields(struct extent_buffer *eb, unsigned long ref_ptr,  	read_extent_buffer(eb, *name, (unsigned long)&extref->name,  			   *namelen); -	*index = btrfs_inode_extref_index(eb, extref); +	if (index) +		*index = btrfs_inode_extref_index(eb, extref);  	if (parent_objectid)  		*parent_objectid = btrfs_inode_extref_parent(eb, extref); @@ -1211,12 +1215,102 @@ static int ref_get_fields(struct extent_buffer *eb, unsigned long ref_ptr,  	read_extent_buffer(eb, *name, (unsigned long)(ref + 1), *namelen); -	*index = btrfs_inode_ref_index(eb, ref); +	if (index) +		*index = btrfs_inode_ref_index(eb, ref);  	return 0;  }  /* + * Take an inode reference item from the log tree and iterate all names from the + * inode reference item in the subvolume tree with the same key (if it exists). + * For any name that is not in the inode reference item from the log tree, do a + * proper unlink of that name (that is, remove its entry from the inode + * reference item and both dir index keys). + */ +static int unlink_old_inode_refs(struct btrfs_trans_handle *trans, +				 struct btrfs_root *root, +				 struct btrfs_path *path, +				 struct btrfs_inode *inode, +				 struct extent_buffer *log_eb, +				 int log_slot, +				 struct btrfs_key *key) +{ +	int ret; +	unsigned long ref_ptr; +	unsigned long ref_end; +	struct extent_buffer *eb; + +again: +	btrfs_release_path(path); +	ret = btrfs_search_slot(NULL, root, key, path, 0, 0); +	if (ret > 0) { +		ret = 0; +		goto out; +	} +	if (ret < 0) +		goto out; + +	eb = path->nodes[0]; +	ref_ptr = btrfs_item_ptr_offset(eb, path->slots[0]); +	ref_end = ref_ptr + btrfs_item_size_nr(eb, path->slots[0]); +	while (ref_ptr < ref_end) { +		char *name = NULL; +		int namelen; +		u64 parent_id; + +		if (key->type == BTRFS_INODE_EXTREF_KEY) { +			ret = extref_get_fields(eb, ref_ptr, &namelen, &name, +						NULL, &parent_id); +		} else { +			parent_id = key->offset; +			ret = ref_get_fields(eb, ref_ptr, &namelen, &name, +					     NULL); +		} +		if (ret) +			goto out; + +		if (key->type == BTRFS_INODE_EXTREF_KEY) +			ret = btrfs_find_name_in_ext_backref(log_eb, log_slot, +							     parent_id, name, +							     namelen, NULL); +		else +			ret = btrfs_find_name_in_backref(log_eb, log_slot, name, +							 namelen, NULL); + +		if (!ret) { +			struct inode *dir; + +			btrfs_release_path(path); +			dir = read_one_inode(root, parent_id); +			if (!dir) { +				ret = -ENOENT; +				kfree(name); +				goto out; +			} +			ret = btrfs_unlink_inode(trans, root, BTRFS_I(dir), +						 inode, name, namelen); +			kfree(name); +			iput(dir); +			if (ret) +				goto out; +			goto again; +		} + +		kfree(name); +		ref_ptr += namelen; +		if (key->type == BTRFS_INODE_EXTREF_KEY) +			ref_ptr += sizeof(struct btrfs_inode_extref); +		else +			ref_ptr += sizeof(struct btrfs_inode_ref); +	} +	ret = 0; + out: +	btrfs_release_path(path); +	return ret; +} + +/*   * replay one inode back reference item found in the log tree.   * eb, slot and key refer to the buffer and key found in the log tree.   * root is the destination we are replaying into, and path is for temp @@ -1344,6 +1438,19 @@ static noinline int add_inode_ref(struct btrfs_trans_handle *trans,  		}  	} +	/* +	 * Before we overwrite the inode reference item in the subvolume tree +	 * with the item from the log tree, we must unlink all names from the +	 * parent directory that are in the subvolume's tree inode reference +	 * item, otherwise we end up with an inconsistent subvolume tree where +	 * dir index entries exist for a name but there is no inode reference +	 * item with the same name. +	 */ +	ret = unlink_old_inode_refs(trans, root, path, BTRFS_I(inode), eb, slot, +				    key); +	if (ret) +		goto out; +  	/* finally write the back reference in the inode */  	ret = overwrite_item(trans, root, path, eb, slot, key);  out: @@ -2472,6 +2579,9 @@ static noinline int walk_down_log_tree(struct btrfs_trans_handle *trans,  					clean_tree_block(fs_info, next);  					btrfs_wait_tree_block_writeback(next);  					btrfs_tree_unlock(next); +				} else { +					if (test_and_clear_bit(EXTENT_BUFFER_DIRTY, &next->bflags)) +						clear_extent_buffer_dirty(next);  				}  				WARN_ON(root_owner != @@ -2552,6 +2662,9 @@ static noinline int walk_up_log_tree(struct btrfs_trans_handle *trans,  					clean_tree_block(fs_info, next);  					btrfs_wait_tree_block_writeback(next);  					btrfs_tree_unlock(next); +				} else { +					if (test_and_clear_bit(EXTENT_BUFFER_DIRTY, &next->bflags)) +						clear_extent_buffer_dirty(next);  				}  				WARN_ON(root_owner != BTRFS_TREE_LOG_OBJECTID); @@ -2630,6 +2743,9 @@ static int walk_log_tree(struct btrfs_trans_handle *trans,  				clean_tree_block(fs_info, next);  				btrfs_wait_tree_block_writeback(next);  				btrfs_tree_unlock(next); +			} else { +				if (test_and_clear_bit(EXTENT_BUFFER_DIRTY, &next->bflags)) +					clear_extent_buffer_dirty(next);  			}  			WARN_ON(log->root_key.objectid != @@ -3018,13 +3134,14 @@ static void free_log_tree(struct btrfs_trans_handle *trans,  	while (1) {  		ret = find_first_extent_bit(&log->dirty_log_pages, -				0, &start, &end, EXTENT_DIRTY | EXTENT_NEW, +				0, &start, &end, +				EXTENT_DIRTY | EXTENT_NEW | EXTENT_NEED_WAIT,  				NULL);  		if (ret)  			break;  		clear_extent_bits(&log->dirty_log_pages, start, end, -				  EXTENT_DIRTY | EXTENT_NEW); +				  EXTENT_DIRTY | EXTENT_NEW | EXTENT_NEED_WAIT);  	}  	/* @@ -5677,6 +5794,23 @@ again:  						      path);  		} +		if (!ret && wc.stage == LOG_WALK_REPLAY_ALL) { +			struct btrfs_root *root = wc.replay_dest; + +			btrfs_release_path(path); + +			/* +			 * We have just replayed everything, and the highest +			 * objectid of fs roots probably has changed in case +			 * some inode_item's got replayed. +			 * +			 * root->objectid_mutex is not acquired as log replay +			 * could only happen during mount. +			 */ +			ret = btrfs_find_highest_objectid(root, +						  &root->highest_objectid); +		} +  		key.offset = found_key.offset - 1;  		wc.replay_dest->log_root = NULL;  		free_extent_buffer(log->node); @@ -5825,7 +5959,7 @@ int btrfs_log_new_name(struct btrfs_trans_handle *trans,  	 * this will force the logging code to walk the dentry chain  	 * up for the file  	 */ -	if (S_ISREG(inode->vfs_inode.i_mode)) +	if (!S_ISDIR(inode->vfs_inode.i_mode))  		inode->last_unlink_trans = trans->transid;  	/* diff --git a/fs/btrfs/volumes.c b/fs/btrfs/volumes.c index b5036bd69e6a..b2d05c6b1c56 100644 --- a/fs/btrfs/volumes.c +++ b/fs/btrfs/volumes.c @@ -645,6 +645,7 @@ static void btrfs_free_stale_devices(const char *path,  				btrfs_sysfs_remove_fsid(fs_devs);  				list_del(&fs_devs->list);  				free_fs_devices(fs_devs); +				break;  			} else {  				fs_devs->num_devices--;  				list_del(&dev->dev_list); @@ -4828,10 +4829,13 @@ static int __btrfs_alloc_chunk(struct btrfs_trans_handle *trans,  	ndevs = min(ndevs, devs_max);  	/* -	 * the primary goal is to maximize the number of stripes, so use as many -	 * devices as possible, even if the stripes are not maximum sized. +	 * The primary goal is to maximize the number of stripes, so use as +	 * many devices as possible, even if the stripes are not maximum sized. +	 * +	 * The DUP profile stores more than one stripe per device, the +	 * max_avail is the total size so we have to adjust.  	 */ -	stripe_size = devices_info[ndevs-1].max_avail; +	stripe_size = div_u64(devices_info[ndevs - 1].max_avail, dev_stripes);  	num_stripes = ndevs * dev_stripes;  	/* @@ -4866,8 +4870,6 @@ static int __btrfs_alloc_chunk(struct btrfs_trans_handle *trans,  			stripe_size = devices_info[ndevs-1].max_avail;  	} -	stripe_size = div_u64(stripe_size, dev_stripes); -  	/* align to BTRFS_STRIPE_LEN */  	stripe_size = round_down(stripe_size, BTRFS_STRIPE_LEN); | 
