From patchwork Sat Dec 7 00:19:17 2024 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: "Darrick J. Wong" X-Patchwork-Id: 13898148 Received: from smtp.kernel.org (aws-us-west-2-korg-mail-1.web.codeaurora.org [10.30.226.201]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id A96434A1C for ; Sat, 7 Dec 2024 00:19:18 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=10.30.226.201 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1733530758; cv=none; b=rL6DncVwtA24CvPZEIG6tg/HowKXVamH3/FmDDKRp21UIHdZax3KhP51DA9TdlIKs/qLiwCs4XXrYqRWy6CRipxH0lcUqyZRTpaxfAgr5dUZfK/78BJDqXlgb6NSpm/DTeYr7hhgJwyV8SDp0TceEqH4TNOwTCwGBBUPAgx8m2I= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1733530758; c=relaxed/simple; bh=g4Q8cVBkl57XgSJZeoSVgyLrcfXynYS16FoPC5P98tw=; h=Date:Subject:From:To:Cc:Message-ID:In-Reply-To:References: MIME-Version:Content-Type; b=kYABCq3H/lt9ipTdKCSE6Dz6AWji2sKQVENeO4gqZqXqxrGC0elOndr5Tqxcbjz0U6vWCpbmfRsVfb6lZZI8Wl2Uu6v5zZuuPw7c1OYNBj5ylSDWR54Ys+J5/aDny6gBTCOa5clHhO78FUR/mx2i5CLAUhOKC8Fuc5X2Jo9VhtE= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=kernel.org header.i=@kernel.org header.b=vL5m/KMo; arc=none smtp.client-ip=10.30.226.201 Authentication-Results: smtp.subspace.kernel.org; dkim=pass (2048-bit key) header.d=kernel.org header.i=@kernel.org header.b="vL5m/KMo" Received: by smtp.kernel.org (Postfix) with ESMTPSA id 8258AC4CED1; Sat, 7 Dec 2024 00:19:18 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=k20201202; t=1733530758; bh=g4Q8cVBkl57XgSJZeoSVgyLrcfXynYS16FoPC5P98tw=; h=Date:Subject:From:To:Cc:In-Reply-To:References:From; b=vL5m/KMo3hk6ko6I+y5K/IpJzQDGGAcn7NhpcBqtUFvgABAB6oTEwZPvRichj+vbc Uzl2ik73jhni0tDdiDarhDAXGsclyEWbA0ZXV+siUIalUiczuH4umr81XHgghKBU4s yQ8O5jOApgYoAHChDdkmVGffkOyIRojiq2vq9I3MQYe7AADhUcpRu7JWUsI4SnzBsx eYCjrelVdTkRtkKxUJKEJKqdlf0DM7sb2kTg54ZqaA00TXGc/IYpp/7JorQQyuLANh hTjw40IrIHxc2pqTDYYy7eisSuLYjJlHNf85BUJhFNUGTSZbHL2PaSdu+3RysNb0Oc SxVLB3Xe7V6Tw== Date: Fri, 06 Dec 2024 16:19:17 -0800 Subject: [PATCH 5/7] xfs_repair: support quota inodes in the metadata directory From: "Darrick J. Wong" To: aalbersh@kernel.org, djwong@kernel.org Cc: hch@lst.de, linux-xfs@vger.kernel.org Message-ID: <173352753310.129683.8156547641009207395.stgit@frogsfrogsfrogs> In-Reply-To: <173352753222.129683.17995064282877591283.stgit@frogsfrogsfrogs> References: <173352753222.129683.17995064282877591283.stgit@frogsfrogsfrogs> Precedence: bulk X-Mailing-List: linux-xfs@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 From: Darrick J. Wong Handle quota inodes on metadir filesystems. Signed-off-by: "Darrick J. Wong" Reviewed-by: Christoph Hellwig --- repair/agheader.c | 3 + repair/phase2.c | 3 + repair/phase6.c | 116 +++++++++++++++++++++++++++++++++++++++++++++++++++ repair/quotacheck.c | 71 +++++++++++++++++++++++++++++++ repair/quotacheck.h | 1 repair/xfs_repair.c | 3 + 6 files changed, 194 insertions(+), 3 deletions(-) diff --git a/repair/agheader.c b/repair/agheader.c index 4a530fd6b8fe96..e6fca07c6cb4c9 100644 --- a/repair/agheader.c +++ b/repair/agheader.c @@ -509,7 +509,8 @@ secondary_sb_whack( rval |= XR_AG_SB_SEC; } - rval |= secondary_sb_quota(mp, sbuf, sb, i, do_bzero); + if (!xfs_has_metadir(mp)) + rval |= secondary_sb_quota(mp, sbuf, sb, i, do_bzero); /* * if the secondaries agree on a stripe unit/width or inode diff --git a/repair/phase2.c b/repair/phase2.c index 27c873fca76747..71576f5806e473 100644 --- a/repair/phase2.c +++ b/repair/phase2.c @@ -15,6 +15,7 @@ #include "progress.h" #include "scan.h" #include "rt.h" +#include "quotacheck.h" /* workaround craziness in the xlog routines */ int xlog_recover_do_trans(struct xlog *log, struct xlog_recover *t, int p) @@ -625,6 +626,8 @@ phase2( } discover_rtgroup_inodes(mp); + if (xfs_has_metadir(mp) && xfs_has_quota(mp)) + discover_quota_inodes(mp); /* * Upgrade the filesystem now that we've done a preliminary check of diff --git a/repair/phase6.c b/repair/phase6.c index 7bfacb66d4464e..6f124f749116e1 100644 --- a/repair/phase6.c +++ b/repair/phase6.c @@ -20,6 +20,7 @@ #include "versions.h" #include "repair/pptr.h" #include "repair/rt.h" +#include "repair/quotacheck.h" static xfs_ino_t orphanage_ino; @@ -3186,7 +3187,7 @@ mark_standalone_inodes(xfs_mount_t *mp) mark_inode(mp, mp->m_sb.sb_rsumino); } - if (!fs_quotas) + if (!fs_quotas || xfs_has_metadir(mp)) return; if (has_quota_inode(XFS_DQTYPE_USER)) @@ -3408,6 +3409,116 @@ _(" - resetting contents of realtime bitmap and summary inodes\n")); } } +static bool +ensure_quota_file( + struct xfs_inode *dp, + xfs_dqtype_t type) +{ + struct xfs_mount *mp = dp->i_mount; + struct xfs_inode *ip; + const char *name = libxfs_dqinode_path(type); + int error; + + if (!has_quota_inode(type)) + return false; + + if (no_modify) { + if (lost_quota_inode(type)) + do_warn(_("would reset %s quota inode\n"), name); + return false; + } + + if (!lost_quota_inode(type)) { + /* + * The /quotas directory has been discarded, but we should + * be able to iget the quota files directly. + */ + error = -libxfs_metafile_iget(mp, get_quota_inode(type), + xfs_dqinode_metafile_type(type), &ip); + if (error) { + do_warn( +_("Could not open %s quota inode, error %d\n"), + name, error); + lose_quota_inode(type); + } + } + + if (lost_quota_inode(type)) { + /* + * The inode was bad or missing, state that we'll make a new + * one even though we always create a new one. + */ + do_warn(_("resetting %s quota inode\n"), name); + error = -libxfs_dqinode_metadir_create(dp, type, &ip); + if (error) { + do_warn( +_("Couldn't create %s quota inode, error %d\n"), + name, error); + goto bad; + } + } else { + struct xfs_trans *tp; + + /* Erase parent pointers before we create the new link */ + try_erase_parent_ptrs(ip); + + error = -libxfs_dqinode_metadir_link(dp, type, ip); + if (error) { + do_warn( +_("Couldn't link %s quota inode, error %d\n"), + name, error); + goto bad; + } + + /* + * Reset the link count to 1 because quota files are never + * hardlinked, but the link above probably bumped it. + */ + error = -libxfs_trans_alloc_inode(ip, &M_RES(mp)->tr_ichange, + 0, 0, false, &tp); + if (!error) { + set_nlink(VFS_I(ip), 1); + libxfs_trans_log_inode(tp, ip, XFS_ILOG_CORE); + error = -libxfs_trans_commit(tp); + } + if (error) + do_error( +_("Couldn't reset link count on %s quota inode, error %d\n"), + name, error); + } + + /* Mark the inode in use. */ + mark_ino_inuse(mp, ip->i_ino, S_IFREG, dp->i_ino); + mark_ino_metadata(mp, ip->i_ino); + libxfs_irele(ip); + return true; +bad: + /* Zeroes qflags */ + quotacheck_skip(); + return false; +} + +static void +reset_quota_metadir_inodes( + struct xfs_mount *mp) +{ + struct xfs_inode *dp = NULL; + int error; + + error = -libxfs_dqinode_mkdir_parent(mp, &dp); + if (error) + do_error(_("failed to create quota metadir (%d)\n"), + error); + + mark_ino_inuse(mp, dp->i_ino, S_IFDIR, mp->m_metadirip->i_ino); + mark_ino_metadata(mp, dp->i_ino); + + ensure_quota_file(dp, XFS_DQTYPE_USER); + ensure_quota_file(dp, XFS_DQTYPE_GROUP); + ensure_quota_file(dp, XFS_DQTYPE_PROJ); + libxfs_irele(dp); +} + void phase6(xfs_mount_t *mp) { @@ -3461,6 +3572,9 @@ phase6(xfs_mount_t *mp) else reset_rt_sb_inodes(mp); + if (xfs_has_metadir(mp) && xfs_has_quota(mp) && !no_modify) + reset_quota_metadir_inodes(mp); + mark_standalone_inodes(mp); do_log(_(" - traversing filesystem ...\n")); diff --git a/repair/quotacheck.c b/repair/quotacheck.c index c4baf70e41d6b1..8c7339b267d8e6 100644 --- a/repair/quotacheck.c +++ b/repair/quotacheck.c @@ -645,3 +645,74 @@ update_sb_quotinos( if (dirty) libxfs_sb_to_disk(sbp->b_addr, &mp->m_sb); } + +static inline int +mark_quota_inode( + struct xfs_trans *tp, + struct xfs_inode *dp, + xfs_dqtype_t type) +{ + struct xfs_inode *ip; + int error; + + error = -libxfs_dqinode_load(tp, dp, type, &ip); + if (error == ENOENT) + return 0; + if (error) + goto out_corrupt; + + set_quota_inode(type, ip->i_ino); + libxfs_irele(ip); + return 0; + +out_corrupt: + lose_quota_inode(type); + return error; +} + +/* Mark the reachable quota metadata inodes prior to the inode scan. */ +void +discover_quota_inodes( + struct xfs_mount *mp) +{ + struct xfs_trans *tp; + struct xfs_inode *dp = NULL; + int error, err2; + + error = -libxfs_trans_alloc_empty(mp, &tp); + if (error) + goto out; + + error = -libxfs_dqinode_load_parent(tp, &dp); + if (error) + goto out_cancel; + + error = mark_quota_inode(tp, dp, XFS_DQTYPE_USER); + err2 = mark_quota_inode(tp, dp, XFS_DQTYPE_GROUP); + if (err2 && !error) + error = err2; + error = mark_quota_inode(tp, dp, XFS_DQTYPE_PROJ); + if (err2 && !error) + error = err2; + + libxfs_irele(dp); +out_cancel: + libxfs_trans_cancel(tp); +out: + if (error) { + switch (error) { + case EFSCORRUPTED: + do_warn( + _("corruption in metadata directory tree while discovering quota inodes\n")); + break; + case ENOENT: + /* Do nothing, we'll just clear qflags later. */ + break; + default: + do_warn( + _("couldn't discover quota inodes, err %d\n"), + error); + break; + } + } +} diff --git a/repair/quotacheck.h b/repair/quotacheck.h index 36f9f5a12f7f3e..24c9ffa418a3bf 100644 --- a/repair/quotacheck.h +++ b/repair/quotacheck.h @@ -14,5 +14,6 @@ int quotacheck_setup(struct xfs_mount *mp); void quotacheck_teardown(void); void update_sb_quotinos(struct xfs_mount *mp, struct xfs_buf *sbp); +void discover_quota_inodes(struct xfs_mount *mp); #endif /* __XFS_REPAIR_QUOTACHECK_H__ */ diff --git a/repair/xfs_repair.c b/repair/xfs_repair.c index 363f8260bd575a..9509f04685c870 100644 --- a/repair/xfs_repair.c +++ b/repair/xfs_repair.c @@ -1487,7 +1487,8 @@ _("Warning: project quota information would be cleared.\n" if (!sbp) do_error(_("couldn't get superblock\n")); - update_sb_quotinos(mp, sbp); + if (!xfs_has_metadir(mp)) + update_sb_quotinos(mp, sbp); if ((mp->m_sb.sb_qflags & XFS_ALL_QUOTA_CHKD) != quotacheck_results()) { do_warn(_("Note - quota info will be regenerated on next "