aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--fs/xfs/xfs_log.c36
-rw-r--r--fs/xfs/xfs_log_priv.h2
2 files changed, 21 insertions, 17 deletions
diff --git a/fs/xfs/xfs_log.c b/fs/xfs/xfs_log.c
index 2e35077ff6b2..1fa980933895 100644
--- a/fs/xfs/xfs_log.c
+++ b/fs/xfs/xfs_log.c
@@ -675,7 +675,7 @@ xfs_log_unmount_write(xfs_mount_t *mp)
675 675
676 spin_lock(&log->l_icloglock); 676 spin_lock(&log->l_icloglock);
677 iclog = log->l_iclog; 677 iclog = log->l_iclog;
678 iclog->ic_refcnt++; 678 atomic_inc(&iclog->ic_refcnt);
679 spin_unlock(&log->l_icloglock); 679 spin_unlock(&log->l_icloglock);
680 xlog_state_want_sync(log, iclog); 680 xlog_state_want_sync(log, iclog);
681 (void) xlog_state_release_iclog(log, iclog); 681 (void) xlog_state_release_iclog(log, iclog);
@@ -713,7 +713,7 @@ xfs_log_unmount_write(xfs_mount_t *mp)
713 */ 713 */
714 spin_lock(&log->l_icloglock); 714 spin_lock(&log->l_icloglock);
715 iclog = log->l_iclog; 715 iclog = log->l_iclog;
716 iclog->ic_refcnt++; 716 atomic_inc(&iclog->ic_refcnt);
717 spin_unlock(&log->l_icloglock); 717 spin_unlock(&log->l_icloglock);
718 718
719 xlog_state_want_sync(log, iclog); 719 xlog_state_want_sync(log, iclog);
@@ -1405,7 +1405,7 @@ xlog_sync(xlog_t *log,
1405 int v2 = xfs_sb_version_haslogv2(&log->l_mp->m_sb); 1405 int v2 = xfs_sb_version_haslogv2(&log->l_mp->m_sb);
1406 1406
1407 XFS_STATS_INC(xs_log_writes); 1407 XFS_STATS_INC(xs_log_writes);
1408 ASSERT(iclog->ic_refcnt == 0); 1408 ASSERT(atomic_read(&iclog->ic_refcnt) == 0);
1409 1409
1410 /* Add for LR header */ 1410 /* Add for LR header */
1411 count_init = log->l_iclog_hsize + iclog->ic_offset; 1411 count_init = log->l_iclog_hsize + iclog->ic_offset;
@@ -2309,7 +2309,7 @@ xlog_state_done_syncing(
2309 2309
2310 ASSERT(iclog->ic_state == XLOG_STATE_SYNCING || 2310 ASSERT(iclog->ic_state == XLOG_STATE_SYNCING ||
2311 iclog->ic_state == XLOG_STATE_IOERROR); 2311 iclog->ic_state == XLOG_STATE_IOERROR);
2312 ASSERT(iclog->ic_refcnt == 0); 2312 ASSERT(atomic_read(&iclog->ic_refcnt) == 0);
2313 ASSERT(iclog->ic_bwritecnt == 1 || iclog->ic_bwritecnt == 2); 2313 ASSERT(iclog->ic_bwritecnt == 1 || iclog->ic_bwritecnt == 2);
2314 2314
2315 2315
@@ -2391,7 +2391,7 @@ restart:
2391 ASSERT(iclog->ic_state == XLOG_STATE_ACTIVE); 2391 ASSERT(iclog->ic_state == XLOG_STATE_ACTIVE);
2392 head = &iclog->ic_header; 2392 head = &iclog->ic_header;
2393 2393
2394 iclog->ic_refcnt++; /* prevents sync */ 2394 atomic_inc(&iclog->ic_refcnt); /* prevents sync */
2395 log_offset = iclog->ic_offset; 2395 log_offset = iclog->ic_offset;
2396 2396
2397 /* On the 1st write to an iclog, figure out lsn. This works 2397 /* On the 1st write to an iclog, figure out lsn. This works
@@ -2423,12 +2423,12 @@ restart:
2423 xlog_state_switch_iclogs(log, iclog, iclog->ic_size); 2423 xlog_state_switch_iclogs(log, iclog, iclog->ic_size);
2424 2424
2425 /* If I'm the only one writing to this iclog, sync it to disk */ 2425 /* If I'm the only one writing to this iclog, sync it to disk */
2426 if (iclog->ic_refcnt == 1) { 2426 if (atomic_read(&iclog->ic_refcnt) == 1) {
2427 spin_unlock(&log->l_icloglock); 2427 spin_unlock(&log->l_icloglock);
2428 if ((error = xlog_state_release_iclog(log, iclog))) 2428 if ((error = xlog_state_release_iclog(log, iclog)))
2429 return error; 2429 return error;
2430 } else { 2430 } else {
2431 iclog->ic_refcnt--; 2431 atomic_dec(&iclog->ic_refcnt);
2432 spin_unlock(&log->l_icloglock); 2432 spin_unlock(&log->l_icloglock);
2433 } 2433 }
2434 goto restart; 2434 goto restart;
@@ -2819,18 +2819,21 @@ xlog_state_release_iclog(
2819{ 2819{
2820 int sync = 0; /* do we sync? */ 2820 int sync = 0; /* do we sync? */
2821 2821
2822 spin_lock(&log->l_icloglock); 2822 if (iclog->ic_state & XLOG_STATE_IOERROR)
2823 return XFS_ERROR(EIO);
2824
2825 ASSERT(atomic_read(&iclog->ic_refcnt) > 0);
2826 if (!atomic_dec_and_lock(&iclog->ic_refcnt, &log->l_icloglock))
2827 return 0;
2828
2823 if (iclog->ic_state & XLOG_STATE_IOERROR) { 2829 if (iclog->ic_state & XLOG_STATE_IOERROR) {
2824 spin_unlock(&log->l_icloglock); 2830 spin_unlock(&log->l_icloglock);
2825 return XFS_ERROR(EIO); 2831 return XFS_ERROR(EIO);
2826 } 2832 }
2827
2828 ASSERT(iclog->ic_refcnt > 0);
2829 ASSERT(iclog->ic_state == XLOG_STATE_ACTIVE || 2833 ASSERT(iclog->ic_state == XLOG_STATE_ACTIVE ||
2830 iclog->ic_state == XLOG_STATE_WANT_SYNC); 2834 iclog->ic_state == XLOG_STATE_WANT_SYNC);
2831 2835
2832 if (--iclog->ic_refcnt == 0 && 2836 if (iclog->ic_state == XLOG_STATE_WANT_SYNC) {
2833 iclog->ic_state == XLOG_STATE_WANT_SYNC) {
2834 /* update tail before writing to iclog */ 2837 /* update tail before writing to iclog */
2835 xlog_assign_tail_lsn(log->l_mp); 2838 xlog_assign_tail_lsn(log->l_mp);
2836 sync++; 2839 sync++;
@@ -2950,7 +2953,8 @@ xlog_state_sync_all(xlog_t *log, uint flags, int *log_flushed)
2950 * previous iclog and go to sleep. 2953 * previous iclog and go to sleep.
2951 */ 2954 */
2952 if (iclog->ic_state == XLOG_STATE_DIRTY || 2955 if (iclog->ic_state == XLOG_STATE_DIRTY ||
2953 (iclog->ic_refcnt == 0 && iclog->ic_offset == 0)) { 2956 (atomic_read(&iclog->ic_refcnt) == 0
2957 && iclog->ic_offset == 0)) {
2954 iclog = iclog->ic_prev; 2958 iclog = iclog->ic_prev;
2955 if (iclog->ic_state == XLOG_STATE_ACTIVE || 2959 if (iclog->ic_state == XLOG_STATE_ACTIVE ||
2956 iclog->ic_state == XLOG_STATE_DIRTY) 2960 iclog->ic_state == XLOG_STATE_DIRTY)
@@ -2958,14 +2962,14 @@ xlog_state_sync_all(xlog_t *log, uint flags, int *log_flushed)
2958 else 2962 else
2959 goto maybe_sleep; 2963 goto maybe_sleep;
2960 } else { 2964 } else {
2961 if (iclog->ic_refcnt == 0) { 2965 if (atomic_read(&iclog->ic_refcnt) == 0) {
2962 /* We are the only one with access to this 2966 /* We are the only one with access to this
2963 * iclog. Flush it out now. There should 2967 * iclog. Flush it out now. There should
2964 * be a roundoff of zero to show that someone 2968 * be a roundoff of zero to show that someone
2965 * has already taken care of the roundoff from 2969 * has already taken care of the roundoff from
2966 * the previous sync. 2970 * the previous sync.
2967 */ 2971 */
2968 iclog->ic_refcnt++; 2972 atomic_inc(&iclog->ic_refcnt);
2969 lsn = be64_to_cpu(iclog->ic_header.h_lsn); 2973 lsn = be64_to_cpu(iclog->ic_header.h_lsn);
2970 xlog_state_switch_iclogs(log, iclog, 0); 2974 xlog_state_switch_iclogs(log, iclog, 0);
2971 spin_unlock(&log->l_icloglock); 2975 spin_unlock(&log->l_icloglock);
@@ -3097,7 +3101,7 @@ try_again:
3097 already_slept = 1; 3101 already_slept = 1;
3098 goto try_again; 3102 goto try_again;
3099 } else { 3103 } else {
3100 iclog->ic_refcnt++; 3104 atomic_inc(&iclog->ic_refcnt);
3101 xlog_state_switch_iclogs(log, iclog, 0); 3105 xlog_state_switch_iclogs(log, iclog, 0);
3102 spin_unlock(&log->l_icloglock); 3106 spin_unlock(&log->l_icloglock);
3103 if (xlog_state_release_iclog(log, iclog)) 3107 if (xlog_state_release_iclog(log, iclog))
diff --git a/fs/xfs/xfs_log_priv.h b/fs/xfs/xfs_log_priv.h
index c6244cc733c0..01c63db25a1d 100644
--- a/fs/xfs/xfs_log_priv.h
+++ b/fs/xfs/xfs_log_priv.h
@@ -339,7 +339,7 @@ typedef struct xlog_iclog_fields {
339#endif 339#endif
340 int ic_size; 340 int ic_size;
341 int ic_offset; 341 int ic_offset;
342 int ic_refcnt; 342 atomic_t ic_refcnt;
343 int ic_bwritecnt; 343 int ic_bwritecnt;
344 ushort_t ic_state; 344 ushort_t ic_state;
345 char *ic_datap; /* pointer to iclog data */ 345 char *ic_datap; /* pointer to iclog data */