diff options
Diffstat (limited to 'fs/btrfs/inode.c')
| -rw-r--r-- | fs/btrfs/inode.c | 25 | 
1 files changed, 18 insertions, 7 deletions
| diff --git a/fs/btrfs/inode.c b/fs/btrfs/inode.c index 0a9b10c5b0a..3601f0aebdd 100644 --- a/fs/btrfs/inode.c +++ b/fs/btrfs/inode.c @@ -2509,6 +2509,11 @@ static void btrfs_read_locked_inode(struct inode *inode)  	int maybe_acls;  	u32 rdev;  	int ret; +	bool filled = false; + +	ret = btrfs_fill_inode(inode, &rdev); +	if (!ret) +		filled = true;  	path = btrfs_alloc_path();  	BUG_ON(!path); @@ -2520,6 +2525,10 @@ static void btrfs_read_locked_inode(struct inode *inode)  		goto make_bad;  	leaf = path->nodes[0]; + +	if (filled) +		goto cache_acl; +  	inode_item = btrfs_item_ptr(leaf, path->slots[0],  				    struct btrfs_inode_item);  	if (!leaf->map_token) @@ -2556,7 +2565,7 @@ static void btrfs_read_locked_inode(struct inode *inode)  	BTRFS_I(inode)->index_cnt = (u64)-1;  	BTRFS_I(inode)->flags = btrfs_inode_flags(leaf, inode_item); - +cache_acl:  	/*  	 * try to precache a NULL acl entry for files that don't have  	 * any xattrs or acls @@ -2572,7 +2581,6 @@ static void btrfs_read_locked_inode(struct inode *inode)  	}  	btrfs_free_path(path); -	inode_item = NULL;  	switch (inode->i_mode & S_IFMT) {  	case S_IFREG: @@ -2670,12 +2678,14 @@ noinline int btrfs_update_inode(struct btrfs_trans_handle *trans,  	int ret;  	/* -	 * If root is tree root, it means this inode is used to -	 * store free space information. And these inodes are updated -	 * when committing the transaction, so they needn't delaye to -	 * be updated, or deadlock will occured. +	 * If the inode is a free space inode, we can deadlock during commit +	 * if we put it into the delayed code. +	 * +	 * The data relocation inode should also be directly updated +	 * without delay  	 */ -	if (!is_free_space_inode(root, inode)) { +	if (!is_free_space_inode(root, inode) +	    && root->root_key.objectid != BTRFS_DATA_RELOC_TREE_OBJECTID) {  		ret = btrfs_delayed_update_inode(trans, root, inode);  		if (!ret)  			btrfs_set_inode_last_trans(trans, inode); @@ -4520,6 +4530,7 @@ static struct inode *btrfs_new_inode(struct btrfs_trans_handle *trans,  	inode_tree_add(inode);  	trace_btrfs_inode_new(inode); +	btrfs_set_inode_last_trans(trans, inode);  	return inode;  fail: | 
