]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs: realtime rmap btree transaction reservations
authorDarrick J. Wong <djwong@kernel.org>
Mon, 24 Feb 2025 18:21:47 +0000 (10:21 -0800)
committerDarrick J. Wong <djwong@kernel.org>
Tue, 25 Feb 2025 17:15:57 +0000 (09:15 -0800)
Source kernel commit: e1c76fce50bb750dff236aa51a3b698de4f7132c

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>
Reviewed-by: Christoph Hellwig <hch@lst.de>
libxfs/xfs_exchmaps.c
libxfs/xfs_trans_resv.c
libxfs/xfs_trans_space.h

index 08bfe6f1b3422d05537e1206d3e05276ccfc16b9..2d7513549429b4a2bd238cd956af9d323d446327 100644 (file)
@@ -659,7 +659,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 93047e149693d6493b7017553b6551f3563c9df4..cdfac7a12906c284a176435bf42757135408a560 100644 (file)
@@ -210,7 +210,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(
@@ -219,10 +221,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]))