diff options
Diffstat (limited to 'fs/btrfs/tree-log.c')
| -rw-r--r-- | fs/btrfs/tree-log.c | 50 | 
1 files changed, 24 insertions, 26 deletions
| diff --git a/fs/btrfs/tree-log.c b/fs/btrfs/tree-log.c index cd5348f352dd..ea8136dcf71f 100644 --- a/fs/btrfs/tree-log.c +++ b/fs/btrfs/tree-log.c @@ -3116,29 +3116,17 @@ int btrfs_sync_log(struct btrfs_trans_handle *trans,  	btrfs_init_log_ctx(&root_log_ctx, NULL);  	mutex_lock(&log_root_tree->log_mutex); -	atomic_inc(&log_root_tree->log_batch); -	atomic_inc(&log_root_tree->log_writers);  	index2 = log_root_tree->log_transid % 2;  	list_add_tail(&root_log_ctx.list, &log_root_tree->log_ctxs[index2]);  	root_log_ctx.log_transid = log_root_tree->log_transid; -	mutex_unlock(&log_root_tree->log_mutex); - -	mutex_lock(&log_root_tree->log_mutex); -  	/*  	 * Now we are safe to update the log_root_tree because we're under the  	 * log_mutex, and we're a current writer so we're holding the commit  	 * open until we drop the log_mutex.  	 */  	ret = update_log_root(trans, log, &new_root_item); - -	if (atomic_dec_and_test(&log_root_tree->log_writers)) { -		/* atomic_dec_and_test implies a barrier */ -		cond_wake_up_nomb(&log_root_tree->log_writer_wait); -	} -  	if (ret) {  		if (!list_empty(&root_log_ctx.list))  			list_del_init(&root_log_ctx.list); @@ -3184,8 +3172,6 @@ int btrfs_sync_log(struct btrfs_trans_handle *trans,  				root_log_ctx.log_transid - 1);  	} -	wait_for_writer(log_root_tree); -  	/*  	 * now that we've moved on to the tree of log tree roots,  	 * check the full commit flag again @@ -3906,6 +3892,7 @@ static int log_inode_item(struct btrfs_trans_handle *trans,  }  static int log_csums(struct btrfs_trans_handle *trans, +		     struct btrfs_inode *inode,  		     struct btrfs_root *log_root,  		     struct btrfs_ordered_sum *sums)  { @@ -3914,6 +3901,14 @@ static int log_csums(struct btrfs_trans_handle *trans,  	int ret;  	/* +	 * If this inode was not used for reflink operations in the current +	 * transaction with new extents, then do the fast path, no need to +	 * worry about logging checksum items with overlapping ranges. +	 */ +	if (inode->last_reflink_trans < trans->transid) +		return btrfs_csum_file_blocks(trans, log_root, sums); + +	/*  	 * Serialize logging for checksums. This is to avoid racing with the  	 * same checksum being logged by another task that is logging another  	 * file which happens to refer to the same extent as well. Such races @@ -4064,7 +4059,7 @@ static noinline int copy_items(struct btrfs_trans_handle *trans,  						   struct btrfs_ordered_sum,  						   list);  		if (!ret) -			ret = log_csums(trans, log, sums); +			ret = log_csums(trans, inode, log, sums);  		list_del(&sums->list);  		kfree(sums);  	} @@ -4123,7 +4118,7 @@ static int log_extent_csums(struct btrfs_trans_handle *trans,  						   struct btrfs_ordered_sum,  						   list);  		if (!ret) -			ret = log_csums(trans, log_root, sums); +			ret = log_csums(trans, inode, log_root, sums);  		list_del(&sums->list);  		kfree(sums);  	} @@ -4151,7 +4146,7 @@ static int log_one_extent(struct btrfs_trans_handle *trans,  	if (ret)  		return ret; -	ret = __btrfs_drop_extents(trans, log, &inode->vfs_inode, path, em->start, +	ret = __btrfs_drop_extents(trans, log, inode, path, em->start,  				   em->start + em->len, NULL, 0, 1,  				   sizeof(*fi), &extent_inserted);  	if (ret) @@ -5123,14 +5118,13 @@ static int btrfs_log_inode(struct btrfs_trans_handle *trans,  			   const loff_t end,  			   struct btrfs_log_ctx *ctx)  { -	struct btrfs_fs_info *fs_info = root->fs_info;  	struct btrfs_path *path;  	struct btrfs_path *dst_path;  	struct btrfs_key min_key;  	struct btrfs_key max_key;  	struct btrfs_root *log = root->log_root;  	int err = 0; -	int ret; +	int ret = 0;  	bool fast_search = false;  	u64 ino = btrfs_ino(inode);  	struct extent_map_tree *em_tree = &inode->extent_tree; @@ -5166,15 +5160,19 @@ static int btrfs_log_inode(struct btrfs_trans_handle *trans,  	max_key.offset = (u64)-1;  	/* -	 * Only run delayed items if we are a dir or a new file. -	 * Otherwise commit the delayed inode only, which is needed in -	 * order for the log replay code to mark inodes for link count -	 * fixup (create temporary BTRFS_TREE_LOG_FIXUP_OBJECTID items). +	 * Only run delayed items if we are a directory. We want to make sure +	 * all directory indexes hit the fs/subvolume tree so we can find them +	 * and figure out which index ranges have to be logged. +	 * +	 * Otherwise commit the delayed inode only if the full sync flag is set, +	 * as we want to make sure an up to date version is in the subvolume +	 * tree so copy_inode_items_to_log() / copy_items() can find it and copy +	 * it to the log tree. For a non full sync, we always log the inode item +	 * based on the in-memory struct btrfs_inode which is always up to date.  	 */ -	if (S_ISDIR(inode->vfs_inode.i_mode) || -	    inode->generation > fs_info->last_trans_committed) +	if (S_ISDIR(inode->vfs_inode.i_mode))  		ret = btrfs_commit_inode_delayed_items(trans, inode); -	else +	else if (test_bit(BTRFS_INODE_NEEDS_FULL_SYNC, &inode->runtime_flags))  		ret = btrfs_commit_inode_delayed_inode(inode);  	if (ret) { |