]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs: use metadir for quota inodes
authorDarrick J. Wong <djwong@kernel.org>
Mon, 25 Nov 2024 21:14:26 +0000 (13:14 -0800)
committerDarrick J. Wong <djwong@kernel.org>
Tue, 24 Dec 2024 02:01:31 +0000 (18:01 -0800)
Source kernel commit: e80fbe1ad8eff7d7d1363e14f1e493d84dd37c84

Store the quota inodes in the /quota metadata directory if metadir is
enabled.  This enables us to stop using the sb_[ugp]uotino fields in the
superblock.  From this point on, all metadata files will be children of
the metadata directory tree root.

Signed-off-by: Darrick J. Wong <djwong@kernel.org>
Reviewed-by: Christoph Hellwig <hch@lst.de>
libxfs/xfs_dquot_buf.c
libxfs/xfs_quota_defs.h
libxfs/xfs_sb.c

index db603cab92138fc5120602ea21d9fc97c343601c..599d03ac960b7ba0783a220898d10e0db5d19b22 100644 (file)
@@ -14,6 +14,9 @@
 #include "xfs_quota_defs.h"
 #include "xfs_inode.h"
 #include "xfs_trans.h"
+#include "xfs_health.h"
+#include "xfs_metadir.h"
+#include "xfs_metafile.h"
 
 int
 xfs_calc_dquots_per_chunk(
@@ -321,3 +324,190 @@ xfs_dquot_to_disk_ts(
 
        return cpu_to_be32(t);
 }
+
+inline unsigned int
+xfs_dqinode_sick_mask(xfs_dqtype_t type)
+{
+       switch (type) {
+       case XFS_DQTYPE_USER:
+               return XFS_SICK_FS_UQUOTA;
+       case XFS_DQTYPE_GROUP:
+               return XFS_SICK_FS_GQUOTA;
+       case XFS_DQTYPE_PROJ:
+               return XFS_SICK_FS_PQUOTA;
+       }
+
+       ASSERT(0);
+       return 0;
+}
+
+/*
+ * Load the inode for a given type of quota, assuming that the sb fields have
+ * been sorted out.  This is not true when switching quota types on a V4
+ * filesystem, so do not use this function for that.  If metadir is enabled,
+ * @dp must be the /quota metadir.
+ *
+ * Returns -ENOENT if the quota inode field is NULLFSINO; 0 and an inode on
+ * success; or a negative errno.
+ */
+int
+xfs_dqinode_load(
+       struct xfs_trans        *tp,
+       struct xfs_inode        *dp,
+       xfs_dqtype_t            type,
+       struct xfs_inode        **ipp)
+{
+       struct xfs_mount        *mp = tp->t_mountp;
+       struct xfs_inode        *ip;
+       enum xfs_metafile_type  metafile_type = xfs_dqinode_metafile_type(type);
+       int                     error;
+
+       if (!xfs_has_metadir(mp)) {
+               xfs_ino_t       ino;
+
+               switch (type) {
+               case XFS_DQTYPE_USER:
+                       ino = mp->m_sb.sb_uquotino;
+                       break;
+               case XFS_DQTYPE_GROUP:
+                       ino = mp->m_sb.sb_gquotino;
+                       break;
+               case XFS_DQTYPE_PROJ:
+                       ino = mp->m_sb.sb_pquotino;
+                       break;
+               default:
+                       ASSERT(0);
+                       return -EFSCORRUPTED;
+               }
+
+               /* Should have set 0 to NULLFSINO when loading superblock */
+               if (ino == NULLFSINO)
+                       return -ENOENT;
+
+               error = xfs_trans_metafile_iget(tp, ino, metafile_type, &ip);
+       } else {
+               error = xfs_metadir_load(tp, dp, xfs_dqinode_path(type),
+                               metafile_type, &ip);
+               if (error == -ENOENT)
+                       return error;
+       }
+       if (error) {
+               if (xfs_metadata_is_sick(error))
+                       xfs_fs_mark_sick(mp, xfs_dqinode_sick_mask(type));
+               return error;
+       }
+
+       if (XFS_IS_CORRUPT(mp, ip->i_df.if_format != XFS_DINODE_FMT_EXTENTS &&
+                              ip->i_df.if_format != XFS_DINODE_FMT_BTREE)) {
+               xfs_irele(ip);
+               xfs_fs_mark_sick(mp, xfs_dqinode_sick_mask(type));
+               return -EFSCORRUPTED;
+       }
+
+       if (XFS_IS_CORRUPT(mp, ip->i_projid != 0)) {
+               xfs_irele(ip);
+               xfs_fs_mark_sick(mp, xfs_dqinode_sick_mask(type));
+               return -EFSCORRUPTED;
+       }
+
+       *ipp = ip;
+       return 0;
+}
+
+/* Create a metadata directory quota inode. */
+int
+xfs_dqinode_metadir_create(
+       struct xfs_inode                *dp,
+       xfs_dqtype_t                    type,
+       struct xfs_inode                **ipp)
+{
+       struct xfs_metadir_update       upd = {
+               .dp                     = dp,
+               .metafile_type          = xfs_dqinode_metafile_type(type),
+               .path                   = xfs_dqinode_path(type),
+       };
+       int                             error;
+
+       error = xfs_metadir_start_create(&upd);
+       if (error)
+               return error;
+
+       error = xfs_metadir_create(&upd, S_IFREG);
+       if (error)
+               return error;
+
+       xfs_trans_log_inode(upd.tp, upd.ip, XFS_ILOG_CORE);
+
+       error = xfs_metadir_commit(&upd);
+       if (error)
+               return error;
+
+       xfs_finish_inode_setup(upd.ip);
+       *ipp = upd.ip;
+       return 0;
+}
+
+#ifndef __KERNEL__
+/* Link a metadata directory quota inode. */
+int
+xfs_dqinode_metadir_link(
+       struct xfs_inode                *dp,
+       xfs_dqtype_t                    type,
+       struct xfs_inode                *ip)
+{
+       struct xfs_metadir_update       upd = {
+               .dp                     = dp,
+               .metafile_type          = xfs_dqinode_metafile_type(type),
+               .path                   = xfs_dqinode_path(type),
+               .ip                     = ip,
+       };
+       int                             error;
+
+       error = xfs_metadir_start_link(&upd);
+       if (error)
+               return error;
+
+       error = xfs_metadir_link(&upd);
+       if (error)
+               return error;
+
+       xfs_trans_log_inode(upd.tp, upd.ip, XFS_ILOG_CORE);
+
+       return xfs_metadir_commit(&upd);
+}
+#endif /* __KERNEL__ */
+
+/* Create the parent directory for all quota inodes and load it. */
+int
+xfs_dqinode_mkdir_parent(
+       struct xfs_mount        *mp,
+       struct xfs_inode        **dpp)
+{
+       if (!mp->m_metadirip) {
+               xfs_fs_mark_sick(mp, XFS_SICK_FS_METADIR);
+               return -EFSCORRUPTED;
+       }
+
+       return xfs_metadir_mkdir(mp->m_metadirip, "quota", dpp);
+}
+
+/*
+ * Load the parent directory of all quota inodes.  Pass the inode to the caller
+ * because quota functions (e.g. QUOTARM) can be called on the quota files even
+ * if quotas are not enabled.
+ */
+int
+xfs_dqinode_load_parent(
+       struct xfs_trans        *tp,
+       struct xfs_inode        **dpp)
+{
+       struct xfs_mount        *mp = tp->t_mountp;
+
+       if (!mp->m_metadirip) {
+               xfs_fs_mark_sick(mp, XFS_SICK_FS_METADIR);
+               return -EFSCORRUPTED;
+       }
+
+       return xfs_metadir_load(tp, mp->m_metadirip, "quota", XFS_METAFILE_DIR,
+                       dpp);
+}
index fb05f44f6c754ac214145d8690a7e42ebc549ece..763d941a8420c5aefb5bf175edba737e80ee76cd 100644 (file)
@@ -143,4 +143,47 @@ time64_t xfs_dquot_from_disk_ts(struct xfs_disk_dquot *ddq,
                __be32 dtimer);
 __be32 xfs_dquot_to_disk_ts(struct xfs_dquot *ddq, time64_t timer);
 
+static inline const char *
+xfs_dqinode_path(xfs_dqtype_t type)
+{
+       switch (type) {
+       case XFS_DQTYPE_USER:
+               return "user";
+       case XFS_DQTYPE_GROUP:
+               return "group";
+       case XFS_DQTYPE_PROJ:
+               return "project";
+       }
+
+       ASSERT(0);
+       return NULL;
+}
+
+static inline enum xfs_metafile_type
+xfs_dqinode_metafile_type(xfs_dqtype_t type)
+{
+       switch (type) {
+       case XFS_DQTYPE_USER:
+               return XFS_METAFILE_USRQUOTA;
+       case XFS_DQTYPE_GROUP:
+               return XFS_METAFILE_GRPQUOTA;
+       case XFS_DQTYPE_PROJ:
+               return XFS_METAFILE_PRJQUOTA;
+       }
+
+       ASSERT(0);
+       return XFS_METAFILE_UNKNOWN;
+}
+
+unsigned int xfs_dqinode_sick_mask(xfs_dqtype_t type);
+
+int xfs_dqinode_load(struct xfs_trans *tp, struct xfs_inode *dp,
+               xfs_dqtype_t type, struct xfs_inode **ipp);
+int xfs_dqinode_metadir_create(struct xfs_inode *dp, xfs_dqtype_t type,
+               struct xfs_inode **ipp);
+int xfs_dqinode_metadir_link(struct xfs_inode *dp, xfs_dqtype_t type,
+               struct xfs_inode *ip);
+int xfs_dqinode_mkdir_parent(struct xfs_mount *mp, struct xfs_inode **dpp);
+int xfs_dqinode_load_parent(struct xfs_trans *tp, struct xfs_inode **dpp);
+
 #endif /* __XFS_QUOTA_H__ */
index 2ab234f1dfce70940b0357e1b9bd6f64a9d168fb..375324b99261af09eef6563983e8d540f5f8d30f 100644 (file)
@@ -855,6 +855,7 @@ xfs_sb_quota_to_disk(
 
        if (xfs_sb_is_v5(from) &&
            (from->sb_features_incompat & XFS_SB_FEAT_INCOMPAT_METADIR)) {
+               to->sb_qflags = cpu_to_be16(from->sb_qflags);
                to->sb_uquotino = cpu_to_be64(0);
                to->sb_gquotino = cpu_to_be64(0);
                to->sb_pquotino = cpu_to_be64(0);