xfs: push iclog state cleaning into xlog_state_clean_log
xlog_state_clean_log() is only called from one place, and it occurs when an iclog is transitioning back to ACTIVE. Prior to calling xlog_state_clean_log, the iclog we are processing has a hard coded state check to DIRTY so that xlog_state_clean_log() processes it correctly. We also have a hard coded wakeup after xlog_state_clean_log() to enfore log force waiters on that iclog are woken correctly. Both of these things are operations required to finish processing an iclog and return it to the ACTIVE state again, so they make little sense to be separated from the rest of the clean state transition code. Hence push these things inside xlog_state_clean_log(), document the behaviour and rename it xlog_state_clean_iclog() to indicate that it's being driven by an iclog state change and does the iclog state change work itself. Signed-off-by: Dave Chinner <dchinner@redhat.com> Reviewed-by: Christoph Hellwig <hch@lst.de> Reviewed-by: Darrick J. Wong <darrick.wong@oracle.com> Signed-off-by: Darrick J. Wong <darrick.wong@oracle.com>
This commit is contained in:
parent
5e96fa8d2b
commit
0383f543d8
|
@ -2523,21 +2523,35 @@ next_lv:
|
|||
*****************************************************************************
|
||||
*/
|
||||
|
||||
/* Clean iclogs starting from the head. This ordering must be
|
||||
* maintained, so an iclog doesn't become ACTIVE beyond one that
|
||||
* is SYNCING. This is also required to maintain the notion that we use
|
||||
* a ordered wait queue to hold off would be writers to the log when every
|
||||
* iclog is trying to sync to disk.
|
||||
/*
|
||||
* An iclog has just finished IO completion processing, so we need to update
|
||||
* the iclog state and propagate that up into the overall log state. Hence we
|
||||
* prepare the iclog for cleaning, and then clean all the pending dirty iclogs
|
||||
* starting from the head, and then wake up any threads that are waiting for the
|
||||
* iclog to be marked clean.
|
||||
*
|
||||
* State Change: DIRTY -> ACTIVE
|
||||
* The ordering of marking iclogs ACTIVE must be maintained, so an iclog
|
||||
* doesn't become ACTIVE beyond one that is SYNCING. This is also required to
|
||||
* maintain the notion that we use a ordered wait queue to hold off would be
|
||||
* writers to the log when every iclog is trying to sync to disk.
|
||||
*
|
||||
* Caller must hold the icloglock before calling us.
|
||||
*
|
||||
* State Change: !IOERROR -> DIRTY -> ACTIVE
|
||||
*/
|
||||
STATIC void
|
||||
xlog_state_clean_log(
|
||||
struct xlog *log)
|
||||
xlog_state_clean_iclog(
|
||||
struct xlog *log,
|
||||
struct xlog_in_core *dirty_iclog)
|
||||
{
|
||||
xlog_in_core_t *iclog;
|
||||
struct xlog_in_core *iclog;
|
||||
int changed = 0;
|
||||
|
||||
/* Prepare the completed iclog. */
|
||||
if (!(dirty_iclog->ic_state & XLOG_STATE_IOERROR))
|
||||
dirty_iclog->ic_state = XLOG_STATE_DIRTY;
|
||||
|
||||
/* Walk all the iclogs to update the ordered active state. */
|
||||
iclog = log->l_iclog;
|
||||
do {
|
||||
if (iclog->ic_state == XLOG_STATE_DIRTY) {
|
||||
|
@ -2575,7 +2589,13 @@ xlog_state_clean_log(
|
|||
iclog = iclog->ic_next;
|
||||
} while (iclog != log->l_iclog);
|
||||
|
||||
/* log is locked when we are called */
|
||||
|
||||
/*
|
||||
* Wake up threads waiting in xfs_log_force() for the dirty iclog
|
||||
* to be cleaned.
|
||||
*/
|
||||
wake_up_all(&dirty_iclog->ic_force_wait);
|
||||
|
||||
/*
|
||||
* Change state for the dummy log recording.
|
||||
* We usually go to NEED. But we go to NEED2 if the changed indicates
|
||||
|
@ -2609,7 +2629,7 @@ xlog_state_clean_log(
|
|||
ASSERT(0);
|
||||
}
|
||||
}
|
||||
} /* xlog_state_clean_log */
|
||||
}
|
||||
|
||||
STATIC xfs_lsn_t
|
||||
xlog_get_lowest_lsn(
|
||||
|
@ -2840,18 +2860,7 @@ xlog_state_do_callback(
|
|||
cycled_icloglock = true;
|
||||
xlog_state_do_iclog_callbacks(log, iclog, aborted);
|
||||
|
||||
if (!(iclog->ic_state & XLOG_STATE_IOERROR))
|
||||
iclog->ic_state = XLOG_STATE_DIRTY;
|
||||
|
||||
/*
|
||||
* Transition from DIRTY to ACTIVE if applicable.
|
||||
* NOP if STATE_IOERROR.
|
||||
*/
|
||||
xlog_state_clean_log(log);
|
||||
|
||||
/* wake up threads waiting in xfs_log_force() */
|
||||
wake_up_all(&iclog->ic_force_wait);
|
||||
|
||||
xlog_state_clean_iclog(log, iclog);
|
||||
iclog = iclog->ic_next;
|
||||
} while (first_iclog != iclog);
|
||||
|
||||
|
|
Loading…
Reference in New Issue