Commit 6b08901a authored by Darrick J. Wong's avatar Darrick J. Wong
Browse files

xfs: add realtime reverse map inode to metadata directory



Add a metadir path to select the realtime rmap btree inode and load
it at mount time.  The rtrmapbt inode will have a unique extent format
code, which means that we also have to update the inode validation and
flush routines to look for it.

Signed-off-by: default avatar"Darrick J. Wong" <djwong@kernel.org>
Reviewed-by: default avatarChristoph Hellwig <hch@lst.de>
parent 702c90f4
Loading
Loading
Loading
Loading
+3 −1
Original line number Diff line number Diff line
@@ -857,6 +857,7 @@ enum xfs_metafile_type {
	XFS_METAFILE_PRJQUOTA,		/* project quota */
	XFS_METAFILE_RTBITMAP,		/* rt bitmap */
	XFS_METAFILE_RTSUMMARY,		/* rt summary */
	XFS_METAFILE_RTRMAP,		/* rt rmap */

	XFS_METAFILE_MAX
} __packed;
@@ -868,7 +869,8 @@ enum xfs_metafile_type {
	{ XFS_METAFILE_GRPQUOTA,	"grpquota" }, \
	{ XFS_METAFILE_PRJQUOTA,	"prjquota" }, \
	{ XFS_METAFILE_RTBITMAP,	"rtbitmap" }, \
	{ XFS_METAFILE_RTSUMMARY,	"rtsummary" }
	{ XFS_METAFILE_RTSUMMARY,	"rtsummary" }, \
	{ XFS_METAFILE_RTRMAP,		"rtrmap" }

/*
 * On-disk inode structure.
+9 −0
Original line number Diff line number Diff line
@@ -447,6 +447,15 @@ xfs_dinode_verify_fork(
		if (!(dip->di_flags2 & cpu_to_be64(XFS_DIFLAG2_METADATA)))
			return __this_address;
		switch (be16_to_cpu(dip->di_metatype)) {
		case XFS_METAFILE_RTRMAP:
			/*
			 * growfs must create the rtrmap inodes before adding a
			 * realtime volume to the filesystem, so we cannot use
			 * the rtrmapbt predicate here.
			 */
			if (!xfs_has_rmapbt(mp))
				return __this_address;
			break;
		default:
			return __this_address;
		}
+6 −0
Original line number Diff line number Diff line
@@ -269,6 +269,9 @@ xfs_iformat_data_fork(
			return xfs_iformat_btree(ip, dip, XFS_DATA_FORK);
		case XFS_DINODE_FMT_META_BTREE:
			switch (ip->i_metatype) {
			case XFS_METAFILE_RTRMAP:
				ASSERT(0); /* to be implemented later */
				return -EFSCORRUPTED;
			default:
				break;
			}
@@ -614,6 +617,9 @@ xfs_iflush_fork(
			break;

		switch (ip->i_metatype) {
		case XFS_METAFILE_RTRMAP:
			ASSERT(0); /* to be implemented later */
			break;
		default:
			ASSERT(0);
			break;
+18 −2
Original line number Diff line number Diff line
@@ -315,6 +315,8 @@ struct xfs_rtginode_ops {

	unsigned int		sick;	/* rtgroup sickness flag */

	unsigned int		fmt_mask; /* all valid data fork formats */

	/* Does the fs have this feature? */
	bool			(*enabled)(struct xfs_mount *mp);

@@ -330,14 +332,29 @@ static const struct xfs_rtginode_ops xfs_rtginode_ops[XFS_RTGI_MAX] = {
		.name		= "bitmap",
		.metafile_type	= XFS_METAFILE_RTBITMAP,
		.sick		= XFS_SICK_RG_BITMAP,
		.fmt_mask	= (1U << XFS_DINODE_FMT_EXTENTS) |
				  (1U << XFS_DINODE_FMT_BTREE),
		.create		= xfs_rtbitmap_create,
	},
	[XFS_RTGI_SUMMARY] = {
		.name		= "summary",
		.metafile_type	= XFS_METAFILE_RTSUMMARY,
		.sick		= XFS_SICK_RG_SUMMARY,
		.fmt_mask	= (1U << XFS_DINODE_FMT_EXTENTS) |
				  (1U << XFS_DINODE_FMT_BTREE),
		.create		= xfs_rtsummary_create,
	},
	[XFS_RTGI_RMAP] = {
		.name		= "rmap",
		.metafile_type	= XFS_METAFILE_RTRMAP,
		.fmt_mask	= 1U << XFS_DINODE_FMT_META_BTREE,
		/*
		 * growfs must create the rtrmap inodes before adding a
		 * realtime volume to the filesystem, so we cannot use the
		 * rtrmapbt predicate here.
		 */
		.enabled	= xfs_has_rmapbt,
	},
};

/* Return the shortname of this rtgroup inode. */
@@ -434,8 +451,7 @@ xfs_rtginode_load(
		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)) {
	if (XFS_IS_CORRUPT(mp, !((1U << ip->i_df.if_format) & ops->fmt_mask))) {
		xfs_irele(ip);
		xfs_rtginode_mark_sick(rtg, type);
		return -EFSCORRUPTED;
+8 −0
Original line number Diff line number Diff line
@@ -14,6 +14,7 @@ struct xfs_trans;
enum xfs_rtg_inodes {
	XFS_RTGI_BITMAP,	/* allocation bitmap */
	XFS_RTGI_SUMMARY,	/* allocation summary */
	XFS_RTGI_RMAP,		/* rmap btree inode */

	XFS_RTGI_MAX,
};
@@ -74,6 +75,11 @@ static inline struct xfs_inode *rtg_summary(const struct xfs_rtgroup *rtg)
	return rtg->rtg_inodes[XFS_RTGI_SUMMARY];
}

static inline struct xfs_inode *rtg_rmap(const struct xfs_rtgroup *rtg)
{
	return rtg->rtg_inodes[XFS_RTGI_RMAP];
}

/* Passive rtgroup references */
static inline struct xfs_rtgroup *
xfs_rtgroup_get(
@@ -284,6 +290,8 @@ int xfs_rtginode_create(struct xfs_rtgroup *rtg, enum xfs_rtg_inodes type,
		bool init);
void xfs_rtginode_irele(struct xfs_inode **ipp);

void xfs_rtginode_irele(struct xfs_inode **ipp);

static inline const char *xfs_rtginode_path(xfs_rgnumber_t rgno,
		enum xfs_rtg_inodes type)
{
Loading