]> www.infradead.org Git - users/hch/xfs.git/commitdiff
xfs: realtime rmap btree transaction reservations
authorDarrick J. Wong <djwong@kernel.org>
Thu, 15 Aug 2024 18:48:46 +0000 (11:48 -0700)
committerChristoph Hellwig <hch@lst.de>
Sun, 22 Sep 2024 08:01:36 +0000 (10:01 +0200)
Make sure that there's enough log reservation to handle mapping
and unmapping realtime extents.  We have to reserve enough space
to handle a split in the rtrmapbt to add the record and a second
split in the regular rmapbt to record the rtrmapbt split.

Signed-off-by: Darrick J. Wong <djwong@kernel.org>
fs/xfs/libxfs/xfs_exchmaps.c
fs/xfs/libxfs/xfs_trans_resv.c
fs/xfs/libxfs/xfs_trans_space.h

index 2021396651de272820aaf3b2408506b751552b3d..3f1d6a98c1181987f3517087397854dc469472d6 100644 (file)
@@ -662,7 +662,9 @@ xfs_exchmaps_rmapbt_blocks(
        if (!xfs_has_rmapbt(mp))
                return 0;
        if (XFS_IS_REALTIME_INODE(req->ip1))
-               return 0;
+               return howmany_64(req->nr_exchanges,
+                                       XFS_MAX_CONTIG_RTRMAPS_PER_BLOCK(mp)) *
+                       XFS_RTRMAPADD_SPACE_RES(mp);
 
        return howmany_64(req->nr_exchanges,
                                        XFS_MAX_CONTIG_RMAPS_PER_BLOCK(mp)) *
index bab402340b5da88c3ba790a10e8a15af582b3f6d..f3392eb2d7f41f459369f2f8bc0dc0fad631bc58 100644 (file)
@@ -213,7 +213,9 @@ xfs_calc_inode_chunk_res(
  * Per-extent log reservation for the btree changes involved in freeing or
  * allocating a realtime extent.  We have to be able to log as many rtbitmap
  * blocks as needed to mark inuse XFS_BMBT_MAX_EXTLEN blocks' worth of realtime
- * extents, as well as the realtime summary block.
+ * extents, as well as the realtime summary block (t1).  Realtime rmap btree
+ * operations happen in a second transaction, so factor in a couple of rtrmapbt
+ * splits (t2).
  */
 static unsigned int
 xfs_rtalloc_block_count(
@@ -222,10 +224,16 @@ xfs_rtalloc_block_count(
 {
        unsigned int            rtbmp_blocks;
        xfs_rtxlen_t            rtxlen;
+       unsigned int            t1, t2 = 0;
 
        rtxlen = xfs_extlen_to_rtxlen(mp, XFS_MAX_BMBT_EXTLEN);
        rtbmp_blocks = xfs_rtbitmap_blockcount_len(mp, rtxlen);
-       return (rtbmp_blocks + 1) * num_ops;
+       t1 = (rtbmp_blocks + 1) * num_ops;
+
+       if (xfs_has_rmapbt(mp))
+               t2 = num_ops * (2 * mp->m_rtrmap_maxlevels - 1);
+
+       return max(t1, t2);
 }
 
 /*
index 1155ff2d37e29f7d1519d2f406c9137562bf9568..d89b570aafcc643be4cdff69c5259b69802c4055 100644 (file)
 #define XFS_MAX_CONTIG_BMAPS_PER_BLOCK(mp)    \
                (((mp)->m_bmap_dmxr[0]) - ((mp)->m_bmap_dmnr[0]))
 
+/* Worst case number of realtime rmaps that can be held in a block. */
+#define XFS_MAX_CONTIG_RTRMAPS_PER_BLOCK(mp)    \
+               (((mp)->m_rtrmap_mxr[0]) - ((mp)->m_rtrmap_mnr[0]))
+
+/* Adding one realtime rmap could split every level to the top of the tree. */
+#define XFS_RTRMAPADD_SPACE_RES(mp) ((mp)->m_rtrmap_maxlevels)
+
+/* Blocks we might need to add "b" realtime rmaps to a tree. */
+#define XFS_NRTRMAPADD_SPACE_RES(mp, b) \
+       ((((b) + XFS_MAX_CONTIG_RTRMAPS_PER_BLOCK(mp) - 1) / \
+         XFS_MAX_CONTIG_RTRMAPS_PER_BLOCK(mp)) * \
+         XFS_RTRMAPADD_SPACE_RES(mp))
+
 /* Worst case number of rmaps that can be held in a block. */
 #define XFS_MAX_CONTIG_RMAPS_PER_BLOCK(mp)    \
                (((mp)->m_rmap_mxr[0]) - ((mp)->m_rmap_mnr[0]))