diff options
Diffstat (limited to 'fs/xfs/xfs_trans.c')
| -rw-r--r-- | fs/xfs/xfs_trans.c | 26 | 
1 files changed, 22 insertions, 4 deletions
| diff --git a/fs/xfs/xfs_trans.c b/fs/xfs/xfs_trans.c index 524f543c5b82..bedc5a5133a5 100644 --- a/fs/xfs/xfs_trans.c +++ b/fs/xfs/xfs_trans.c @@ -100,6 +100,8 @@ xfs_trans_dup(  	ntp->t_mountp = tp->t_mountp;  	INIT_LIST_HEAD(&ntp->t_items);  	INIT_LIST_HEAD(&ntp->t_busy); +	INIT_LIST_HEAD(&ntp->t_dfops); +	ntp->t_firstblock = NULLFSBLOCK;  	ASSERT(tp->t_flags & XFS_TRANS_PERM_LOG_RES);  	ASSERT(tp->t_ticket != NULL); @@ -118,7 +120,9 @@ xfs_trans_dup(  	ntp->t_rtx_res = tp->t_rtx_res - tp->t_rtx_res_used;  	tp->t_rtx_res = tp->t_rtx_res_used;  	ntp->t_pflags = tp->t_pflags; -	ntp->t_agfl_dfops = tp->t_agfl_dfops; + +	/* move deferred ops over to the new tp */ +	xfs_defer_move(ntp, tp);  	xfs_trans_dup_dqinfo(tp, ntp); @@ -273,6 +277,8 @@ xfs_trans_alloc(  	tp->t_mountp = mp;  	INIT_LIST_HEAD(&tp->t_items);  	INIT_LIST_HEAD(&tp->t_busy); +	INIT_LIST_HEAD(&tp->t_dfops); +	tp->t_firstblock = NULLFSBLOCK;  	error = xfs_trans_reserve(tp, resp, blocks, rtextents);  	if (error) { @@ -914,12 +920,21 @@ __xfs_trans_commit(  	int			error = 0;  	int			sync = tp->t_flags & XFS_TRANS_SYNC; -	ASSERT(!tp->t_agfl_dfops || -	       !xfs_defer_has_unfinished_work(tp->t_agfl_dfops) || regrant); -  	trace_xfs_trans_commit(tp, _RET_IP_);  	/* +	 * Finish deferred items on final commit. Only permanent transactions +	 * should ever have deferred ops. +	 */ +	WARN_ON_ONCE(!list_empty(&tp->t_dfops) && +		     !(tp->t_flags & XFS_TRANS_PERM_LOG_RES)); +	if (!regrant && (tp->t_flags & XFS_TRANS_PERM_LOG_RES)) { +		error = xfs_defer_finish_noroll(&tp); +		if (error) +			goto out_unreserve; +	} + +	/*  	 * If there is nothing to be logged by the transaction,  	 * then unlock all of the items associated with the  	 * transaction and free the transaction structure. @@ -1008,6 +1023,9 @@ xfs_trans_cancel(  	trace_xfs_trans_cancel(tp, _RET_IP_); +	if (tp->t_flags & XFS_TRANS_PERM_LOG_RES) +		xfs_defer_cancel(tp); +  	/*  	 * See if the caller is relying on us to shut down the  	 * filesystem.  This happens in paths where we detect | 
