xfs: make for_each_perag... a first class citizen
authorDave Chinner <dchinner@redhat.com>
Wed, 2 Jun 2021 00:48:24 +0000 (10:48 +1000)
committerDave Chinner <david@fromorbit.com>
Wed, 2 Jun 2021 00:48:24 +0000 (10:48 +1000)
for_each_perag_tag() is defined in xfs_icache.c for local use.
Promote this to xfs_ag.h and define equivalent iteration functions
so that we can use them to iterate AGs instead to replace open coded
perag walks and perag lookups.

We also convert as many of the straight forward open coded AG walks
to use these iterators as possible. Anything that is not a direct
conversion to an iterator is ignored and will be updated in future
commits.

Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Brian Foster <bfoster@redhat.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
fs/xfs/libxfs/xfs_ag.h
fs/xfs/scrub/fscounters.c
fs/xfs/xfs_extent_busy.c
fs/xfs/xfs_fsops.c
fs/xfs/xfs_health.c
fs/xfs/xfs_icache.c

index ec37f9d..3378312 100644 (file)
@@ -114,6 +114,23 @@ struct xfs_perag *xfs_perag_get_tag(struct xfs_mount *, xfs_agnumber_t,
                                   int tag);
 void   xfs_perag_put(struct xfs_perag *pag);
 
+/*
+ * Perag iteration APIs
+ */
+#define for_each_perag(mp, next_agno, pag) \
+       for ((next_agno) = 0, (pag) = xfs_perag_get((mp), 0); \
+               (pag) != NULL; \
+               (next_agno) = (pag)->pag_agno + 1, \
+               xfs_perag_put(pag), \
+               (pag) = xfs_perag_get((mp), (next_agno)))
+
+#define for_each_perag_tag(mp, next_agno, pag, tag) \
+       for ((next_agno) = 0, (pag) = xfs_perag_get_tag((mp), 0, (tag)); \
+               (pag) != NULL; \
+               (next_agno) = (pag)->pag_agno + 1, \
+               xfs_perag_put(pag), \
+               (pag) = xfs_perag_get_tag((mp), (next_agno), (tag)))
+
 struct aghdr_init_data {
        /* per ag data */
        xfs_agblock_t           agno;           /* ag to init */
index 453ae9a..fd7941e 100644 (file)
@@ -71,11 +71,11 @@ xchk_fscount_warmup(
        xfs_agnumber_t          agno;
        int                     error = 0;
 
-       for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
-               pag = xfs_perag_get(mp, agno);
-
+       for_each_perag(mp, agno, pag) {
+               if (xchk_should_terminate(sc, &error))
+                       break;
                if (pag->pagi_init && pag->pagf_init)
-                       goto next_loop_perag;
+                       continue;
 
                /* Lock both AG headers. */
                error = xfs_ialloc_read_agi(mp, sc->tp, agno, &agi_bp);
@@ -89,21 +89,15 @@ xchk_fscount_warmup(
                 * These are supposed to be initialized by the header read
                 * function.
                 */
-               error = -EFSCORRUPTED;
-               if (!pag->pagi_init || !pag->pagf_init)
+               if (!pag->pagi_init || !pag->pagf_init) {
+                       error = -EFSCORRUPTED;
                        break;
+               }
 
                xfs_buf_relse(agf_bp);
                agf_bp = NULL;
                xfs_buf_relse(agi_bp);
                agi_bp = NULL;
-next_loop_perag:
-               xfs_perag_put(pag);
-               pag = NULL;
-               error = 0;
-
-               if (xchk_should_terminate(sc, &error))
-                       break;
        }
 
        if (agf_bp)
@@ -196,13 +190,14 @@ retry:
        fsc->ifree = 0;
        fsc->fdblocks = 0;
 
-       for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
-               pag = xfs_perag_get(mp, agno);
+       for_each_perag(mp, agno, pag) {
+               if (xchk_should_terminate(sc, &error))
+                       break;
 
                /* This somehow got unset since the warmup? */
                if (!pag->pagi_init || !pag->pagf_init) {
-                       xfs_perag_put(pag);
-                       return -EFSCORRUPTED;
+                       error = -EFSCORRUPTED;
+                       break;
                }
 
                /* Count all the inodes */
@@ -216,10 +211,8 @@ retry:
                        fsc->fdblocks += pag->pagf_btreeblks;
                } else {
                        error = xchk_fscount_btreeblks(sc, fsc, agno);
-                       if (error) {
-                               xfs_perag_put(pag);
+                       if (error)
                                break;
-                       }
                }
 
                /*
@@ -229,12 +222,9 @@ retry:
                fsc->fdblocks -= pag->pag_meta_resv.ar_reserved;
                fsc->fdblocks -= pag->pag_rmapbt_resv.ar_orig_reserved;
 
-               xfs_perag_put(pag);
-
-               if (xchk_should_terminate(sc, &error))
-                       break;
        }
-
+       if (pag)
+               xfs_perag_put(pag);
        if (error)
                return error;
 
index cb037d7..422667e 100644 (file)
@@ -605,12 +605,11 @@ void
 xfs_extent_busy_wait_all(
        struct xfs_mount        *mp)
 {
+       struct xfs_perag        *pag;
        DEFINE_WAIT             (wait);
        xfs_agnumber_t          agno;
 
-       for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
-               struct xfs_perag *pag = xfs_perag_get(mp, agno);
-
+       for_each_perag(mp, agno, pag) {
                do {
                        prepare_to_wait(&pag->pagb_wait, &wait, TASK_KILLABLE);
                        if  (RB_EMPTY_ROOT(&pag->pagb_tree))
@@ -618,8 +617,6 @@ xfs_extent_busy_wait_all(
                        schedule();
                } while (1);
                finish_wait(&pag->pagb_wait, &wait);
-
-               xfs_perag_put(pag);
        }
 }
 
index be9cf88..07c745c 100644 (file)
@@ -576,10 +576,8 @@ xfs_fs_reserve_ag_blocks(
        int                     err2;
 
        mp->m_finobt_nores = false;
-       for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
-               pag = xfs_perag_get(mp, agno);
+       for_each_perag(mp, agno, pag) {
                err2 = xfs_ag_resv_init(pag, NULL);
-               xfs_perag_put(pag);
                if (err2 && !error)
                        error = err2;
        }
@@ -605,10 +603,8 @@ xfs_fs_unreserve_ag_blocks(
        int                     error = 0;
        int                     err2;
 
-       for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
-               pag = xfs_perag_get(mp, agno);
+       for_each_perag(mp, agno, pag) {
                err2 = xfs_ag_resv_free(pag);
-               xfs_perag_put(pag);
                if (err2 && !error)
                        error = err2;
        }
index b79475e..5de3195 100644 (file)
@@ -34,14 +34,12 @@ xfs_health_unmount(
                return;
 
        /* Measure AG corruption levels. */
-       for (agno = 0; agno < mp->m_sb.sb_agcount; agno++) {
-               pag = xfs_perag_get(mp, agno);
+       for_each_perag(mp, agno, pag) {
                xfs_ag_measure_sickness(pag, &sick, &checked);
                if (sick) {
                        trace_xfs_ag_unfixed_corruption(mp, agno, sick);
                        warn = true;
                }
-               xfs_perag_put(pag);
        }
 
        /* Measure realtime volume corruption levels. */
index 588ea2b..7dad83a 100644 (file)
@@ -1061,15 +1061,13 @@ xfs_reclaim_inodes_ag(
        int                     *nr_to_scan)
 {
        struct xfs_perag        *pag;
-       xfs_agnumber_t          ag = 0;
+       xfs_agnumber_t          agno;
 
-       while ((pag = xfs_perag_get_tag(mp, ag, XFS_ICI_RECLAIM_TAG))) {
+       for_each_perag_tag(mp, agno, pag, XFS_ICI_RECLAIM_TAG) {
                unsigned long   first_index = 0;
                int             done = 0;
                int             nr_found = 0;
 
-               ag = pag->pag_agno + 1;
-
                first_index = READ_ONCE(pag->pag_ici_reclaim_cursor);
                do {
                        struct xfs_inode *batch[XFS_LOOKUP_BATCH];
@@ -1134,7 +1132,6 @@ xfs_reclaim_inodes_ag(
                if (done)
                        first_index = 0;
                WRITE_ONCE(pag->pag_ici_reclaim_cursor, first_index);
-               xfs_perag_put(pag);
        }
 }
 
@@ -1554,14 +1551,6 @@ xfs_inode_clear_cowblocks_tag(
        return xfs_blockgc_clear_iflag(ip, XFS_ICOWBLOCKS);
 }
 
-#define for_each_perag_tag(mp, next_agno, pag, tag) \
-       for ((next_agno) = 0, (pag) = xfs_perag_get_tag((mp), 0, (tag)); \
-               (pag) != NULL; \
-               (next_agno) = (pag)->pag_agno + 1, \
-               xfs_perag_put(pag), \
-               (pag) = xfs_perag_get_tag((mp), (next_agno), (tag)))
-
-
 /* Disable post-EOF and CoW block auto-reclamation. */
 void
 xfs_blockgc_stop(