diff options
Diffstat (limited to 'fs/xfs')
| -rw-r--r-- | fs/xfs/scrub/agheader.c | 3 | ||||
| -rw-r--r-- | fs/xfs/xfs_iomap.c | 42 | ||||
| -rw-r--r-- | fs/xfs/xfs_refcount_item.c | 9 | ||||
| -rw-r--r-- | fs/xfs/xfs_rmap_item.c | 4 | ||||
| -rw-r--r-- | fs/xfs/xfs_super.c | 2 | 
5 files changed, 43 insertions, 17 deletions
| diff --git a/fs/xfs/scrub/agheader.c b/fs/xfs/scrub/agheader.c index fd975524f460..05c66e05ae20 100644 --- a/fs/xfs/scrub/agheader.c +++ b/fs/xfs/scrub/agheader.c @@ -767,7 +767,7 @@ int  xfs_scrub_agfl(  	struct xfs_scrub_context	*sc)  { -	struct xfs_scrub_agfl_info	sai = { 0 }; +	struct xfs_scrub_agfl_info	sai;  	struct xfs_agf			*agf;  	xfs_agnumber_t			agno;  	unsigned int			agflcount; @@ -795,6 +795,7 @@ xfs_scrub_agfl(  		xfs_scrub_block_set_corrupt(sc, sc->sa.agf_bp);  		goto out;  	} +	memset(&sai, 0, sizeof(sai));  	sai.sz_entries = agflcount;  	sai.entries = kmem_zalloc(sizeof(xfs_agblock_t) * agflcount, KM_NOFS);  	if (!sai.entries) { diff --git a/fs/xfs/xfs_iomap.c b/fs/xfs/xfs_iomap.c index 66e1edbfb2b2..046469fcc1b8 100644 --- a/fs/xfs/xfs_iomap.c +++ b/fs/xfs/xfs_iomap.c @@ -955,15 +955,29 @@ static inline bool imap_needs_alloc(struct inode *inode,  		(IS_DAX(inode) && imap->br_state == XFS_EXT_UNWRITTEN);  } +static inline bool needs_cow_for_zeroing(struct xfs_bmbt_irec *imap, int nimaps) +{ +	return nimaps && +		imap->br_startblock != HOLESTARTBLOCK && +		imap->br_state != XFS_EXT_UNWRITTEN; +} +  static inline bool need_excl_ilock(struct xfs_inode *ip, unsigned flags)  {  	/* -	 * COW writes will allocate delalloc space, so we need to make sure -	 * to take the lock exclusively here. +	 * COW writes may allocate delalloc space or convert unwritten COW +	 * extents, so we need to make sure to take the lock exclusively here.  	 */  	if (xfs_is_reflink_inode(ip) && (flags & (IOMAP_WRITE | IOMAP_ZERO)))  		return true; -	if ((flags & IOMAP_DIRECT) && (flags & IOMAP_WRITE)) + +	/* +	 * Extents not yet cached requires exclusive access, don't block. +	 * This is an opencoded xfs_ilock_data_map_shared() to cater for the +	 * non-blocking behaviour. +	 */ +	if (ip->i_d.di_format == XFS_DINODE_FMT_BTREE && +	    !(ip->i_df.if_flags & XFS_IFEXTENTS))  		return true;  	return false;  } @@ -993,16 +1007,18 @@ xfs_file_iomap_begin(  		return xfs_file_iomap_begin_delay(inode, offset, length, iomap);  	} -	if (need_excl_ilock(ip, flags)) { +	if (need_excl_ilock(ip, flags))  		lockmode = XFS_ILOCK_EXCL; -		xfs_ilock(ip, XFS_ILOCK_EXCL); -	} else { -		lockmode = xfs_ilock_data_map_shared(ip); -	} +	else +		lockmode = XFS_ILOCK_SHARED; -	if ((flags & IOMAP_NOWAIT) && !(ip->i_df.if_flags & XFS_IFEXTENTS)) { -		error = -EAGAIN; -		goto out_unlock; +	if (flags & IOMAP_NOWAIT) { +		if (!(ip->i_df.if_flags & XFS_IFEXTENTS)) +			return -EAGAIN; +		if (!xfs_ilock_nowait(ip, lockmode)) +			return -EAGAIN; +	} else { +		xfs_ilock(ip, lockmode);  	}  	ASSERT(offset <= mp->m_super->s_maxbytes); @@ -1024,7 +1040,9 @@ xfs_file_iomap_begin(  			goto out_unlock;  	} -	if ((flags & (IOMAP_WRITE | IOMAP_ZERO)) && xfs_is_reflink_inode(ip)) { +	if (xfs_is_reflink_inode(ip) && +	    ((flags & IOMAP_WRITE) || +	     ((flags & IOMAP_ZERO) && needs_cow_for_zeroing(&imap, nimaps)))) {  		if (flags & IOMAP_DIRECT) {  			/*  			 * A reflinked inode will result in CoW alloc. diff --git a/fs/xfs/xfs_refcount_item.c b/fs/xfs/xfs_refcount_item.c index 3a55d6fc271b..7a39f40645f7 100644 --- a/fs/xfs/xfs_refcount_item.c +++ b/fs/xfs/xfs_refcount_item.c @@ -23,6 +23,7 @@  #include "xfs_log_format.h"  #include "xfs_trans_resv.h"  #include "xfs_bit.h" +#include "xfs_shared.h"  #include "xfs_mount.h"  #include "xfs_defer.h"  #include "xfs_trans.h" @@ -456,10 +457,12 @@ xfs_cui_recover(  	 * transaction.  Normally, any work that needs to be deferred  	 * gets attached to the same defer_ops that scheduled the  	 * refcount update.  However, we're in log recovery here, so we -	 * we create our own defer_ops and use that to finish up any -	 * work that doesn't fit. +	 * we use the passed in defer_ops and to finish up any work that +	 * doesn't fit.  We need to reserve enough blocks to handle a +	 * full btree split on either end of the refcount range.  	 */ -	error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, 0, 0, 0, &tp); +	error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, +			mp->m_refc_maxlevels * 2, 0, XFS_TRANS_RESERVE, &tp);  	if (error)  		return error;  	cudp = xfs_trans_get_cud(tp, cuip); diff --git a/fs/xfs/xfs_rmap_item.c b/fs/xfs/xfs_rmap_item.c index f3b139c9aa16..49d3124863a8 100644 --- a/fs/xfs/xfs_rmap_item.c +++ b/fs/xfs/xfs_rmap_item.c @@ -23,6 +23,7 @@  #include "xfs_log_format.h"  #include "xfs_trans_resv.h"  #include "xfs_bit.h" +#include "xfs_shared.h"  #include "xfs_mount.h"  #include "xfs_defer.h"  #include "xfs_trans.h" @@ -470,7 +471,8 @@ xfs_rui_recover(  		}  	} -	error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, 0, 0, 0, &tp); +	error = xfs_trans_alloc(mp, &M_RES(mp)->tr_itruncate, +			mp->m_rmap_maxlevels, 0, XFS_TRANS_RESERVE, &tp);  	if (error)  		return error;  	rudp = xfs_trans_get_rud(tp, ruip); diff --git a/fs/xfs/xfs_super.c b/fs/xfs/xfs_super.c index 7aba628dc527..93588ea3d3d2 100644 --- a/fs/xfs/xfs_super.c +++ b/fs/xfs/xfs_super.c @@ -250,6 +250,7 @@ xfs_parseargs(  				return -EINVAL;  			break;  		case Opt_logdev: +			kfree(mp->m_logname);  			mp->m_logname = match_strdup(args);  			if (!mp->m_logname)  				return -ENOMEM; @@ -258,6 +259,7 @@ xfs_parseargs(  			xfs_warn(mp, "%s option not allowed on this system", p);  			return -EINVAL;  		case Opt_rtdev: +			kfree(mp->m_rtname);  			mp->m_rtname = match_strdup(args);  			if (!mp->m_rtname)  				return -ENOMEM; | 
