diff options
Diffstat (limited to 'fs/xfs/xfs_log_cil.c')
| -rw-r--r-- | fs/xfs/xfs_log_cil.c | 195 | 
1 files changed, 141 insertions, 54 deletions
| diff --git a/fs/xfs/xfs_log_cil.c b/fs/xfs/xfs_log_cil.c index e5ab62f08c19..70f718d76ceb 100644 --- a/fs/xfs/xfs_log_cil.c +++ b/fs/xfs/xfs_log_cil.c @@ -48,6 +48,38 @@ xlog_cil_ticket_alloc(  }  /* + * Check if the current log item was first committed in this sequence. + * We can't rely on just the log item being in the CIL, we have to check + * the recorded commit sequence number. + * + * Note: for this to be used in a non-racy manner, it has to be called with + * CIL flushing locked out. As a result, it should only be used during the + * transaction commit process when deciding what to format into the item. + */ +static bool +xlog_item_in_current_chkpt( +	struct xfs_cil		*cil, +	struct xfs_log_item	*lip) +{ +	if (list_empty(&lip->li_cil)) +		return false; + +	/* +	 * li_seq is written on the first commit of a log item to record the +	 * first checkpoint it is written to. Hence if it is different to the +	 * current sequence, we're in a new checkpoint. +	 */ +	return lip->li_seq == READ_ONCE(cil->xc_current_sequence); +} + +bool +xfs_log_item_in_current_chkpt( +	struct xfs_log_item *lip) +{ +	return xlog_item_in_current_chkpt(lip->li_log->l_cilp, lip); +} + +/*   * Unavoidable forward declaration - xlog_cil_push_work() calls   * xlog_cil_ctx_alloc() itself.   */ @@ -444,7 +476,8 @@ insert:  static void  xlog_cil_insert_items(  	struct xlog		*log, -	struct xfs_trans	*tp) +	struct xfs_trans	*tp, +	uint32_t		released_space)  {  	struct xfs_cil		*cil = log->l_cilp;  	struct xfs_cil_ctx	*ctx = cil->xc_ctx; @@ -493,7 +526,9 @@ xlog_cil_insert_items(  		ASSERT(tp->t_ticket->t_curr_res >= len);  	}  	tp->t_ticket->t_curr_res -= len; +	tp->t_ticket->t_curr_res += released_space;  	ctx->space_used += len; +	ctx->space_used -= released_space;  	/*  	 * If we've overrun the reservation, dump the tx details before we move @@ -935,6 +970,65 @@ xlog_cil_build_trans_hdr(  }  /* + * Pull all the log vectors off the items in the CIL, and remove the items from + * the CIL. We don't need the CIL lock here because it's only needed on the + * transaction commit side which is currently locked out by the flush lock. + * + * If a log item is marked with a whiteout, we do not need to write it to the + * journal and so we just move them to the whiteout list for the caller to + * dispose of appropriately. + */ +static void +xlog_cil_build_lv_chain( +	struct xfs_cil		*cil, +	struct xfs_cil_ctx	*ctx, +	struct list_head	*whiteouts, +	uint32_t		*num_iovecs, +	uint32_t		*num_bytes) +{ +	struct xfs_log_vec	*lv = NULL; + +	while (!list_empty(&cil->xc_cil)) { +		struct xfs_log_item	*item; + +		item = list_first_entry(&cil->xc_cil, +					struct xfs_log_item, li_cil); + +		if (test_bit(XFS_LI_WHITEOUT, &item->li_flags)) { +			list_move(&item->li_cil, whiteouts); +			trace_xfs_cil_whiteout_skip(item); +			continue; +		} + +		list_del_init(&item->li_cil); +		if (!ctx->lv_chain) +			ctx->lv_chain = item->li_lv; +		else +			lv->lv_next = item->li_lv; +		lv = item->li_lv; +		item->li_lv = NULL; +		*num_iovecs += lv->lv_niovecs; + +		/* we don't write ordered log vectors */ +		if (lv->lv_buf_len != XFS_LOG_VEC_ORDERED) +			*num_bytes += lv->lv_bytes; +	} +} + +static void +xlog_cil_cleanup_whiteouts( +	struct list_head	*whiteouts) +{ +	while (!list_empty(whiteouts)) { +		struct xfs_log_item *item = list_first_entry(whiteouts, +						struct xfs_log_item, li_cil); +		list_del_init(&item->li_cil); +		trace_xfs_cil_whiteout_unpin(item); +		item->li_ops->iop_unpin(item, 1); +	} +} + +/*   * Push the Committed Item List to the log.   *   * If the current sequence is the same as xc_push_seq we need to do a flush. If @@ -956,7 +1050,6 @@ xlog_cil_push_work(  		container_of(work, struct xfs_cil_ctx, push_work);  	struct xfs_cil		*cil = ctx->cil;  	struct xlog		*log = cil->xc_log; -	struct xfs_log_vec	*lv;  	struct xfs_cil_ctx	*new_ctx;  	int			num_iovecs = 0;  	int			num_bytes = 0; @@ -965,6 +1058,7 @@ xlog_cil_push_work(  	struct xfs_log_vec	lvhdr = { NULL };  	xfs_csn_t		push_seq;  	bool			push_commit_stable; +	LIST_HEAD		(whiteouts);  	new_ctx = xlog_cil_ctx_alloc();  	new_ctx->ticket = xlog_cil_ticket_alloc(log); @@ -1033,31 +1127,7 @@ xlog_cil_push_work(  	list_add(&ctx->committing, &cil->xc_committing);  	spin_unlock(&cil->xc_push_lock); -	/* -	 * Pull all the log vectors off the items in the CIL, and remove the -	 * items from the CIL. We don't need the CIL lock here because it's only -	 * needed on the transaction commit side which is currently locked out -	 * by the flush lock. -	 */ -	lv = NULL; -	while (!list_empty(&cil->xc_cil)) { -		struct xfs_log_item	*item; - -		item = list_first_entry(&cil->xc_cil, -					struct xfs_log_item, li_cil); -		list_del_init(&item->li_cil); -		if (!ctx->lv_chain) -			ctx->lv_chain = item->li_lv; -		else -			lv->lv_next = item->li_lv; -		lv = item->li_lv; -		item->li_lv = NULL; -		num_iovecs += lv->lv_niovecs; - -		/* we don't write ordered log vectors */ -		if (lv->lv_buf_len != XFS_LOG_VEC_ORDERED) -			num_bytes += lv->lv_bytes; -	} +	xlog_cil_build_lv_chain(cil, ctx, &whiteouts, &num_iovecs, &num_bytes);  	/*  	 * Switch the contexts so we can drop the context lock and move out @@ -1160,6 +1230,7 @@ xlog_cil_push_work(  	/* Not safe to reference ctx now! */  	spin_unlock(&log->l_icloglock); +	xlog_cil_cleanup_whiteouts(&whiteouts);  	return;  out_skip: @@ -1171,6 +1242,7 @@ out_skip:  out_abort_free_ticket:  	xfs_log_ticket_ungrant(log, ctx->ticket);  	ASSERT(xlog_is_shutdown(log)); +	xlog_cil_cleanup_whiteouts(&whiteouts);  	if (!ctx->commit_iclog) {  		xlog_cil_committed(ctx);  		return; @@ -1320,6 +1392,43 @@ xlog_cil_empty(  }  /* + * If there are intent done items in this transaction and the related intent was + * committed in the current (same) CIL checkpoint, we don't need to write either + * the intent or intent done item to the journal as the change will be + * journalled atomically within this checkpoint. As we cannot remove items from + * the CIL here, mark the related intent with a whiteout so that the CIL push + * can remove it rather than writing it to the journal. Then remove the intent + * done item from the current transaction and release it so it doesn't get put + * into the CIL at all. + */ +static uint32_t +xlog_cil_process_intents( +	struct xfs_cil		*cil, +	struct xfs_trans	*tp) +{ +	struct xfs_log_item	*lip, *ilip, *next; +	uint32_t		len = 0; + +	list_for_each_entry_safe(lip, next, &tp->t_items, li_trans) { +		if (!(lip->li_ops->flags & XFS_ITEM_INTENT_DONE)) +			continue; + +		ilip = lip->li_ops->iop_intent(lip); +		if (!ilip || !xlog_item_in_current_chkpt(cil, ilip)) +			continue; +		set_bit(XFS_LI_WHITEOUT, &ilip->li_flags); +		trace_xfs_cil_whiteout_mark(ilip); +		len += ilip->li_lv->lv_bytes; +		kmem_free(ilip->li_lv); +		ilip->li_lv = NULL; + +		xfs_trans_del_item(lip); +		lip->li_ops->iop_release(lip); +	} +	return len; +} + +/*   * Commit a transaction with the given vector to the Committed Item List.   *   * To do this, we need to format the item, pin it in memory if required and @@ -1341,6 +1450,7 @@ xlog_cil_commit(  {  	struct xfs_cil		*cil = log->l_cilp;  	struct xfs_log_item	*lip, *next; +	uint32_t		released_space = 0;  	/*  	 * Do all necessary memory allocation before we lock the CIL. @@ -1352,7 +1462,10 @@ xlog_cil_commit(  	/* lock out background commit */  	down_read(&cil->xc_ctx_lock); -	xlog_cil_insert_items(log, tp); +	if (tp->t_flags & XFS_TRANS_HAS_INTENT_DONE) +		released_space = xlog_cil_process_intents(cil, tp); + +	xlog_cil_insert_items(log, tp, released_space);  	if (regrant && !xlog_is_shutdown(log))  		xfs_log_ticket_regrant(log, tp->t_ticket); @@ -1509,32 +1622,6 @@ out_shutdown:  }  /* - * Check if the current log item was first committed in this sequence. - * We can't rely on just the log item being in the CIL, we have to check - * the recorded commit sequence number. - * - * Note: for this to be used in a non-racy manner, it has to be called with - * CIL flushing locked out. As a result, it should only be used during the - * transaction commit process when deciding what to format into the item. - */ -bool -xfs_log_item_in_current_chkpt( -	struct xfs_log_item	*lip) -{ -	struct xfs_cil		*cil = lip->li_log->l_cilp; - -	if (list_empty(&lip->li_cil)) -		return false; - -	/* -	 * li_seq is written on the first commit of a log item to record the -	 * first checkpoint it is written to. Hence if it is different to the -	 * current sequence, we're in a new checkpoint. -	 */ -	return lip->li_seq == READ_ONCE(cil->xc_current_sequence); -} - -/*   * Perform initial CIL structure initialisation.   */  int | 
