提交 e1d06e5f 编写于 作者: D Dave Chinner 提交者: Darrick J. Wong

xfs: convert log flags to an operational state field

log->l_flags doesn't actually contain "flags" as such, it contains
operational state information that can change at runtime. For the
shutdown state, this at least should be an atomic bit because
it is read without holding locks in many places and so using atomic
bitops for the state field modifications makes sense.

This allows us to use things like test_and_set_bit() on state
changes (e.g. setting XLOG_TAIL_WARN) to avoid races in setting the
state when we aren't holding locks.
Signed-off-by: NDave Chinner <dchinner@redhat.com>
Reviewed-by: NChristoph Hellwig <hch@lst.de>
Reviewed-by: NDarrick J. Wong <djwong@kernel.org>
Signed-off-by: NDarrick J. Wong <djwong@kernel.org>
上级 fd67d8a0
...@@ -298,7 +298,7 @@ xlog_grant_head_check( ...@@ -298,7 +298,7 @@ xlog_grant_head_check(
int free_bytes; int free_bytes;
int error = 0; int error = 0;
ASSERT(!(log->l_flags & XLOG_ACTIVE_RECOVERY)); ASSERT(!xlog_in_recovery(log));
/* /*
* If there are other waiters on the queue then give them a chance at * If there are other waiters on the queue then give them a chance at
...@@ -580,6 +580,7 @@ xfs_log_mount( ...@@ -580,6 +580,7 @@ xfs_log_mount(
xfs_daddr_t blk_offset, xfs_daddr_t blk_offset,
int num_bblks) int num_bblks)
{ {
struct xlog *log;
bool fatal = xfs_sb_version_hascrc(&mp->m_sb); bool fatal = xfs_sb_version_hascrc(&mp->m_sb);
int error = 0; int error = 0;
int min_logfsbs; int min_logfsbs;
...@@ -594,11 +595,12 @@ xfs_log_mount( ...@@ -594,11 +595,12 @@ xfs_log_mount(
ASSERT(mp->m_flags & XFS_MOUNT_RDONLY); ASSERT(mp->m_flags & XFS_MOUNT_RDONLY);
} }
mp->m_log = xlog_alloc_log(mp, log_target, blk_offset, num_bblks); log = xlog_alloc_log(mp, log_target, blk_offset, num_bblks);
if (IS_ERR(mp->m_log)) { if (IS_ERR(log)) {
error = PTR_ERR(mp->m_log); error = PTR_ERR(log);
goto out; goto out;
} }
mp->m_log = log;
/* /*
* Validate the given log space and drop a critical message via syslog * Validate the given log space and drop a critical message via syslog
...@@ -663,7 +665,7 @@ xfs_log_mount( ...@@ -663,7 +665,7 @@ xfs_log_mount(
xfs_warn(mp, "AIL initialisation failed: error %d", error); xfs_warn(mp, "AIL initialisation failed: error %d", error);
goto out_free_log; goto out_free_log;
} }
mp->m_log->l_ailp = mp->m_ail; log->l_ailp = mp->m_ail;
/* /*
* skip log recovery on a norecovery mount. pretend it all * skip log recovery on a norecovery mount. pretend it all
...@@ -675,39 +677,39 @@ xfs_log_mount( ...@@ -675,39 +677,39 @@ xfs_log_mount(
if (readonly) if (readonly)
mp->m_flags &= ~XFS_MOUNT_RDONLY; mp->m_flags &= ~XFS_MOUNT_RDONLY;
error = xlog_recover(mp->m_log); error = xlog_recover(log);
if (readonly) if (readonly)
mp->m_flags |= XFS_MOUNT_RDONLY; mp->m_flags |= XFS_MOUNT_RDONLY;
if (error) { if (error) {
xfs_warn(mp, "log mount/recovery failed: error %d", xfs_warn(mp, "log mount/recovery failed: error %d",
error); error);
xlog_recover_cancel(mp->m_log); xlog_recover_cancel(log);
goto out_destroy_ail; goto out_destroy_ail;
} }
} }
error = xfs_sysfs_init(&mp->m_log->l_kobj, &xfs_log_ktype, &mp->m_kobj, error = xfs_sysfs_init(&log->l_kobj, &xfs_log_ktype, &mp->m_kobj,
"log"); "log");
if (error) if (error)
goto out_destroy_ail; goto out_destroy_ail;
/* Normal transactions can now occur */ /* Normal transactions can now occur */
mp->m_log->l_flags &= ~XLOG_ACTIVE_RECOVERY; clear_bit(XLOG_ACTIVE_RECOVERY, &log->l_opstate);
/* /*
* Now the log has been fully initialised and we know were our * Now the log has been fully initialised and we know were our
* space grant counters are, we can initialise the permanent ticket * space grant counters are, we can initialise the permanent ticket
* needed for delayed logging to work. * needed for delayed logging to work.
*/ */
xlog_cil_init_post_recovery(mp->m_log); xlog_cil_init_post_recovery(log);
return 0; return 0;
out_destroy_ail: out_destroy_ail:
xfs_trans_ail_destroy(mp); xfs_trans_ail_destroy(mp);
out_free_log: out_free_log:
xlog_dealloc_log(mp->m_log); xlog_dealloc_log(log);
out: out:
return error; return error;
} }
...@@ -759,7 +761,7 @@ xfs_log_mount_finish( ...@@ -759,7 +761,7 @@ xfs_log_mount_finish(
* mount failure occurs. * mount failure occurs.
*/ */
mp->m_super->s_flags |= SB_ACTIVE; mp->m_super->s_flags |= SB_ACTIVE;
if (log->l_flags & XLOG_RECOVERY_NEEDED) if (xlog_recovery_needed(log))
error = xlog_recover_finish(log); error = xlog_recover_finish(log);
if (!error) if (!error)
xfs_log_work_queue(mp); xfs_log_work_queue(mp);
...@@ -775,7 +777,7 @@ xfs_log_mount_finish( ...@@ -775,7 +777,7 @@ xfs_log_mount_finish(
* Don't push in the error case because the AIL may have pending intents * Don't push in the error case because the AIL may have pending intents
* that aren't removed until recovery is cancelled. * that aren't removed until recovery is cancelled.
*/ */
if (log->l_flags & XLOG_RECOVERY_NEEDED) { if (xlog_recovery_needed(log)) {
if (!error) { if (!error) {
xfs_log_force(mp, XFS_LOG_SYNC); xfs_log_force(mp, XFS_LOG_SYNC);
xfs_ail_push_all_sync(mp->m_ail); xfs_ail_push_all_sync(mp->m_ail);
...@@ -787,7 +789,7 @@ xfs_log_mount_finish( ...@@ -787,7 +789,7 @@ xfs_log_mount_finish(
} }
xfs_buftarg_drain(mp->m_ddev_targp); xfs_buftarg_drain(mp->m_ddev_targp);
log->l_flags &= ~XLOG_RECOVERY_NEEDED; clear_bit(XLOG_RECOVERY_NEEDED, &log->l_opstate);
if (readonly) if (readonly)
mp->m_flags |= XFS_MOUNT_RDONLY; mp->m_flags |= XFS_MOUNT_RDONLY;
...@@ -1075,7 +1077,7 @@ xfs_log_space_wake( ...@@ -1075,7 +1077,7 @@ xfs_log_space_wake(
return; return;
if (!list_empty_careful(&log->l_write_head.waiters)) { if (!list_empty_careful(&log->l_write_head.waiters)) {
ASSERT(!(log->l_flags & XLOG_ACTIVE_RECOVERY)); ASSERT(!xlog_in_recovery(log));
spin_lock(&log->l_write_head.lock); spin_lock(&log->l_write_head.lock);
free_bytes = xlog_space_left(log, &log->l_write_head.grant); free_bytes = xlog_space_left(log, &log->l_write_head.grant);
...@@ -1084,7 +1086,7 @@ xfs_log_space_wake( ...@@ -1084,7 +1086,7 @@ xfs_log_space_wake(
} }
if (!list_empty_careful(&log->l_reserve_head.waiters)) { if (!list_empty_careful(&log->l_reserve_head.waiters)) {
ASSERT(!(log->l_flags & XLOG_ACTIVE_RECOVERY)); ASSERT(!xlog_in_recovery(log));
spin_lock(&log->l_reserve_head.lock); spin_lock(&log->l_reserve_head.lock);
free_bytes = xlog_space_left(log, &log->l_reserve_head.grant); free_bytes = xlog_space_left(log, &log->l_reserve_head.grant);
...@@ -1466,7 +1468,7 @@ xlog_alloc_log( ...@@ -1466,7 +1468,7 @@ xlog_alloc_log(
log->l_logBBstart = blk_offset; log->l_logBBstart = blk_offset;
log->l_logBBsize = num_bblks; log->l_logBBsize = num_bblks;
log->l_covered_state = XLOG_STATE_COVER_IDLE; log->l_covered_state = XLOG_STATE_COVER_IDLE;
log->l_flags |= XLOG_ACTIVE_RECOVERY; set_bit(XLOG_ACTIVE_RECOVERY, &log->l_opstate);
INIT_DELAYED_WORK(&log->l_work, xfs_log_worker); INIT_DELAYED_WORK(&log->l_work, xfs_log_worker);
log->l_prev_block = -1; log->l_prev_block = -1;
...@@ -3648,17 +3650,15 @@ xlog_verify_grant_tail( ...@@ -3648,17 +3650,15 @@ xlog_verify_grant_tail(
xlog_crack_atomic_lsn(&log->l_tail_lsn, &tail_cycle, &tail_blocks); xlog_crack_atomic_lsn(&log->l_tail_lsn, &tail_cycle, &tail_blocks);
if (tail_cycle != cycle) { if (tail_cycle != cycle) {
if (cycle - 1 != tail_cycle && if (cycle - 1 != tail_cycle &&
!(log->l_flags & XLOG_TAIL_WARN)) { !test_and_set_bit(XLOG_TAIL_WARN, &log->l_opstate)) {
xfs_alert_tag(log->l_mp, XFS_PTAG_LOGRES, xfs_alert_tag(log->l_mp, XFS_PTAG_LOGRES,
"%s: cycle - 1 != tail_cycle", __func__); "%s: cycle - 1 != tail_cycle", __func__);
log->l_flags |= XLOG_TAIL_WARN;
} }
if (space > BBTOB(tail_blocks) && if (space > BBTOB(tail_blocks) &&
!(log->l_flags & XLOG_TAIL_WARN)) { !test_and_set_bit(XLOG_TAIL_WARN, &log->l_opstate)) {
xfs_alert_tag(log->l_mp, XFS_PTAG_LOGRES, xfs_alert_tag(log->l_mp, XFS_PTAG_LOGRES,
"%s: space > BBTOB(tail_blocks)", __func__); "%s: space > BBTOB(tail_blocks)", __func__);
log->l_flags |= XLOG_TAIL_WARN;
} }
} }
} }
...@@ -3825,8 +3825,7 @@ xfs_log_force_umount( ...@@ -3825,8 +3825,7 @@ xfs_log_force_umount(
* If this happens during log recovery, don't worry about * If this happens during log recovery, don't worry about
* locking; the log isn't open for business yet. * locking; the log isn't open for business yet.
*/ */
if (!log || if (!log || xlog_in_recovery(log)) {
log->l_flags & XLOG_ACTIVE_RECOVERY) {
mp->m_flags |= XFS_MOUNT_FS_SHUTDOWN; mp->m_flags |= XFS_MOUNT_FS_SHUTDOWN;
if (mp->m_sb_bp) if (mp->m_sb_bp)
mp->m_sb_bp->b_flags |= XBF_DONE; mp->m_sb_bp->b_flags |= XBF_DONE;
...@@ -3863,10 +3862,8 @@ xfs_log_force_umount( ...@@ -3863,10 +3862,8 @@ xfs_log_force_umount(
* Mark the log and the iclogs with IO error flags to prevent any * Mark the log and the iclogs with IO error flags to prevent any
* further log IO from being issued or completed. * further log IO from being issued or completed.
*/ */
if (!(log->l_flags & XLOG_IO_ERROR)) { if (!test_and_set_bit(XLOG_IO_ERROR, &log->l_opstate))
log->l_flags |= XLOG_IO_ERROR;
retval = 1; retval = 1;
}
spin_unlock(&log->l_icloglock); spin_unlock(&log->l_icloglock);
/* /*
...@@ -3954,15 +3951,6 @@ xfs_log_check_lsn( ...@@ -3954,15 +3951,6 @@ xfs_log_check_lsn(
return valid; return valid;
} }
bool
xfs_log_in_recovery(
struct xfs_mount *mp)
{
struct xlog *log = mp->m_log;
return log->l_flags & XLOG_ACTIVE_RECOVERY;
}
/* /*
* Notify the log that we're about to start using a feature that is protected * Notify the log that we're about to start using a feature that is protected
* by a log incompat feature flag. This will prevent log covering from * by a log incompat feature flag. This will prevent log covering from
......
...@@ -138,7 +138,6 @@ void xfs_log_work_queue(struct xfs_mount *mp); ...@@ -138,7 +138,6 @@ void xfs_log_work_queue(struct xfs_mount *mp);
int xfs_log_quiesce(struct xfs_mount *mp); int xfs_log_quiesce(struct xfs_mount *mp);
void xfs_log_clean(struct xfs_mount *mp); void xfs_log_clean(struct xfs_mount *mp);
bool xfs_log_check_lsn(struct xfs_mount *, xfs_lsn_t); bool xfs_log_check_lsn(struct xfs_mount *, xfs_lsn_t);
bool xfs_log_in_recovery(struct xfs_mount *);
xfs_lsn_t xlog_grant_push_threshold(struct xlog *log, int need_bytes); xfs_lsn_t xlog_grant_push_threshold(struct xlog *log, int need_bytes);
......
...@@ -11,15 +11,6 @@ struct xlog; ...@@ -11,15 +11,6 @@ struct xlog;
struct xlog_ticket; struct xlog_ticket;
struct xfs_mount; struct xfs_mount;
/*
* Flags for log structure
*/
#define XLOG_ACTIVE_RECOVERY 0x2 /* in the middle of recovery */
#define XLOG_RECOVERY_NEEDED 0x4 /* log was recovered */
#define XLOG_IO_ERROR 0x8 /* log hit an I/O error, and being
shutdown */
#define XLOG_TAIL_WARN 0x10 /* log tail verify warning issued */
/* /*
* get client id from packed copy. * get client id from packed copy.
* *
...@@ -405,7 +396,7 @@ struct xlog { ...@@ -405,7 +396,7 @@ struct xlog {
struct xfs_buftarg *l_targ; /* buftarg of log */ struct xfs_buftarg *l_targ; /* buftarg of log */
struct workqueue_struct *l_ioend_workqueue; /* for I/O completions */ struct workqueue_struct *l_ioend_workqueue; /* for I/O completions */
struct delayed_work l_work; /* background flush work */ struct delayed_work l_work; /* background flush work */
uint l_flags; long l_opstate; /* operational state */
uint l_quotaoffs_flag; /* XFS_DQ_*, for QUOTAOFFs */ uint l_quotaoffs_flag; /* XFS_DQ_*, for QUOTAOFFs */
struct list_head *l_buf_cancel_table; struct list_head *l_buf_cancel_table;
int l_iclog_hsize; /* size of iclog header */ int l_iclog_hsize; /* size of iclog header */
...@@ -462,10 +453,31 @@ struct xlog { ...@@ -462,10 +453,31 @@ struct xlog {
#define XLOG_BUF_CANCEL_BUCKET(log, blkno) \ #define XLOG_BUF_CANCEL_BUCKET(log, blkno) \
((log)->l_buf_cancel_table + ((uint64_t)blkno % XLOG_BC_TABLE_SIZE)) ((log)->l_buf_cancel_table + ((uint64_t)blkno % XLOG_BC_TABLE_SIZE))
/*
* Bits for operational state
*/
#define XLOG_ACTIVE_RECOVERY 0 /* in the middle of recovery */
#define XLOG_RECOVERY_NEEDED 1 /* log was recovered */
#define XLOG_IO_ERROR 2 /* log hit an I/O error, and being
shutdown */
#define XLOG_TAIL_WARN 3 /* log tail verify warning issued */
static inline bool
xlog_recovery_needed(struct xlog *log)
{
return test_bit(XLOG_RECOVERY_NEEDED, &log->l_opstate);
}
static inline bool
xlog_in_recovery(struct xlog *log)
{
return test_bit(XLOG_ACTIVE_RECOVERY, &log->l_opstate);
}
static inline bool static inline bool
xlog_is_shutdown(struct xlog *log) xlog_is_shutdown(struct xlog *log)
{ {
return (log->l_flags & XLOG_IO_ERROR); return test_bit(XLOG_IO_ERROR, &log->l_opstate);
} }
/* common routines */ /* common routines */
......
...@@ -3359,7 +3359,7 @@ xlog_do_recover( ...@@ -3359,7 +3359,7 @@ xlog_do_recover(
xlog_recover_check_summary(log); xlog_recover_check_summary(log);
/* Normal transactions can now occur */ /* Normal transactions can now occur */
log->l_flags &= ~XLOG_ACTIVE_RECOVERY; clear_bit(XLOG_ACTIVE_RECOVERY, &log->l_opstate);
return 0; return 0;
} }
...@@ -3443,7 +3443,7 @@ xlog_recover( ...@@ -3443,7 +3443,7 @@ xlog_recover(
: "internal"); : "internal");
error = xlog_do_recover(log, head_blk, tail_blk); error = xlog_do_recover(log, head_blk, tail_blk);
log->l_flags |= XLOG_RECOVERY_NEEDED; set_bit(XLOG_RECOVERY_NEEDED, &log->l_opstate);
} }
return error; return error;
} }
...@@ -3508,7 +3508,7 @@ void ...@@ -3508,7 +3508,7 @@ void
xlog_recover_cancel( xlog_recover_cancel(
struct xlog *log) struct xlog *log)
{ {
if (log->l_flags & XLOG_RECOVERY_NEEDED) if (xlog_recovery_needed(log))
xlog_recover_cancel_intents(log); xlog_recover_cancel_intents(log);
} }
......
...@@ -710,7 +710,7 @@ xfs_fs_drop_inode( ...@@ -710,7 +710,7 @@ xfs_fs_drop_inode(
* that. See the comment for this inode flag. * that. See the comment for this inode flag.
*/ */
if (ip->i_flags & XFS_IRECOVERY) { if (ip->i_flags & XFS_IRECOVERY) {
ASSERT(ip->i_mount->m_log->l_flags & XLOG_RECOVERY_NEEDED); ASSERT(xlog_recovery_needed(ip->i_mount->m_log));
return 0; return 0;
} }
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册