]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs_db: display the realtime rmap btree contents
authorDarrick J. Wong <djwong@kernel.org>
Mon, 24 Feb 2025 18:21:58 +0000 (10:21 -0800)
committerDarrick J. Wong <djwong@kernel.org>
Tue, 25 Feb 2025 17:16:00 +0000 (09:16 -0800)
Implement all the code we need to dump rtrmapbt contents, starting
from the inode root.

Signed-off-by: "Darrick J. Wong" <djwong@kernel.org>
Reviewed-by: Christoph Hellwig <hch@lst.de>
db/bmroot.c
db/bmroot.h
db/btblock.c
db/btblock.h
db/field.c
db/field.h
db/inode.c
db/type.c
db/type.h
libxfs/libxfs_api_defs.h
man/man8/xfs_db.8

index 7ef07da181e6ff72684714954675bcf3ca47c3ae..65d808c46819460aed416b353ca351c4d71444d4 100644 (file)
@@ -24,6 +24,13 @@ static int   bmrootd_key_offset(void *obj, int startoff, int idx);
 static int     bmrootd_ptr_count(void *obj, int startoff);
 static int     bmrootd_ptr_offset(void *obj, int startoff, int idx);
 
+static int     rtrmaproot_rec_count(void *obj, int startoff);
+static int     rtrmaproot_rec_offset(void *obj, int startoff, int idx);
+static int     rtrmaproot_key_count(void *obj, int startoff);
+static int     rtrmaproot_key_offset(void *obj, int startoff, int idx);
+static int     rtrmaproot_ptr_count(void *obj, int startoff);
+static int     rtrmaproot_ptr_offset(void *obj, int startoff, int idx);
+
 #define        OFF(f)  bitize(offsetof(xfs_bmdr_block_t, bb_ ## f))
 const field_t  bmroota_flds[] = {
        { "level", FLDT_UINT16D, OI(OFF(level)), C1, 0, TYP_NONE },
@@ -54,6 +61,20 @@ const field_t        bmrootd_key_flds[] = {
        { NULL }
 };
 
+/* realtime rmap btree root */
+const field_t  rtrmaproot_flds[] = {
+       { "level", FLDT_UINT16D, OI(OFF(level)), C1, 0, TYP_NONE },
+       { "numrecs", FLDT_UINT16D, OI(OFF(numrecs)), C1, 0, TYP_NONE },
+       { "recs", FLDT_RTRMAPBTREC, rtrmaproot_rec_offset, rtrmaproot_rec_count,
+         FLD_ARRAY|FLD_ABASE1|FLD_COUNT|FLD_OFFSET, TYP_NONE },
+       { "keys", FLDT_RTRMAPBTKEY, rtrmaproot_key_offset, rtrmaproot_key_count,
+         FLD_ARRAY|FLD_ABASE1|FLD_COUNT|FLD_OFFSET, TYP_NONE },
+       { "ptrs", FLDT_RTRMAPBTPTR, rtrmaproot_ptr_offset, rtrmaproot_ptr_count,
+         FLD_ARRAY|FLD_ABASE1|FLD_COUNT|FLD_OFFSET, TYP_RTRMAPBT },
+       { NULL }
+};
+#undef OFF
+
 static int
 bmroota_key_count(
        void                    *obj,
@@ -241,3 +262,117 @@ bmrootd_size(
        dip = obj;
        return bitize((int)XFS_DFORK_DSIZE(dip, mp));
 }
+
+/* realtime rmap root */
+static int
+rtrmaproot_rec_count(
+       void                    *obj,
+       int                     startoff)
+{
+       struct xfs_dinode       *dip = obj;
+       struct xfs_rtrmap_root  *block = XFS_DFORK_DPTR(dip);
+
+       ASSERT(block == obj + byteize(startoff));
+
+       if (be16_to_cpu(block->bb_level) > 0)
+               return 0;
+       return be16_to_cpu(block->bb_numrecs);
+}
+
+static int
+rtrmaproot_rec_offset(
+       void                    *obj,
+       int                     startoff,
+       int                     idx)
+{
+       struct xfs_dinode       *dip = obj;
+       struct xfs_rtrmap_root  *block = XFS_DFORK_DPTR(dip);
+       struct xfs_rmap_rec     *kp;
+
+       ASSERT(block == obj + byteize(startoff));
+       ASSERT(be16_to_cpu(block->bb_level) == 0);
+
+       kp = xfs_rtrmap_droot_rec_addr(block, idx);
+       return bitize((int)((char *)kp - (char *)block));
+}
+
+static int
+rtrmaproot_key_count(
+       void                    *obj,
+       int                     startoff)
+{
+       struct xfs_dinode       *dip = obj;
+       struct xfs_rtrmap_root  *block = XFS_DFORK_DPTR(dip);
+
+       ASSERT(block == obj + byteize(startoff));
+
+       if (be16_to_cpu(block->bb_level) == 0)
+               return 0;
+       return be16_to_cpu(block->bb_numrecs);
+}
+
+static int
+rtrmaproot_key_offset(
+       void                    *obj,
+       int                     startoff,
+       int                     idx)
+{
+       struct xfs_dinode       *dip = obj;
+       struct xfs_rtrmap_root  *block = XFS_DFORK_DPTR(dip);
+       struct xfs_rmap_key     *kp;
+
+       ASSERT(block == obj + byteize(startoff));
+       ASSERT(be16_to_cpu(block->bb_level) > 0);
+
+       kp = xfs_rtrmap_droot_key_addr(block, idx);
+       return bitize((int)((char *)kp - (char *)block));
+}
+
+static int
+rtrmaproot_ptr_count(
+       void                    *obj,
+       int                     startoff)
+{
+       struct xfs_dinode       *dip = obj;
+       struct xfs_rtrmap_root  *block = XFS_DFORK_DPTR(dip);
+
+       ASSERT(block == obj + byteize(startoff));
+
+       if (be16_to_cpu(block->bb_level) == 0)
+               return 0;
+       return be16_to_cpu(block->bb_numrecs);
+}
+
+static int
+rtrmaproot_ptr_offset(
+       void                    *obj,
+       int                     startoff,
+       int                     idx)
+{
+       struct xfs_dinode       *dip = obj;
+       struct xfs_rtrmap_root  *block = XFS_DFORK_DPTR(dip);
+       xfs_rtrmap_ptr_t        *pp;
+       int                     dmxr;
+
+       ASSERT(block == obj + byteize(startoff));
+       ASSERT(be16_to_cpu(block->bb_level) > 0);
+
+       dmxr = libxfs_rtrmapbt_droot_maxrecs(XFS_DFORK_DSIZE(dip, mp), false);
+       pp = xfs_rtrmap_droot_ptr_addr(block, idx, dmxr);
+       return bitize((int)((char *)pp - (char *)block));
+}
+
+int
+rtrmaproot_size(
+       void                    *obj,
+       int                     startoff,
+       int                     idx)
+{
+       struct xfs_dinode       *dip;
+
+       ASSERT(bitoffs(startoff) == 0);
+       ASSERT(obj == iocur_top->data);
+       ASSERT(idx == 0);
+       dip = obj;
+       return bitize((int)XFS_DFORK_DSIZE(dip, mp));
+}
index a1274cf6a94bdf3484e9cb1afdb3f96f5c64df70..a2c5cfb18f0bb2ec0474f1a8317fb5036ffa2ad2 100644 (file)
@@ -8,6 +8,8 @@ extern const struct field       bmroota_flds[];
 extern const struct field      bmroota_key_flds[];
 extern const struct field      bmrootd_flds[];
 extern const struct field      bmrootd_key_flds[];
+extern const struct field      rtrmaproot_flds[];
 
 extern int     bmroota_size(void *obj, int startoff, int idx);
 extern int     bmrootd_size(void *obj, int startoff, int idx);
+extern int     rtrmaproot_size(void *obj, int startoff, int idx);
index d5be6adb734cef06f828609ecad378fa54cd4aa7..5cad166278d98b034ab2dcbe4b67f322c00cf919 100644 (file)
@@ -92,6 +92,12 @@ static struct xfs_db_btree {
                sizeof(struct xfs_rmap_rec),
                sizeof(__be32),
        },
+       {       XFS_RTRMAP_CRC_MAGIC,
+               XFS_BTREE_LBLOCK_CRC_LEN,
+               2 * sizeof(struct xfs_rmap_key),
+               sizeof(struct xfs_rmap_rec),
+               sizeof(__be64),
+       },
        {       XFS_REFC_CRC_MAGIC,
                XFS_BTREE_SBLOCK_CRC_LEN,
                sizeof(struct xfs_refcount_key),
@@ -813,6 +819,100 @@ const field_t     rmapbt_rec_flds[] = {
        { NULL }
 };
 
+/* realtime RMAP btree blocks */
+const field_t  rtrmapbt_crc_hfld[] = {
+       { "", FLDT_RTRMAPBT_CRC, OI(0), C1, 0, TYP_NONE },
+       { NULL }
+};
+
+#define        OFF(f)  bitize(offsetof(struct xfs_btree_block, bb_ ## f))
+const field_t  rtrmapbt_crc_flds[] = {
+       { "magic", FLDT_UINT32X, OI(OFF(magic)), C1, 0, TYP_NONE },
+       { "level", FLDT_UINT16D, OI(OFF(level)), C1, 0, TYP_NONE },
+       { "numrecs", FLDT_UINT16D, OI(OFF(numrecs)), C1, 0, TYP_NONE },
+       { "leftsib", FLDT_DFSBNO, OI(OFF(u.l.bb_leftsib)), C1, 0, TYP_RTRMAPBT },
+       { "rightsib", FLDT_DFSBNO, OI(OFF(u.l.bb_rightsib)), C1, 0, TYP_RTRMAPBT },
+       { "bno", FLDT_DFSBNO, OI(OFF(u.l.bb_blkno)), C1, 0, TYP_RTRMAPBT },
+       { "lsn", FLDT_UINT64X, OI(OFF(u.l.bb_lsn)), C1, 0, TYP_NONE },
+       { "uuid", FLDT_UUID, OI(OFF(u.l.bb_uuid)), C1, 0, TYP_NONE },
+       { "owner", FLDT_INO, OI(OFF(u.l.bb_owner)), C1, 0, TYP_NONE },
+       { "crc", FLDT_CRC, OI(OFF(u.l.bb_crc)), C1, 0, TYP_NONE },
+       { "recs", FLDT_RTRMAPBTREC, btblock_rec_offset, btblock_rec_count,
+         FLD_ARRAY|FLD_ABASE1|FLD_COUNT|FLD_OFFSET, TYP_NONE },
+       { "keys", FLDT_RTRMAPBTKEY, btblock_key_offset, btblock_key_count,
+         FLD_ARRAY|FLD_ABASE1|FLD_COUNT|FLD_OFFSET, TYP_NONE },
+       { "ptrs", FLDT_RTRMAPBTPTR, btblock_ptr_offset, btblock_key_count,
+         FLD_ARRAY|FLD_ABASE1|FLD_COUNT|FLD_OFFSET, TYP_RTRMAPBT },
+       { NULL }
+};
+#undef OFF
+
+#define        KOFF(f) bitize(offsetof(struct xfs_rmap_key, rm_ ## f))
+
+#define RTRMAPBK_STARTBLOCK_BITOFF     0
+#define RTRMAPBK_OWNER_BITOFF          (RTRMAPBK_STARTBLOCK_BITOFF + RMAPBT_STARTBLOCK_BITLEN)
+#define RTRMAPBK_ATTRFLAG_BITOFF       (RTRMAPBK_OWNER_BITOFF + RMAPBT_OWNER_BITLEN)
+#define RTRMAPBK_BMBTFLAG_BITOFF       (RTRMAPBK_ATTRFLAG_BITOFF + RMAPBT_ATTRFLAG_BITLEN)
+#define RTRMAPBK_EXNTFLAG_BITOFF       (RTRMAPBK_BMBTFLAG_BITOFF + RMAPBT_BMBTFLAG_BITLEN)
+#define RTRMAPBK_UNUSED_OFFSET_BITOFF  (RTRMAPBK_EXNTFLAG_BITOFF + RMAPBT_EXNTFLAG_BITLEN)
+#define RTRMAPBK_OFFSET_BITOFF         (RTRMAPBK_UNUSED_OFFSET_BITOFF + RMAPBT_UNUSED_OFFSET_BITLEN)
+
+#define HI_KOFF(f)     bitize(sizeof(struct xfs_rmap_key) + offsetof(struct xfs_rmap_key, rm_ ## f))
+
+#define RTRMAPBK_STARTBLOCKHI_BITOFF   (bitize(sizeof(struct xfs_rmap_key)))
+#define RTRMAPBK_OWNERHI_BITOFF                (RTRMAPBK_STARTBLOCKHI_BITOFF + RMAPBT_STARTBLOCK_BITLEN)
+#define RTRMAPBK_ATTRFLAGHI_BITOFF     (RTRMAPBK_OWNERHI_BITOFF + RMAPBT_OWNER_BITLEN)
+#define RTRMAPBK_BMBTFLAGHI_BITOFF     (RTRMAPBK_ATTRFLAGHI_BITOFF + RMAPBT_ATTRFLAG_BITLEN)
+#define RTRMAPBK_EXNTFLAGHI_BITOFF     (RTRMAPBK_BMBTFLAGHI_BITOFF + RMAPBT_BMBTFLAG_BITLEN)
+#define RTRMAPBK_UNUSED_OFFSETHI_BITOFF        (RTRMAPBK_EXNTFLAGHI_BITOFF + RMAPBT_EXNTFLAG_BITLEN)
+#define RTRMAPBK_OFFSETHI_BITOFF       (RTRMAPBK_UNUSED_OFFSETHI_BITOFF + RMAPBT_UNUSED_OFFSET_BITLEN)
+
+const field_t  rtrmapbt_key_flds[] = {
+       { "startblock", FLDT_RGBLOCK, OI(KOFF(startblock)), C1, 0, TYP_DATA },
+       { "owner", FLDT_INT64D, OI(KOFF(owner)), C1, 0, TYP_NONE },
+       { "offset", FLDT_RFILEOFFD, OI(RTRMAPBK_OFFSET_BITOFF), C1, 0, TYP_NONE },
+       { "attrfork", FLDT_RATTRFORKFLG, OI(RTRMAPBK_ATTRFLAG_BITOFF), C1, 0,
+         TYP_NONE },
+       { "bmbtblock", FLDT_RBMBTFLG, OI(RTRMAPBK_BMBTFLAG_BITOFF), C1, 0,
+         TYP_NONE },
+       { "startblock_hi", FLDT_RGBLOCK, OI(HI_KOFF(startblock)), C1, 0, TYP_DATA },
+       { "owner_hi", FLDT_INT64D, OI(HI_KOFF(owner)), C1, 0, TYP_NONE },
+       { "offset_hi", FLDT_RFILEOFFD, OI(RTRMAPBK_OFFSETHI_BITOFF), C1, 0, TYP_NONE },
+       { "attrfork_hi", FLDT_RATTRFORKFLG, OI(RTRMAPBK_ATTRFLAGHI_BITOFF), C1, 0,
+         TYP_NONE },
+       { "bmbtblock_hi", FLDT_RBMBTFLG, OI(RTRMAPBK_BMBTFLAGHI_BITOFF), C1, 0,
+         TYP_NONE },
+       { NULL }
+};
+#undef HI_KOFF
+#undef KOFF
+
+#define        ROFF(f) bitize(offsetof(struct xfs_rmap_rec, rm_ ## f))
+
+#define RTRMAPBT_STARTBLOCK_BITOFF     0
+#define RTRMAPBT_BLOCKCOUNT_BITOFF     (RTRMAPBT_STARTBLOCK_BITOFF + RMAPBT_STARTBLOCK_BITLEN)
+#define RTRMAPBT_OWNER_BITOFF          (RTRMAPBT_BLOCKCOUNT_BITOFF + RMAPBT_BLOCKCOUNT_BITLEN)
+#define RTRMAPBT_ATTRFLAG_BITOFF       (RTRMAPBT_OWNER_BITOFF + RMAPBT_OWNER_BITLEN)
+#define RTRMAPBT_BMBTFLAG_BITOFF       (RTRMAPBT_ATTRFLAG_BITOFF + RMAPBT_ATTRFLAG_BITLEN)
+#define RTRMAPBT_EXNTFLAG_BITOFF       (RTRMAPBT_BMBTFLAG_BITOFF + RMAPBT_BMBTFLAG_BITLEN)
+#define RTRMAPBT_UNUSED_OFFSET_BITOFF  (RTRMAPBT_EXNTFLAG_BITOFF + RMAPBT_EXNTFLAG_BITLEN)
+#define RTRMAPBT_OFFSET_BITOFF         (RTRMAPBT_UNUSED_OFFSET_BITOFF + RMAPBT_UNUSED_OFFSET_BITLEN)
+
+const field_t  rtrmapbt_rec_flds[] = {
+       { "startblock", FLDT_RGBLOCK, OI(RTRMAPBT_STARTBLOCK_BITOFF), C1, 0, TYP_DATA },
+       { "blockcount", FLDT_EXTLEN, OI(RTRMAPBT_BLOCKCOUNT_BITOFF), C1, 0, TYP_NONE },
+       { "owner", FLDT_INT64D, OI(RTRMAPBT_OWNER_BITOFF), C1, 0, TYP_NONE },
+       { "offset", FLDT_RFILEOFFD, OI(RTRMAPBT_OFFSET_BITOFF), C1, 0, TYP_NONE },
+       { "extentflag", FLDT_REXTFLG, OI(RTRMAPBT_EXNTFLAG_BITOFF), C1, 0,
+         TYP_NONE },
+       { "attrfork", FLDT_RATTRFORKFLG, OI(RTRMAPBT_ATTRFLAG_BITOFF), C1, 0,
+         TYP_NONE },
+       { "bmbtblock", FLDT_RBMBTFLG, OI(RTRMAPBT_BMBTFLAG_BITOFF), C1, 0,
+         TYP_NONE },
+       { NULL }
+};
+#undef ROFF
+
 /* refcount btree blocks */
 const field_t  refcbt_crc_hfld[] = {
        { "", FLDT_REFCBT_CRC, OI(0), C1, 0, TYP_NONE },
index 4168c9e2e15ac443c49a78e8340f7b482e34c37b..b4013ea8073ec62b7bdad927b3d059d6c94eacb2 100644 (file)
@@ -53,6 +53,11 @@ extern const struct field    rmapbt_crc_hfld[];
 extern const struct field      rmapbt_key_flds[];
 extern const struct field      rmapbt_rec_flds[];
 
+extern const struct field      rtrmapbt_crc_flds[];
+extern const struct field      rtrmapbt_crc_hfld[];
+extern const struct field      rtrmapbt_key_flds[];
+extern const struct field      rtrmapbt_rec_flds[];
+
 extern const struct field      refcbt_crc_flds[];
 extern const struct field      refcbt_crc_hfld[];
 extern const struct field      refcbt_key_flds[];
index ca0fe1826f9a805e23ab89065896036e8969a896..60c4e16d781f488d6d81d24613c8628336b548b0 100644 (file)
@@ -194,6 +194,17 @@ const ftattr_t     ftattrtab[] = {
        { FLDT_RMAPBTREC, "rmapbtrec", fp_sarray, (char *)rmapbt_rec_flds,
          SI(bitsz(struct xfs_rmap_rec)), 0, NULL, rmapbt_rec_flds },
 
+       { FLDT_RTRMAPBT_CRC, "rtrmapbt", NULL, (char *)rtrmapbt_crc_flds, btblock_size,
+         FTARG_SIZE, NULL, rtrmapbt_crc_flds },
+       { FLDT_RTRMAPBTKEY, "rtrmapbtkey", fp_sarray, (char *)rtrmapbt_key_flds,
+         SI(bitize(2 * sizeof(struct xfs_rmap_key))), 0, NULL, rtrmapbt_key_flds },
+       { FLDT_RTRMAPBTPTR, "rtrmapbtptr", fp_num, "%llu",
+         SI(bitsz(xfs_rtrmap_ptr_t)), 0, fa_dfsbno, NULL },
+       { FLDT_RTRMAPBTREC, "rtrmapbtrec", fp_sarray, (char *)rtrmapbt_rec_flds,
+         SI(bitsz(struct xfs_rmap_rec)), 0, NULL, rtrmapbt_rec_flds },
+       { FLDT_RTRMAPROOT, "rtrmaproot", NULL, (char *)rtrmaproot_flds, rtrmaproot_size,
+         FTARG_SIZE, NULL, rtrmaproot_flds },
+
        { FLDT_REFCBT_CRC, "refcntbt", NULL, (char *)refcbt_crc_flds, btblock_size,
          FTARG_SIZE, NULL, refcbt_crc_flds },
        { FLDT_REFCBTKEY, "refcntbtkey", fp_sarray, (char *)refcbt_key_flds,
index 1d7465b4d3e5621b8d0e0262cdc80bd86f75cd5c..67b6cb2a798719412ef69ec3554f93cd1f1384d0 100644 (file)
@@ -84,6 +84,11 @@ typedef enum fldt    {
        FLDT_RMAPBTKEY,
        FLDT_RMAPBTPTR,
        FLDT_RMAPBTREC,
+       FLDT_RTRMAPBT_CRC,
+       FLDT_RTRMAPBTKEY,
+       FLDT_RTRMAPBTPTR,
+       FLDT_RTRMAPBTREC,
+       FLDT_RTRMAPROOT,
        FLDT_REFCBT_CRC,
        FLDT_REFCBTKEY,
        FLDT_REFCBTPTR,
index 0a80b8d063603fc602761cf724750647bb0e6f6d..45368a3343a17a710839c851299cd5a70b532cf9 100644 (file)
@@ -48,6 +48,7 @@ static int    inode_u_muuid_count(void *obj, int startoff);
 static int     inode_u_sfdir2_count(void *obj, int startoff);
 static int     inode_u_sfdir3_count(void *obj, int startoff);
 static int     inode_u_symlink_count(void *obj, int startoff);
+static int     inode_u_rtrmapbt_count(void *obj, int startoff);
 
 static const cmdinfo_t inode_cmd =
        { "inode", NULL, inode_f, 0, 1, 1, "[inode#]",
@@ -233,6 +234,8 @@ const field_t       inode_u_flds[] = {
        { "sfdir3", FLDT_DIR3SF, NULL, inode_u_sfdir3_count, FLD_COUNT, TYP_NONE },
        { "symlink", FLDT_CHARNS, NULL, inode_u_symlink_count, FLD_COUNT,
          TYP_NONE },
+       { "rtrmapbt", FLDT_RTRMAPROOT, NULL, inode_u_rtrmapbt_count, FLD_COUNT,
+         TYP_NONE },
        { NULL }
 };
 
@@ -246,7 +249,7 @@ const field_t       inode_a_flds[] = {
 };
 
 static const char      *dinode_fmt_name[] =
-       { "dev", "local", "extents", "btree", "uuid" };
+       { "dev", "local", "extents", "btree", "uuid", "meta_btree" };
 static const int       dinode_fmt_name_size =
        sizeof(dinode_fmt_name) / sizeof(dinode_fmt_name[0]);
 
@@ -299,7 +302,7 @@ fp_dinode_fmt(
 
 static const char      *metatype_name[] =
        { "unknown", "dir", "usrquota", "grpquota", "prjquota", "rtbitmap",
-         "rtsummary"
+         "rtsummary", "rtrmap"
        };
 static const int       metatype_name_size = ARRAY_SIZE(metatype_name);
 
@@ -717,6 +720,8 @@ inode_next_type(void)
                                return TYP_RGBITMAP;
                        case XFS_METAFILE_RTSUMMARY:
                                return TYP_RGSUMMARY;
+                       case XFS_METAFILE_RTRMAP:
+                               return TYP_RTRMAPBT;
                        default:
                                return TYP_DATA;
                        }
@@ -866,6 +871,21 @@ inode_u_sfdir3_count(
               xfs_has_ftype(mp);
 }
 
+static int
+inode_u_rtrmapbt_count(
+       void                    *obj,
+       int                     startoff)
+{
+       struct xfs_dinode       *dip;
+
+       ASSERT(bitoffs(startoff) == 0);
+       ASSERT(obj == iocur_top->data);
+       dip = obj;
+       ASSERT((char *)XFS_DFORK_DPTR(dip) - (char *)dip == byteize(startoff));
+       return dip->di_format == XFS_DINODE_FMT_META_BTREE &&
+              dip->di_metatype == cpu_to_be16(XFS_METAFILE_RTRMAP);
+}
+
 int
 inode_u_size(
        void                    *obj,
index 2091b4ac8b139b4daa3548d161fbdf977b4da12a..1dfc33ffb44b87f975a3b6d859071c5d03cc389b 100644 (file)
--- a/db/type.c
+++ b/db/type.c
@@ -51,6 +51,7 @@ static const typ_t    __typtab[] = {
        { TYP_BNOBT, "bnobt", handle_struct, bnobt_hfld, NULL, TYP_F_NO_CRC_OFF },
        { TYP_CNTBT, "cntbt", handle_struct, cntbt_hfld, NULL, TYP_F_NO_CRC_OFF },
        { TYP_RMAPBT, NULL },
+       { TYP_RTRMAPBT, NULL },
        { TYP_REFCBT, NULL },
        { TYP_DATA, "data", handle_block, NULL, NULL, TYP_F_NO_CRC_OFF },
        { TYP_DIR2, "dir2", handle_struct, dir2_hfld, NULL, TYP_F_NO_CRC_OFF },
@@ -91,6 +92,8 @@ static const typ_t    __typtab_crc[] = {
                &xfs_cntbt_buf_ops, XFS_BTREE_SBLOCK_CRC_OFF },
        { TYP_RMAPBT, "rmapbt", handle_struct, rmapbt_crc_hfld,
                &xfs_rmapbt_buf_ops, XFS_BTREE_SBLOCK_CRC_OFF },
+       { TYP_RTRMAPBT, "rtrmapbt", handle_struct, rtrmapbt_crc_hfld,
+               &xfs_rtrmapbt_buf_ops, XFS_BTREE_LBLOCK_CRC_OFF },
        { TYP_REFCBT, "refcntbt", handle_struct, refcbt_crc_hfld,
                &xfs_refcountbt_buf_ops, XFS_BTREE_SBLOCK_CRC_OFF },
        { TYP_DATA, "data", handle_block, NULL, NULL, TYP_F_NO_CRC_OFF },
@@ -141,6 +144,8 @@ static const typ_t  __typtab_spcrc[] = {
                &xfs_cntbt_buf_ops, XFS_BTREE_SBLOCK_CRC_OFF },
        { TYP_RMAPBT, "rmapbt", handle_struct, rmapbt_crc_hfld,
                &xfs_rmapbt_buf_ops, XFS_BTREE_SBLOCK_CRC_OFF },
+       { TYP_RTRMAPBT, "rtrmapbt", handle_struct, rtrmapbt_crc_hfld,
+               &xfs_rtrmapbt_buf_ops, XFS_BTREE_LBLOCK_CRC_OFF },
        { TYP_REFCBT, "refcntbt", handle_struct, refcbt_crc_hfld,
                &xfs_refcountbt_buf_ops, XFS_BTREE_SBLOCK_CRC_OFF },
        { TYP_DATA, "data", handle_block, NULL, NULL, TYP_F_NO_CRC_OFF },
index e7f0ecc17680bf24d9fc1c04506063f690b6669b..c98f3640202e876ddd31b5b41c3ae0213a687964 100644 (file)
--- a/db/type.h
+++ b/db/type.h
@@ -20,6 +20,7 @@ typedef enum typnm
        TYP_BNOBT,
        TYP_CNTBT,
        TYP_RMAPBT,
+       TYP_RTRMAPBT,
        TYP_REFCBT,
        TYP_DATA,
        TYP_DIR2,
index 6b2dc7a30d25472c76bb943cf5f00d828dcebbf0..fcbcaaa11f1025ec8cee5dcef55c09c0bf737e4c 100644 (file)
 #define xfs_rtginode_name              libxfs_rtginode_name
 #define xfs_rtsummary_create           libxfs_rtsummary_create
 
+#define xfs_rtginode_load              libxfs_rtginode_load
+#define xfs_rtginode_load_parent       libxfs_rtginode_load_parent
 #define xfs_rtgroup_alloc              libxfs_rtgroup_alloc
 #define xfs_rtgroup_extents            libxfs_rtgroup_extents
 #define xfs_rtgroup_grab               libxfs_rtgroup_grab
 #define xfs_rtfree_extent              libxfs_rtfree_extent
 #define xfs_rtfree_blocks              libxfs_rtfree_blocks
 #define xfs_update_rtsb                        libxfs_update_rtsb
+#define xfs_rtrmapbt_droot_maxrecs     libxfs_rtrmapbt_droot_maxrecs
+#define xfs_rtrmapbt_maxrecs           libxfs_rtrmapbt_maxrecs
+
 #define xfs_sb_from_disk               libxfs_sb_from_disk
 #define xfs_sb_mount_rextsize          libxfs_sb_mount_rextsize
 #define xfs_sb_quota_from_disk         libxfs_sb_quota_from_disk
index acee900adbda505d8fe5b54c3bb195829b716d06..ddbabe36b5fc4149121c58d9f10fc01cca4f13df 100644 (file)
@@ -1302,7 +1302,7 @@ The possible data types are:
 .BR agf ", " agfl ", " agi ", " attr ", " bmapbta ", " bmapbtd ,
 .BR bnobt ", " cntbt ", " data ", " dir ", " dir2 ", " dqblk ,
 .BR inobt ", " inode ", " log ", " refcntbt ", " rmapbt ", " rtbitmap ,
-.BR rtsummary ", " sb ", " symlink " and " text .
+.BR rtsummary ", " sb ", " symlink ", " rtrmapbt ", and " text .
 See the TYPES section below for more information on these data types.
 .TP
 .BI "timelimit [" OPTIONS ]
@@ -2450,6 +2450,64 @@ block number within the allocation group to the next level in the Btree.
 .PD
 .RE
 .TP
+.B rtrmapbt
+There is one reverse mapping Btree for each realtime group.
+The
+.BR startblock " and "
+.B blockcount
+fields are 32 bits wide and record blocks within a realtime group.
+The root of this Btree is the reverse-mapping inode, which is recorded in the
+metadata directory.
+Blocks are linked to sibling left and right blocks at each level, as well as by
+pointers from parent to child blocks.
+Each block has the following fields:
+.RS 1.4i
+.PD 0
+.TP 1.2i
+.B magic
+RTRMAP block magic number, 0x4d415052 ('MAPR').
+.TP
+.B level
+level number of this block, 0 is a leaf.
+.TP
+.B numrecs
+number of data entries in the block.
+.TP
+.B leftsib
+left (logically lower) sibling block, 0 if none.
+.TP
+.B rightsib
+right (logically higher) sibling block, 0 if none.
+.TP
+.B recs
+[leaf blocks only] array of reference count records. Each record contains
+.BR startblock ,
+.BR blockcount ,
+.BR owner ,
+.BR offset ,
+.BR attr_fork ,
+.BR bmbt_block ,
+and
+.BR unwritten .
+.TP
+.B keys
+[non-leaf blocks only] array of double-key records. The first ("low") key
+contains the first value of each block in the level below this one. The second
+("high") key contains the largest key that can be used to identify any record
+in the subtree. Each record contains
+.BR startblock ,
+.BR owner ,
+.BR offset ,
+.BR attr_fork ,
+and
+.BR bmbt_block .
+.TP
+.B ptrs
+[non-leaf blocks only] array of child block pointers. Each pointer is a
+block number within the allocation group to the next level in the Btree.
+.PD
+.RE
+.TP
 .B rtbitmap
 If the filesystem has a realtime subvolume, then the
 .B rbmino