]> git.ipfire.org Git - thirdparty/xfsprogs-dev.git/commitdiff
xfs: support creating per-RTG files in growfs
authorChristoph Hellwig <hch@lst.de>
Mon, 25 Nov 2024 21:14:20 +0000 (13:14 -0800)
committerDarrick J. Wong <djwong@kernel.org>
Tue, 24 Dec 2024 02:01:29 +0000 (18:01 -0800)
Source kernel commit: ae897e0bed0f5461a6b1c3259c7d899759ba2a62

To support adding new RT groups in growfs, we need to be able to create
the per-RT group files.  Add a new xfs_rtginode_create helper to create
a given per-RTG file.  Most of the code for that is shared, but the
details of the actual file are abstracted out using a new create method
in struct xfs_rtginode_ops.

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

index e4d0646ba19bfd2b56f0728840426db8315c71f9..c686cd5309e87c5d98b35ed5a0ff1de7502f488b 100644 (file)
@@ -1295,3 +1295,35 @@ xfs_rtfile_initialize_blocks(
 
        return 0;
 }
+
+int
+xfs_rtbitmap_create(
+       struct xfs_rtgroup      *rtg,
+       struct xfs_inode        *ip,
+       struct xfs_trans        *tp,
+       bool                    init)
+{
+       struct xfs_mount        *mp = rtg_mount(rtg);
+
+       ip->i_disk_size = mp->m_sb.sb_rbmblocks * mp->m_sb.sb_blocksize;
+       if (init && !xfs_has_rtgroups(mp)) {
+               ip->i_diflags |= XFS_DIFLAG_NEWRTBM;
+               inode_set_atime(VFS_I(ip), 0, 0);
+       }
+       xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
+       return 0;
+}
+
+int
+xfs_rtsummary_create(
+       struct xfs_rtgroup      *rtg,
+       struct xfs_inode        *ip,
+       struct xfs_trans        *tp,
+       bool                    init)
+{
+       struct xfs_mount        *mp = rtg_mount(rtg);
+
+       ip->i_disk_size = mp->m_rsumblocks * mp->m_sb.sb_blocksize;
+       xfs_trans_log_inode(tp, ip, XFS_ILOG_CORE);
+       return 0;
+}
index b3cbc56aa255ed44829f09da472bed848295f0c8..e4994a3e461d33810fab4376165684f2cbf9c1f1 100644 (file)
@@ -315,6 +315,10 @@ xfs_filblks_t xfs_rtsummary_blockcount(struct xfs_mount *mp,
 int xfs_rtfile_initialize_blocks(struct xfs_rtgroup *rtg,
                enum xfs_rtg_inodes type, xfs_fileoff_t offset_fsb,
                xfs_fileoff_t end_fsb, void *data);
+int xfs_rtbitmap_create(struct xfs_rtgroup *rtg, struct xfs_inode *ip,
+               struct xfs_trans *tp, bool init);
+int xfs_rtsummary_create(struct xfs_rtgroup *rtg, struct xfs_inode *ip,
+               struct xfs_trans *tp, bool init);
 
 #else /* CONFIG_XFS_RT */
 # define xfs_rtfree_extent(t,b,l)                      (-ENOSYS)
index 41c794718e06c9f8b25a4a34f70b4871cd726dd8..4f2e6be2ae48cc120d88b49cac3afb7cde12d11e 100644 (file)
@@ -270,16 +270,24 @@ struct xfs_rtginode_ops {
 
        /* Does the fs have this feature? */
        bool                    (*enabled)(struct xfs_mount *mp);
+
+       /* Create this rtgroup metadata inode and initialize it. */
+       int                     (*create)(struct xfs_rtgroup *rtg,
+                                         struct xfs_inode *ip,
+                                         struct xfs_trans *tp,
+                                         bool init);
 };
 
 static const struct xfs_rtginode_ops xfs_rtginode_ops[XFS_RTGI_MAX] = {
        [XFS_RTGI_BITMAP] = {
                .name           = "bitmap",
                .metafile_type  = XFS_METAFILE_RTBITMAP,
+               .create         = xfs_rtbitmap_create,
        },
        [XFS_RTGI_SUMMARY] = {
                .name           = "summary",
                .metafile_type  = XFS_METAFILE_RTSUMMARY,
+               .create         = xfs_rtsummary_create,
        },
 };
 
@@ -387,6 +395,67 @@ xfs_rtginode_irele(
        *ipp = NULL;
 }
 
+/* Add a metadata inode for a realtime rmap btree. */
+int
+xfs_rtginode_create(
+       struct xfs_rtgroup              *rtg,
+       enum xfs_rtg_inodes             type,
+       bool                            init)
+{
+       const struct xfs_rtginode_ops   *ops = &xfs_rtginode_ops[type];
+       struct xfs_mount                *mp = rtg_mount(rtg);
+       struct xfs_metadir_update       upd = {
+               .dp                     = mp->m_rtdirip,
+               .metafile_type          = ops->metafile_type,
+       };
+       int                             error;
+
+       if (!xfs_rtginode_enabled(rtg, type))
+               return 0;
+
+       if (!mp->m_rtdirip)
+               return -EFSCORRUPTED;
+
+       upd.path = xfs_rtginode_path(rtg_rgno(rtg), type);
+       if (!upd.path)
+               return -ENOMEM;
+
+       error = xfs_metadir_start_create(&upd);
+       if (error)
+               goto out_path;
+
+       error = xfs_metadir_create(&upd, S_IFREG);
+       if (error)
+               return error;
+
+       xfs_rtginode_lockdep_setup(upd.ip, rtg_rgno(rtg), type);
+
+       upd.ip->i_projid = rtg_rgno(rtg);
+       error = ops->create(rtg, upd.ip, upd.tp, init);
+       if (error)
+               goto out_cancel;
+
+       error = xfs_metadir_commit(&upd);
+       if (error)
+               goto out_path;
+
+       kfree(upd.path);
+       xfs_finish_inode_setup(upd.ip);
+       rtg->rtg_inodes[type] = upd.ip;
+       return 0;
+
+out_cancel:
+       xfs_metadir_cancel(&upd, error);
+       /* Have to finish setting up the inode to ensure it's deleted. */
+       if (upd.ip) {
+               xfs_finish_inode_setup(upd.ip);
+               xfs_irele(upd.ip);
+       }
+out_path:
+       kfree(upd.path);
+       return error;
+}
+
 /* Create the parent directory for all rtgroup inodes and load it. */
 int
 xfs_rtginode_mkdir_parent(
index 3732f65ba8a1f65690a04fca034055ece12e3162..6ccf31bb6bc7a7a643bb6ea05cd7403b0d72c013 100644 (file)
@@ -242,6 +242,8 @@ enum xfs_metafile_type xfs_rtginode_metafile_type(enum xfs_rtg_inodes type);
 bool xfs_rtginode_enabled(struct xfs_rtgroup *rtg, enum xfs_rtg_inodes type);
 int xfs_rtginode_load(struct xfs_rtgroup *rtg, enum xfs_rtg_inodes type,
                struct xfs_trans *tp);
+int xfs_rtginode_create(struct xfs_rtgroup *rtg, enum xfs_rtg_inodes type,
+               bool init);
 void xfs_rtginode_irele(struct xfs_inode **ipp);
 
 static inline const char *xfs_rtginode_path(xfs_rgnumber_t rgno,