mirror of
https://github.com/torvalds/linux.git
synced 2024-11-21 19:41:42 +00:00
1433f8f9ce
Repair the realtime superblock if it has become out of date with the primary superblock. Signed-off-by: Darrick J. Wong <djwong@kernel.org> Reviewed-by: Christoph Hellwig <hch@lst.de>
85 lines
2.0 KiB
C
85 lines
2.0 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Copyright (c) 2022-2024 Oracle. All Rights Reserved.
|
|
* Author: Darrick J. Wong <djwong@kernel.org>
|
|
*/
|
|
#include "xfs.h"
|
|
#include "xfs_fs.h"
|
|
#include "xfs_shared.h"
|
|
#include "xfs_format.h"
|
|
#include "xfs_trans_resv.h"
|
|
#include "xfs_mount.h"
|
|
#include "xfs_rtgroup.h"
|
|
#include "xfs_log_format.h"
|
|
#include "xfs_trans.h"
|
|
#include "xfs_sb.h"
|
|
#include "scrub/scrub.h"
|
|
#include "scrub/common.h"
|
|
#include "scrub/repair.h"
|
|
|
|
/* Set us up with a transaction and an empty context. */
|
|
int
|
|
xchk_setup_rgsuperblock(
|
|
struct xfs_scrub *sc)
|
|
{
|
|
return xchk_trans_alloc(sc, 0);
|
|
}
|
|
|
|
/* Cross-reference with the other rt metadata. */
|
|
STATIC void
|
|
xchk_rgsuperblock_xref(
|
|
struct xfs_scrub *sc)
|
|
{
|
|
if (sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT)
|
|
return;
|
|
|
|
xchk_xref_is_used_rt_space(sc, xfs_rgbno_to_rtb(sc->sr.rtg, 0), 1);
|
|
}
|
|
|
|
int
|
|
xchk_rgsuperblock(
|
|
struct xfs_scrub *sc)
|
|
{
|
|
xfs_rgnumber_t rgno = sc->sm->sm_agno;
|
|
int error;
|
|
|
|
/*
|
|
* Only rtgroup 0 has a superblock. We may someday want to use higher
|
|
* rgno for other functions, similar to what we do with the primary
|
|
* super scrub function.
|
|
*/
|
|
if (rgno != 0)
|
|
return -ENOENT;
|
|
|
|
/*
|
|
* Grab an active reference to the rtgroup structure. If we can't get
|
|
* it, we're racing with something that's tearing down the group, so
|
|
* signal that the group no longer exists. Take the rtbitmap in shared
|
|
* mode so that the group can't change while we're doing things.
|
|
*/
|
|
error = xchk_rtgroup_init_existing(sc, rgno, &sc->sr);
|
|
if (!xchk_xref_process_error(sc, 0, 0, &error))
|
|
return error;
|
|
|
|
xchk_rtgroup_lock(&sc->sr, XFS_RTGLOCK_BITMAP_SHARED);
|
|
|
|
/*
|
|
* Since we already validated the rt superblock at mount time, we don't
|
|
* need to check its contents again. All we need is to cross-reference.
|
|
*/
|
|
xchk_rgsuperblock_xref(sc);
|
|
return 0;
|
|
}
|
|
|
|
#ifdef CONFIG_XFS_ONLINE_REPAIR
|
|
int
|
|
xrep_rgsuperblock(
|
|
struct xfs_scrub *sc)
|
|
{
|
|
ASSERT(rtg_rgno(sc->sr.rtg) == 0);
|
|
|
|
xfs_log_sb(sc->tp);
|
|
return 0;
|
|
}
|
|
#endif /* CONFIG_XFS_ONLINE_REPAIR */
|