]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs: introduce in-core global counter of allocbt blocks libxfs-5.13-sync_2021-05-21
authorBrian Foster <bfoster@redhat.com>
Wed, 5 May 2021 16:11:33 +0000 (09:11 -0700)
committerDarrick J. Wong <djwong@kernel.org>
Fri, 21 May 2021 15:31:36 +0000 (08:31 -0700)
Source kernel commit: 16eaab839a9273ed156ebfccbd40c15d1e72f3d8

Introduce an in-core counter to track the sum of all allocbt blocks
used by the filesystem. This value is currently tracked per-ag via
the ->agf_btreeblks field in the AGF, which also happens to include
rmapbt blocks. A global, in-core count of allocbt blocks is required
to identify the subset of global ->m_fdblocks that consists of
unavailable blocks currently used for allocation btrees. To support
this calculation at block reservation time, construct a similar
global counter for allocbt blocks, populate it on first read of each
AGF and update it as allocbt blocks are used and released.

Signed-off-by: Brian Foster <bfoster@redhat.com>
Reviewed-by: Chandan Babu R <chandanrlinux@gmail.com>
Reviewed-by: Allison Henderson <allison.henderson@oracle.com>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
include/atomic.h
include/xfs_mount.h
libxfs/xfs_alloc.c
libxfs/xfs_alloc_btree.c

index 1aabecc3ae57b7994fe6c4d140a98e9ba4796915..e30f8435dbe9f4ae5962136193877265055b7dbf 100644 (file)
@@ -20,5 +20,8 @@ typedef       int64_t atomic64_t;
 
 #define atomic64_read(x)       *(x)
 #define atomic64_set(x, v)     (*(x) = v)
+#define atomic64_inc(x)                atomic_inc_return(x)
+#define atomic64_dec(x)                atomic_dec_return(x)
+#define atomic64_add(v, x)     (*(x) += (v))
 
 #endif /* __ATOMIC_H__ */
index f93a9f11420138ae70fbf05d4180b13b3fdd0588..05619b4710de2848a1396fa233108e4c58996271 100644 (file)
@@ -30,6 +30,9 @@ typedef struct xfs_mount {
        uint8_t                 m_fs_checked;
        uint8_t                 m_fs_sick;
 
+       /* dummy variable used by the kernel to avoid ENOSPC */
+       uint8_t                 m_allocbt_blks;
+
        char                    *m_fsname;      /* filesystem name */
        int                     m_bsize;        /* fs logical block size */
        xfs_agnumber_t          m_agfrotor;     /* last ag where space found */
index 699c15400d523c3827cc1c839ff823d8121d6446..d99622a6848f9b3ded458b1901e6682694061013 100644 (file)
@@ -3029,6 +3029,7 @@ xfs_alloc_read_agf(
        struct xfs_agf          *agf;           /* ag freelist header */
        struct xfs_perag        *pag;           /* per allocation group data */
        int                     error;
+       int                     allocbt_blks;
 
        trace_xfs_alloc_read_agf(mp, agno);
 
@@ -3059,6 +3060,19 @@ xfs_alloc_read_agf(
                pag->pagf_refcount_level = be32_to_cpu(agf->agf_refcount_level);
                pag->pagf_init = 1;
                pag->pagf_agflreset = xfs_agfl_needs_reset(mp, agf);
+
+               /*
+                * Update the in-core allocbt counter. Filter out the rmapbt
+                * subset of the btreeblks counter because the rmapbt is managed
+                * by perag reservation. Subtract one for the rmapbt root block
+                * because the rmap counter includes it while the btreeblks
+                * counter only tracks non-root blocks.
+                */
+               allocbt_blks = pag->pagf_btreeblks;
+               if (xfs_sb_version_hasrmapbt(&mp->m_sb))
+                       allocbt_blks -= be32_to_cpu(agf->agf_rmap_blocks) - 1;
+               if (allocbt_blks > 0)
+                       atomic64_add(allocbt_blks, &mp->m_allocbt_blks);
        }
 #ifdef DEBUG
        else if (!XFS_FORCED_SHUTDOWN(mp)) {
index 9b5dce55e33e238261f2d861ebf9f050c518d925..4611ed0f7dc439acefe8712e5ffbb513bf06b032 100644 (file)
@@ -69,6 +69,7 @@ xfs_allocbt_alloc_block(
                return 0;
        }
 
+       atomic64_inc(&cur->bc_mp->m_allocbt_blks);
        xfs_extent_busy_reuse(cur->bc_mp, cur->bc_ag.agno, bno, 1, false);
 
        new->s = cpu_to_be32(bno);
@@ -92,6 +93,7 @@ xfs_allocbt_free_block(
        if (error)
                return error;
 
+       atomic64_dec(&cur->bc_mp->m_allocbt_blks);
        xfs_extent_busy_insert(cur->bc_tp, be32_to_cpu(agf->agf_seqno), bno, 1,
                              XFS_EXTENT_BUSY_SKIP_DISCARD);
        return 0;