]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs: reflect sb features in xfs_mount
authorDave Chinner <dchinner@redhat.com>
Mon, 31 Jan 2022 20:25:48 +0000 (15:25 -0500)
committerEric Sandeen <sandeen@redhat.com>
Mon, 31 Jan 2022 20:25:48 +0000 (15:25 -0500)
Source kernel commit: a1d86e8dec8c1325d301c9d5594bb794bc428fc3

Currently on-disk feature checks require decoding the superblock
fileds and so can be non-trivial. We have almost 400 hundred
individual feature checks in the XFS code, so this is a significant
amount of code. To reduce runtime check overhead, pre-process all
the version flags into a features field in the xfs_mount at mount
time so we can convert all the feature checks to a simple flag
check.

There is also a need to convert the dynamic feature flags to update
the m_features field. This is required for attr, attr2 and quota
features. New xfs_mount based wrappers are added for this.

Signed-off-by: Dave Chinner <dchinner@redhat.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Reviewed-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Darrick J. Wong <djwong@kernel.org>
Signed-off-by: Eric Sandeen <sandeen@sandeen.net>
db/sb.c
include/xfs_mount.h
libxfs/init.c
libxfs/libxfs_api_defs.h
libxfs/xfs_format.h
libxfs/xfs_sb.c
libxfs/xfs_sb.h
logprint/logprint.c
mkfs/xfs_mkfs.c
repair/phase2.c
repair/versions.c

diff --git a/db/sb.c b/db/sb.c
index b4c14276100e799c99eac0b8082b4d3c33c4cd11..7909acae85f3d0cc3ecc77a7bc0fb11eba96a715 100644 (file)
--- a/db/sb.c
+++ b/db/sb.c
@@ -720,6 +720,7 @@ version_f(
 {
        uint16_t        version = 0;
        uint32_t        features = 0;
+       unsigned long   old_mfeatures = 0;
        xfs_agnumber_t  ag;
 
        if (argc == 2) {        /* WRITE VERSION */
@@ -802,7 +803,7 @@ version_f(
                        version = mp->m_sb.sb_versionnum;
                        features = mp->m_sb.sb_features2;
                } else if (!strcasecmp(argv[1], "projid32bit")) {
-                       xfs_sb_version_addprojid32bit(&mp->m_sb);
+                       xfs_sb_version_addprojid32(&mp->m_sb);
                        version = mp->m_sb.sb_versionnum;
                        features = mp->m_sb.sb_features2;
                } else {
@@ -821,6 +822,8 @@ version_f(
                                }
                        mp->m_sb.sb_versionnum = version;
                        mp->m_sb.sb_features2 = features;
+                       mp->m_features &= ~XFS_FEAT_ATTR2;
+                       mp->m_features |= libxfs_sb_version_to_features(&mp->m_sb);
                }
        }
 
@@ -831,6 +834,8 @@ version_f(
                features = mp->m_sb.sb_features2;
                mp->m_sb.sb_versionnum = strtoul(argv[1], &sp, 0);
                mp->m_sb.sb_features2 = strtoul(argv[2], &sp, 0);
+               old_mfeatures = mp->m_features;
+               mp->m_features = libxfs_sb_version_to_features(&mp->m_sb);
        }
 
        dbprintf(_("versionnum [0x%x+0x%x] = %s\n"), mp->m_sb.sb_versionnum,
@@ -839,6 +844,7 @@ version_f(
        if (argc == 3) {        /* now reset... */
                mp->m_sb.sb_versionnum = version;
                mp->m_sb.sb_features2 = features;
+               mp->m_features = old_mfeatures;
                return 0;
        }
 
index cc4682e2d9d5875d8d28e680b18cfb79857533ff..a995140d265ed0edf0afe5694edd105a15493bd9 100644 (file)
@@ -80,6 +80,7 @@ typedef struct xfs_mount {
        uint                    m_ag_max_usable; /* max space per AG */
        struct radix_tree_root  m_perag_tree;
        uint                    m_flags;        /* global mount flags */
+       uint64_t                m_features;     /* active filesystem features */
        bool                    m_finobt_nores; /* no per-AG finobt resv. */
        uint                    m_qflags;       /* quota status flags */
        uint                    m_attroffset;   /* inode attribute offset */
@@ -120,6 +121,82 @@ typedef struct xfs_mount {
 
 #define M_IGEO(mp)             (&(mp)->m_ino_geo)
 
+/*
+ * Flags for m_features.
+ *
+ * These are all the active features in the filesystem, regardless of how
+ * they are configured.
+ */
+#define XFS_FEAT_ATTR          (1ULL << 0)     /* xattrs present in fs */
+#define XFS_FEAT_NLINK         (1ULL << 1)     /* 32 bit link counts */
+#define XFS_FEAT_QUOTA         (1ULL << 2)     /* quota active */
+#define XFS_FEAT_ALIGN         (1ULL << 3)     /* inode alignment */
+#define XFS_FEAT_DALIGN                (1ULL << 4)     /* data alignment */
+#define XFS_FEAT_LOGV2         (1ULL << 5)     /* version 2 logs */
+#define XFS_FEAT_SECTOR                (1ULL << 6)     /* sector size > 512 bytes */
+#define XFS_FEAT_EXTFLG                (1ULL << 7)     /* unwritten extents */
+#define XFS_FEAT_ASCIICI       (1ULL << 8)     /* ASCII only case-insens. */
+#define XFS_FEAT_LAZYSBCOUNT   (1ULL << 9)     /* Superblk counters */
+#define XFS_FEAT_ATTR2         (1ULL << 10)    /* dynamic attr fork */
+#define XFS_FEAT_PARENT                (1ULL << 11)    /* parent pointers */
+#define XFS_FEAT_PROJID32      (1ULL << 12)    /* 32 bit project id */
+#define XFS_FEAT_CRC           (1ULL << 13)    /* metadata CRCs */
+#define XFS_FEAT_V3INODES      (1ULL << 14)    /* Version 3 inodes */
+#define XFS_FEAT_PQUOTINO      (1ULL << 15)    /* non-shared proj/grp quotas */
+#define XFS_FEAT_FTYPE         (1ULL << 16)    /* inode type in dir */
+#define XFS_FEAT_FINOBT                (1ULL << 17)    /* free inode btree */
+#define XFS_FEAT_RMAPBT                (1ULL << 18)    /* reverse map btree */
+#define XFS_FEAT_REFLINK       (1ULL << 19)    /* reflinked files */
+#define XFS_FEAT_SPINODES      (1ULL << 20)    /* sparse inode chunks */
+#define XFS_FEAT_META_UUID     (1ULL << 21)    /* metadata UUID */
+#define XFS_FEAT_REALTIME      (1ULL << 22)    /* realtime device present */
+#define XFS_FEAT_INOBTCNT      (1ULL << 23)    /* inobt block counts */
+#define XFS_FEAT_BIGTIME       (1ULL << 24)    /* large timestamps */
+#define XFS_FEAT_NEEDSREPAIR   (1ULL << 25)    /* needs xfs_repair */
+
+#define __XFS_HAS_FEAT(name, NAME) \
+static inline bool xfs_has_ ## name (struct xfs_mount *mp) \
+{ \
+       return mp->m_features & XFS_FEAT_ ## NAME; \
+}
+
+/* Some features can be added dynamically so they need a set wrapper, too. */
+#define __XFS_ADD_FEAT(name, NAME) \
+       __XFS_HAS_FEAT(name, NAME); \
+static inline void xfs_add_ ## name (struct xfs_mount *mp) \
+{ \
+       mp->m_features |= XFS_FEAT_ ## NAME; \
+       xfs_sb_version_add ## name(&mp->m_sb); \
+}
+
+/* Superblock features */
+__XFS_ADD_FEAT(attr, ATTR)
+__XFS_HAS_FEAT(nlink, NLINK)
+__XFS_ADD_FEAT(quota, QUOTA)
+__XFS_HAS_FEAT(align, ALIGN)
+__XFS_HAS_FEAT(dalign, DALIGN)
+__XFS_HAS_FEAT(logv2, LOGV2)
+__XFS_HAS_FEAT(sector, SECTOR)
+__XFS_HAS_FEAT(extflg, EXTFLG)
+__XFS_HAS_FEAT(asciici, ASCIICI)
+__XFS_HAS_FEAT(lazysbcount, LAZYSBCOUNT)
+__XFS_ADD_FEAT(attr2, ATTR2)
+__XFS_HAS_FEAT(parent, PARENT)
+__XFS_ADD_FEAT(projid32, PROJID32)
+__XFS_HAS_FEAT(crc, CRC)
+__XFS_HAS_FEAT(v3inodes, V3INODES)
+__XFS_HAS_FEAT(pquotino, PQUOTINO)
+__XFS_HAS_FEAT(ftype, FTYPE)
+__XFS_HAS_FEAT(finobt, FINOBT)
+__XFS_HAS_FEAT(rmapbt, RMAPBT)
+__XFS_HAS_FEAT(reflink, REFLINK)
+__XFS_HAS_FEAT(sparseinodes, SPINODES)
+__XFS_HAS_FEAT(metauuid, META_UUID)
+__XFS_HAS_FEAT(realtime, REALTIME)
+__XFS_HAS_FEAT(inobtcounts, INOBTCNT)
+__XFS_HAS_FEAT(bigtime, BIGTIME)
+__XFS_HAS_FEAT(needsrepair, NEEDSREPAIR)
+
 #define LIBXFS_MOUNT_DEBUGGER          0x0001
 #define LIBXFS_MOUNT_32BITINODES       0x0002
 #define LIBXFS_MOUNT_32BITINOOPT       0x0004
index 9d8f20280dafbde37c658da0cf0b94f1dc71c428..7d94b721e8d33646064518aba7ce98cad29e81af 100644 (file)
@@ -723,6 +723,7 @@ libxfs_mount(
        bool                    debugger = (flags & LIBXFS_MOUNT_DEBUGGER);
        int                     error;
 
+       mp->m_features = xfs_sb_version_to_features(sb);
        libxfs_buftarg_init(mp, dev, logdev, rtdev);
 
        mp->m_finobt_nores = true;
index b76e638098bbb26bde5a7ca97d404fc1daad033c..a086fca23586192c2a91d6bb80f0aca1d57ddc4d 100644 (file)
 #define xfs_sb_quota_from_disk         libxfs_sb_quota_from_disk
 #define xfs_sb_read_secondary          libxfs_sb_read_secondary
 #define xfs_sb_to_disk                 libxfs_sb_to_disk
+#define xfs_sb_version_to_features     libxfs_sb_version_to_features
 #define xfs_symlink_blocks             libxfs_symlink_blocks
 #define xfs_symlink_hdr_ok             libxfs_symlink_hdr_ok
 
index ac739e6a921e99d6cded56dad893e29aec4dd5e0..fdd35202f92c9cd57a9b1cbbdf4eec9a42724f49 100644 (file)
@@ -405,7 +405,7 @@ static inline bool xfs_sb_version_hasprojid32bit(struct xfs_sb *sbp)
                (sbp->sb_features2 & XFS_SB_VERSION2_PROJID32BIT));
 }
 
-static inline void xfs_sb_version_addprojid32bit(struct xfs_sb *sbp)
+static inline void xfs_sb_version_addprojid32(struct xfs_sb *sbp)
 {
        sbp->sb_versionnum |= XFS_SB_VERSION_MOREBITSBIT;
        sbp->sb_features2 |= XFS_SB_VERSION2_PROJID32BIT;
index f29a59ae807b283790497a47d32808da89c53cf9..100dd87dae963be6f21c4d5035b03e93b8fd6ea0 100644 (file)
  * Physical superblock buffer manipulations. Shared with libxfs in userspace.
  */
 
+uint64_t
+xfs_sb_version_to_features(
+       struct xfs_sb   *sbp)
+{
+       uint64_t        features = 0;
+
+       /* optional V4 features */
+       if (sbp->sb_rblocks > 0)
+               features |= XFS_FEAT_REALTIME;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_ATTRBIT)
+               features |= XFS_FEAT_ATTR;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_QUOTABIT)
+               features |= XFS_FEAT_QUOTA;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_ALIGNBIT)
+               features |= XFS_FEAT_ALIGN;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_LOGV2BIT)
+               features |= XFS_FEAT_LOGV2;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_DALIGNBIT)
+               features |= XFS_FEAT_DALIGN;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_EXTFLGBIT)
+               features |= XFS_FEAT_EXTFLG;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_SECTORBIT)
+               features |= XFS_FEAT_SECTOR;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_BORGBIT)
+               features |= XFS_FEAT_ASCIICI;
+       if (sbp->sb_versionnum & XFS_SB_VERSION_MOREBITSBIT) {
+               if (sbp->sb_features2 & XFS_SB_VERSION2_LAZYSBCOUNTBIT)
+                       features |= XFS_FEAT_LAZYSBCOUNT;
+               if (sbp->sb_features2 & XFS_SB_VERSION2_ATTR2BIT)
+                       features |= XFS_FEAT_ATTR2;
+               if (sbp->sb_features2 & XFS_SB_VERSION2_PROJID32BIT)
+                       features |= XFS_FEAT_PROJID32;
+               if (sbp->sb_features2 & XFS_SB_VERSION2_FTYPE)
+                       features |= XFS_FEAT_FTYPE;
+       }
+
+       if (XFS_SB_VERSION_NUM(sbp) != XFS_SB_VERSION_5)
+               return features;
+
+       /* Always on V5 features */
+       features |= XFS_FEAT_ALIGN | XFS_FEAT_LOGV2 | XFS_FEAT_EXTFLG |
+                   XFS_FEAT_LAZYSBCOUNT | XFS_FEAT_ATTR2 | XFS_FEAT_PROJID32 |
+                   XFS_FEAT_V3INODES | XFS_FEAT_CRC | XFS_FEAT_PQUOTINO;
+
+       /* Optional V5 features */
+       if (sbp->sb_features_ro_compat & XFS_SB_FEAT_RO_COMPAT_FINOBT)
+               features |= XFS_FEAT_FINOBT;
+       if (sbp->sb_features_ro_compat & XFS_SB_FEAT_RO_COMPAT_RMAPBT)
+               features |= XFS_FEAT_RMAPBT;
+       if (sbp->sb_features_ro_compat & XFS_SB_FEAT_RO_COMPAT_REFLINK)
+               features |= XFS_FEAT_REFLINK;
+       if (sbp->sb_features_ro_compat & XFS_SB_FEAT_RO_COMPAT_INOBTCNT)
+               features |= XFS_FEAT_INOBTCNT;
+       if (sbp->sb_features_incompat & XFS_SB_FEAT_INCOMPAT_FTYPE)
+               features |= XFS_FEAT_FTYPE;
+       if (sbp->sb_features_incompat & XFS_SB_FEAT_INCOMPAT_SPINODES)
+               features |= XFS_FEAT_SPINODES;
+       if (sbp->sb_features_incompat & XFS_SB_FEAT_INCOMPAT_META_UUID)
+               features |= XFS_FEAT_META_UUID;
+       if (sbp->sb_features_incompat & XFS_SB_FEAT_INCOMPAT_BIGTIME)
+               features |= XFS_FEAT_BIGTIME;
+       if (sbp->sb_features_incompat & XFS_SB_FEAT_INCOMPAT_NEEDSREPAIR)
+               features |= XFS_FEAT_NEEDSREPAIR;
+       return features;
+}
+
 /* Check all the superblock fields we care about when reading one in. */
 STATIC int
 xfs_validate_sb_read(
index 0c1602d9b53d71f0faee868b95f3a0e5a468796f..d2dd99cb6921cf2a64c9f036e6ce4c91e9b7c08a 100644 (file)
@@ -20,6 +20,7 @@ extern void   xfs_sb_mount_common(struct xfs_mount *mp, struct xfs_sb *sbp);
 extern void    xfs_sb_from_disk(struct xfs_sb *to, struct xfs_dsb *from);
 extern void    xfs_sb_to_disk(struct xfs_dsb *to, struct xfs_sb *from);
 extern void    xfs_sb_quota_from_disk(struct xfs_sb *sbp);
+extern uint64_t        xfs_sb_version_to_features(struct xfs_sb *sbp);
 
 extern int     xfs_update_secondary_sbs(struct xfs_mount *mp);
 
index 18adf1022edaa3da098149c15dbb15b9619a9ecf..430961ff718ea00e34bf864e3d493fa262053457 100644 (file)
@@ -80,6 +80,7 @@ logstat(xfs_mount_t *mp)
                 */
                sb = &mp->m_sb;
                libxfs_sb_from_disk(sb, (xfs_dsb_t *)buf);
+               mp->m_features |= libxfs_sb_version_to_features(&mp->m_sb);
                mp->m_blkbb_log = sb->sb_blocklog - BBSHIFT;
 
                x.logBBsize = XFS_FSB_TO_BB(mp, sb->sb_logblocks);
index b8c11ce9c0071f88fd2e0513161d0beb1b048197..2340b7b134cd0c8097e91c61e26f9d1a874a261e 100644 (file)
@@ -3425,6 +3425,7 @@ start_superblock_setup(
        /* log reservation calculations depend on rt geometry */
        sbp->sb_rblocks = cfg->rtblocks;
        sbp->sb_rextsize = cfg->rtextblocks;
+       mp->m_features |= libxfs_sb_version_to_features(sbp);
 }
 
 static void
@@ -3484,6 +3485,7 @@ finish_superblock_setup(
        sbp->sb_qflags = 0;
        sbp->sb_unit = cfg->dsunit;
        sbp->sb_width = cfg->dswidth;
+       mp->m_features |= libxfs_sb_version_to_features(sbp);
 
 }
 
index 32ffe18b301d9ff44c70aad602e6c7a438b5df71..f13f785b31b69f8cfb802ad8aba5554b322e8e56 100644 (file)
@@ -192,8 +192,11 @@ upgrade_filesystem(
                dirty |= set_inobtcount(mp);
        if (add_bigtime)
                dirty |= set_bigtime(mp);
+       if (!dirty)
+               return;
 
-        if (no_modify || !dirty)
+       mp->m_features |= libxfs_sb_version_to_features(&mp->m_sb);
+        if (no_modify)
                 return;
 
         bp = libxfs_getsb(mp);
index 7f268f6117cd23e67eed07c1c601c94749e8b23d..0b5376d74de463068a84c6428a616289ecfaa85c 100644 (file)
@@ -82,6 +82,9 @@ update_sb_version(
 
        if (!fs_aligned_inodes && xfs_sb_version_hasalign(&mp->m_sb))
                mp->m_sb.sb_versionnum &= ~XFS_SB_VERSION_ALIGNBIT;
+
+       mp->m_features &= ~(XFS_FEAT_QUOTA | XFS_FEAT_ALIGN);
+       mp->m_features |= libxfs_sb_version_to_features(&mp->m_sb);
 }
 
 /*