diff options
Diffstat (limited to 'fs/xfs/xfs_vnodeops.c')
| -rw-r--r-- | fs/xfs/xfs_vnodeops.c | 87 | 
1 files changed, 29 insertions, 58 deletions
| diff --git a/fs/xfs/xfs_vnodeops.c b/fs/xfs/xfs_vnodeops.c index b572f7e840e..6558ffd8d14 100644 --- a/fs/xfs/xfs_vnodeops.c +++ b/fs/xfs/xfs_vnodeops.c @@ -53,6 +53,7 @@  #include "xfs_log_priv.h"  #include "xfs_filestream.h"  #include "xfs_vnodeops.h" +#include "xfs_trace.h"  int  xfs_setattr( @@ -538,9 +539,8 @@ xfs_readlink_bmap(  		d = XFS_FSB_TO_DADDR(mp, mval[n].br_startblock);  		byte_cnt = XFS_FSB_TO_B(mp, mval[n].br_blockcount); -		bp = xfs_buf_read_flags(mp->m_ddev_targp, d, BTOBB(byte_cnt), -					XBF_LOCK | XBF_MAPPED | -					XBF_DONT_BLOCK); +		bp = xfs_buf_read(mp->m_ddev_targp, d, BTOBB(byte_cnt), +				  XBF_LOCK | XBF_MAPPED | XBF_DONT_BLOCK);  		error = XFS_BUF_GETERROR(bp);  		if (error) {  			xfs_ioerror_alert("xfs_readlink", @@ -709,6 +709,11 @@ xfs_fsync(  }  /* + * Flags for xfs_free_eofblocks + */ +#define XFS_FREE_EOF_TRYLOCK	(1<<0) + +/*   * This is called by xfs_inactive to free any blocks beyond eof   * when the link count isn't zero and by xfs_dm_punch_hole() when   * punching a hole to EOF. @@ -726,7 +731,6 @@ xfs_free_eofblocks(  	xfs_filblks_t	map_len;  	int		nimaps;  	xfs_bmbt_irec_t	imap; -	int		use_iolock = (flags & XFS_FREE_EOF_LOCK);  	/*  	 * Figure out if there are any blocks beyond the end @@ -768,14 +772,19 @@ xfs_free_eofblocks(  		 * cache and we can't  		 * do that within a transaction.  		 */ -		if (use_iolock) +		if (flags & XFS_FREE_EOF_TRYLOCK) { +			if (!xfs_ilock_nowait(ip, XFS_IOLOCK_EXCL)) { +				xfs_trans_cancel(tp, 0); +				return 0; +			} +		} else {  			xfs_ilock(ip, XFS_IOLOCK_EXCL); +		}  		error = xfs_itruncate_start(ip, XFS_ITRUNC_DEFINITE,  				    ip->i_size);  		if (error) {  			xfs_trans_cancel(tp, 0); -			if (use_iolock) -				xfs_iunlock(ip, XFS_IOLOCK_EXCL); +			xfs_iunlock(ip, XFS_IOLOCK_EXCL);  			return error;  		} @@ -812,8 +821,7 @@ xfs_free_eofblocks(  			error = xfs_trans_commit(tp,  						XFS_TRANS_RELEASE_LOG_RES);  		} -		xfs_iunlock(ip, (use_iolock ? (XFS_IOLOCK_EXCL|XFS_ILOCK_EXCL) -					    : XFS_ILOCK_EXCL)); +		xfs_iunlock(ip, XFS_IOLOCK_EXCL|XFS_ILOCK_EXCL);  	}  	return error;  } @@ -1113,7 +1121,17 @@ xfs_release(  		     (ip->i_df.if_flags & XFS_IFEXTENTS))  &&  		    (!(ip->i_d.di_flags &  				(XFS_DIFLAG_PREALLOC | XFS_DIFLAG_APPEND)))) { -			error = xfs_free_eofblocks(mp, ip, XFS_FREE_EOF_LOCK); + +			/* +			 * If we can't get the iolock just skip truncating +			 * the blocks past EOF because we could deadlock +			 * with the mmap_sem otherwise.  We'll get another +			 * chance to drop them once the last reference to +			 * the inode is dropped, so we'll never leak blocks +			 * permanently. +			 */ +			error = xfs_free_eofblocks(mp, ip, +						   XFS_FREE_EOF_TRYLOCK);  			if (error)  				return error;  		} @@ -1184,7 +1202,7 @@ xfs_inactive(  		     (!(ip->i_d.di_flags &  				(XFS_DIFLAG_PREALLOC | XFS_DIFLAG_APPEND)) ||  		      (ip->i_delayed_blks != 0)))) { -			error = xfs_free_eofblocks(mp, ip, XFS_FREE_EOF_LOCK); +			error = xfs_free_eofblocks(mp, ip, 0);  			if (error)  				return VN_INACTIVE_CACHE;  		} @@ -1380,7 +1398,6 @@ xfs_lookup(  	if (error)  		goto out_free_name; -	xfs_itrace_ref(*ipp);  	return 0;  out_free_name: @@ -1526,7 +1543,6 @@ xfs_create(  	 * At this point, we've gotten a newly allocated inode.  	 * It is locked (and joined to the transaction).  	 */ -	xfs_itrace_ref(ip);  	ASSERT(xfs_isilocked(ip, XFS_ILOCK_EXCL));  	/* @@ -1986,9 +2002,6 @@ xfs_remove(  	if (!is_dir && link_zero && xfs_inode_is_filestream(ip))  		xfs_filestream_deassociate(ip); -	xfs_itrace_exit(ip); -	xfs_itrace_exit(dp); -   std_return:  	if (DM_EVENT_ENABLED(dp, DM_EVENT_POSTREMOVE)) {  		XFS_SEND_NAMESP(mp, DM_EVENT_POSTREMOVE, dp, DM_RIGHT_NULL, @@ -2285,7 +2298,6 @@ xfs_symlink(  			goto error_return;  		goto error1;  	} -	xfs_itrace_ref(ip);  	/*  	 * An error after we've joined dp to the transaction will result in the @@ -2456,46 +2468,6 @@ xfs_set_dmattrs(  	return error;  } -int -xfs_reclaim( -	xfs_inode_t	*ip) -{ - -	xfs_itrace_entry(ip); - -	ASSERT(!VN_MAPPED(VFS_I(ip))); - -	/* bad inode, get out here ASAP */ -	if (is_bad_inode(VFS_I(ip))) { -		xfs_ireclaim(ip); -		return 0; -	} - -	xfs_ioend_wait(ip); - -	ASSERT(XFS_FORCED_SHUTDOWN(ip->i_mount) || ip->i_delayed_blks == 0); - -	/* -	 * If we have nothing to flush with this inode then complete the -	 * teardown now, otherwise break the link between the xfs inode and the -	 * linux inode and clean up the xfs inode later. This avoids flushing -	 * the inode to disk during the delete operation itself. -	 * -	 * When breaking the link, we need to set the XFS_IRECLAIMABLE flag -	 * first to ensure that xfs_iunpin() will never see an xfs inode -	 * that has a linux inode being reclaimed. Synchronisation is provided -	 * by the i_flags_lock. -	 */ -	if (!ip->i_update_core && (ip->i_itemp == NULL)) { -		xfs_ilock(ip, XFS_ILOCK_EXCL); -		xfs_iflock(ip); -		xfs_iflags_set(ip, XFS_IRECLAIMABLE); -		return xfs_reclaim_inode(ip, 1, XFS_IFLUSH_DELWRI_ELSE_SYNC); -	} -	xfs_inode_set_reclaim_tag(ip); -	return 0; -} -  /*   * xfs_alloc_file_space()   *      This routine allocates disk space for the given file. @@ -2868,7 +2840,6 @@ xfs_free_file_space(  	ioffset = offset & ~(rounding - 1);  	if (VN_CACHED(VFS_I(ip)) != 0) { -		xfs_inval_cached_trace(ip, ioffset, -1, ioffset, -1);  		error = xfs_flushinval_pages(ip, ioffset, -1, FI_REMAPF_LOCKED);  		if (error)  			goto out_unlock_iolock; | 
