32080a9b9b
Rebuild the reverse mapping btree from all primary metadata. This first patch establishes the bare mechanics of finding records and putting together a new ondisk tree; more complex pieces are needed to make it work properly. Link: Documentation/filesystems/xfs-online-fsck-design.rst Signed-off-by: Darrick J. Wong <djwong@kernel.org> Reviewed-by: Christoph Hellwig <hch@lst.de>
649 lines
16 KiB
C
649 lines
16 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Copyright (C) 2017-2023 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_log_format.h"
|
|
#include "xfs_trans_resv.h"
|
|
#include "xfs_mount.h"
|
|
#include "xfs_trans.h"
|
|
#include "xfs_btree.h"
|
|
#include "xfs_rmap.h"
|
|
#include "xfs_refcount.h"
|
|
#include "xfs_ag.h"
|
|
#include "xfs_bit.h"
|
|
#include "xfs_alloc.h"
|
|
#include "xfs_alloc_btree.h"
|
|
#include "xfs_ialloc_btree.h"
|
|
#include "xfs_refcount_btree.h"
|
|
#include "scrub/scrub.h"
|
|
#include "scrub/common.h"
|
|
#include "scrub/btree.h"
|
|
#include "scrub/bitmap.h"
|
|
#include "scrub/agb_bitmap.h"
|
|
#include "scrub/repair.h"
|
|
|
|
/*
|
|
* Set us up to scrub reverse mapping btrees.
|
|
*/
|
|
int
|
|
xchk_setup_ag_rmapbt(
|
|
struct xfs_scrub *sc)
|
|
{
|
|
if (xchk_need_intent_drain(sc))
|
|
xchk_fsgates_enable(sc, XCHK_FSGATES_DRAIN);
|
|
|
|
if (xchk_could_repair(sc)) {
|
|
int error;
|
|
|
|
error = xrep_setup_ag_rmapbt(sc);
|
|
if (error)
|
|
return error;
|
|
}
|
|
|
|
return xchk_setup_ag_btree(sc, false);
|
|
}
|
|
|
|
/* Reverse-mapping scrubber. */
|
|
|
|
struct xchk_rmap {
|
|
/*
|
|
* The furthest-reaching of the rmapbt records that we've already
|
|
* processed. This enables us to detect overlapping records for space
|
|
* allocations that cannot be shared.
|
|
*/
|
|
struct xfs_rmap_irec overlap_rec;
|
|
|
|
/*
|
|
* The previous rmapbt record, so that we can check for two records
|
|
* that could be one.
|
|
*/
|
|
struct xfs_rmap_irec prev_rec;
|
|
|
|
/* Bitmaps containing all blocks for each type of AG metadata. */
|
|
struct xagb_bitmap fs_owned;
|
|
struct xagb_bitmap log_owned;
|
|
struct xagb_bitmap ag_owned;
|
|
struct xagb_bitmap inobt_owned;
|
|
struct xagb_bitmap refcbt_owned;
|
|
|
|
/* Did we complete the AG space metadata bitmaps? */
|
|
bool bitmaps_complete;
|
|
};
|
|
|
|
/* Cross-reference a rmap against the refcount btree. */
|
|
STATIC void
|
|
xchk_rmapbt_xref_refc(
|
|
struct xfs_scrub *sc,
|
|
struct xfs_rmap_irec *irec)
|
|
{
|
|
xfs_agblock_t fbno;
|
|
xfs_extlen_t flen;
|
|
bool non_inode;
|
|
bool is_bmbt;
|
|
bool is_attr;
|
|
bool is_unwritten;
|
|
int error;
|
|
|
|
if (!sc->sa.refc_cur || xchk_skip_xref(sc->sm))
|
|
return;
|
|
|
|
non_inode = XFS_RMAP_NON_INODE_OWNER(irec->rm_owner);
|
|
is_bmbt = irec->rm_flags & XFS_RMAP_BMBT_BLOCK;
|
|
is_attr = irec->rm_flags & XFS_RMAP_ATTR_FORK;
|
|
is_unwritten = irec->rm_flags & XFS_RMAP_UNWRITTEN;
|
|
|
|
/* If this is shared, must be a data fork extent. */
|
|
error = xfs_refcount_find_shared(sc->sa.refc_cur, irec->rm_startblock,
|
|
irec->rm_blockcount, &fbno, &flen, false);
|
|
if (!xchk_should_check_xref(sc, &error, &sc->sa.refc_cur))
|
|
return;
|
|
if (flen != 0 && (non_inode || is_attr || is_bmbt || is_unwritten))
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.refc_cur, 0);
|
|
}
|
|
|
|
/* Cross-reference with the other btrees. */
|
|
STATIC void
|
|
xchk_rmapbt_xref(
|
|
struct xfs_scrub *sc,
|
|
struct xfs_rmap_irec *irec)
|
|
{
|
|
xfs_agblock_t agbno = irec->rm_startblock;
|
|
xfs_extlen_t len = irec->rm_blockcount;
|
|
|
|
if (sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT)
|
|
return;
|
|
|
|
xchk_xref_is_used_space(sc, agbno, len);
|
|
if (irec->rm_owner == XFS_RMAP_OWN_INODES)
|
|
xchk_xref_is_inode_chunk(sc, agbno, len);
|
|
else
|
|
xchk_xref_is_not_inode_chunk(sc, agbno, len);
|
|
if (irec->rm_owner == XFS_RMAP_OWN_COW)
|
|
xchk_xref_is_cow_staging(sc, irec->rm_startblock,
|
|
irec->rm_blockcount);
|
|
else
|
|
xchk_rmapbt_xref_refc(sc, irec);
|
|
}
|
|
|
|
/*
|
|
* Check for bogus UNWRITTEN flags in the rmapbt node block keys.
|
|
*
|
|
* In reverse mapping records, the file mapping extent state
|
|
* (XFS_RMAP_OFF_UNWRITTEN) is a record attribute, not a key field. It is not
|
|
* involved in lookups in any way. In older kernels, the functions that
|
|
* convert rmapbt records to keys forgot to filter out the extent state bit,
|
|
* even though the key comparison functions have filtered the flag correctly.
|
|
* If we spot an rmap key with the unwritten bit set in rm_offset, we should
|
|
* mark the btree as needing optimization to rebuild the btree without those
|
|
* flags.
|
|
*/
|
|
STATIC void
|
|
xchk_rmapbt_check_unwritten_in_keyflags(
|
|
struct xchk_btree *bs)
|
|
{
|
|
struct xfs_scrub *sc = bs->sc;
|
|
struct xfs_btree_cur *cur = bs->cur;
|
|
struct xfs_btree_block *keyblock;
|
|
union xfs_btree_key *lkey, *hkey;
|
|
__be64 badflag = cpu_to_be64(XFS_RMAP_OFF_UNWRITTEN);
|
|
unsigned int level;
|
|
|
|
if (sc->sm->sm_flags & XFS_SCRUB_OFLAG_PREEN)
|
|
return;
|
|
|
|
for (level = 1; level < cur->bc_nlevels; level++) {
|
|
struct xfs_buf *bp;
|
|
unsigned int ptr;
|
|
|
|
/* Only check the first time we've seen this node block. */
|
|
if (cur->bc_levels[level].ptr > 1)
|
|
continue;
|
|
|
|
keyblock = xfs_btree_get_block(cur, level, &bp);
|
|
for (ptr = 1; ptr <= be16_to_cpu(keyblock->bb_numrecs); ptr++) {
|
|
lkey = xfs_btree_key_addr(cur, ptr, keyblock);
|
|
|
|
if (lkey->rmap.rm_offset & badflag) {
|
|
xchk_btree_set_preen(sc, cur, level);
|
|
break;
|
|
}
|
|
|
|
hkey = xfs_btree_high_key_addr(cur, ptr, keyblock);
|
|
if (hkey->rmap.rm_offset & badflag) {
|
|
xchk_btree_set_preen(sc, cur, level);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static inline bool
|
|
xchk_rmapbt_is_shareable(
|
|
struct xfs_scrub *sc,
|
|
const struct xfs_rmap_irec *irec)
|
|
{
|
|
if (!xfs_has_reflink(sc->mp))
|
|
return false;
|
|
if (XFS_RMAP_NON_INODE_OWNER(irec->rm_owner))
|
|
return false;
|
|
if (irec->rm_flags & (XFS_RMAP_BMBT_BLOCK | XFS_RMAP_ATTR_FORK |
|
|
XFS_RMAP_UNWRITTEN))
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
/* Flag failures for records that overlap but cannot. */
|
|
STATIC void
|
|
xchk_rmapbt_check_overlapping(
|
|
struct xchk_btree *bs,
|
|
struct xchk_rmap *cr,
|
|
const struct xfs_rmap_irec *irec)
|
|
{
|
|
xfs_agblock_t pnext, inext;
|
|
|
|
if (bs->sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT)
|
|
return;
|
|
|
|
/* No previous record? */
|
|
if (cr->overlap_rec.rm_blockcount == 0)
|
|
goto set_prev;
|
|
|
|
/* Do overlap_rec and irec overlap? */
|
|
pnext = cr->overlap_rec.rm_startblock + cr->overlap_rec.rm_blockcount;
|
|
if (pnext <= irec->rm_startblock)
|
|
goto set_prev;
|
|
|
|
/* Overlap is only allowed if both records are data fork mappings. */
|
|
if (!xchk_rmapbt_is_shareable(bs->sc, &cr->overlap_rec) ||
|
|
!xchk_rmapbt_is_shareable(bs->sc, irec))
|
|
xchk_btree_set_corrupt(bs->sc, bs->cur, 0);
|
|
|
|
/* Save whichever rmap record extends furthest. */
|
|
inext = irec->rm_startblock + irec->rm_blockcount;
|
|
if (pnext > inext)
|
|
return;
|
|
|
|
set_prev:
|
|
memcpy(&cr->overlap_rec, irec, sizeof(struct xfs_rmap_irec));
|
|
}
|
|
|
|
/* Decide if two reverse-mapping records can be merged. */
|
|
static inline bool
|
|
xchk_rmap_mergeable(
|
|
struct xchk_rmap *cr,
|
|
const struct xfs_rmap_irec *r2)
|
|
{
|
|
const struct xfs_rmap_irec *r1 = &cr->prev_rec;
|
|
|
|
/* Ignore if prev_rec is not yet initialized. */
|
|
if (cr->prev_rec.rm_blockcount == 0)
|
|
return false;
|
|
|
|
if (r1->rm_owner != r2->rm_owner)
|
|
return false;
|
|
if (r1->rm_startblock + r1->rm_blockcount != r2->rm_startblock)
|
|
return false;
|
|
if ((unsigned long long)r1->rm_blockcount + r2->rm_blockcount >
|
|
XFS_RMAP_LEN_MAX)
|
|
return false;
|
|
if (XFS_RMAP_NON_INODE_OWNER(r2->rm_owner))
|
|
return true;
|
|
/* must be an inode owner below here */
|
|
if (r1->rm_flags != r2->rm_flags)
|
|
return false;
|
|
if (r1->rm_flags & XFS_RMAP_BMBT_BLOCK)
|
|
return true;
|
|
return r1->rm_offset + r1->rm_blockcount == r2->rm_offset;
|
|
}
|
|
|
|
/* Flag failures for records that could be merged. */
|
|
STATIC void
|
|
xchk_rmapbt_check_mergeable(
|
|
struct xchk_btree *bs,
|
|
struct xchk_rmap *cr,
|
|
const struct xfs_rmap_irec *irec)
|
|
{
|
|
if (bs->sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT)
|
|
return;
|
|
|
|
if (xchk_rmap_mergeable(cr, irec))
|
|
xchk_btree_set_corrupt(bs->sc, bs->cur, 0);
|
|
|
|
memcpy(&cr->prev_rec, irec, sizeof(struct xfs_rmap_irec));
|
|
}
|
|
|
|
/* Compare an rmap for AG metadata against the metadata walk. */
|
|
STATIC int
|
|
xchk_rmapbt_mark_bitmap(
|
|
struct xchk_btree *bs,
|
|
struct xchk_rmap *cr,
|
|
const struct xfs_rmap_irec *irec)
|
|
{
|
|
struct xfs_scrub *sc = bs->sc;
|
|
struct xagb_bitmap *bmp = NULL;
|
|
xfs_extlen_t fsbcount = irec->rm_blockcount;
|
|
|
|
/*
|
|
* Skip corrupt records. It is essential that we detect records in the
|
|
* btree that cannot overlap but do, flag those as CORRUPT, and skip
|
|
* the bitmap comparison to avoid generating false XCORRUPT reports.
|
|
*/
|
|
if (sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT)
|
|
return 0;
|
|
|
|
/*
|
|
* If the AG metadata walk didn't complete, there's no point in
|
|
* comparing against partial results.
|
|
*/
|
|
if (!cr->bitmaps_complete)
|
|
return 0;
|
|
|
|
switch (irec->rm_owner) {
|
|
case XFS_RMAP_OWN_FS:
|
|
bmp = &cr->fs_owned;
|
|
break;
|
|
case XFS_RMAP_OWN_LOG:
|
|
bmp = &cr->log_owned;
|
|
break;
|
|
case XFS_RMAP_OWN_AG:
|
|
bmp = &cr->ag_owned;
|
|
break;
|
|
case XFS_RMAP_OWN_INOBT:
|
|
bmp = &cr->inobt_owned;
|
|
break;
|
|
case XFS_RMAP_OWN_REFC:
|
|
bmp = &cr->refcbt_owned;
|
|
break;
|
|
}
|
|
|
|
if (!bmp)
|
|
return 0;
|
|
|
|
if (xagb_bitmap_test(bmp, irec->rm_startblock, &fsbcount)) {
|
|
/*
|
|
* The start of this reverse mapping corresponds to a set
|
|
* region in the bitmap. If the mapping covers more area than
|
|
* the set region, then it covers space that wasn't found by
|
|
* the AG metadata walk.
|
|
*/
|
|
if (fsbcount < irec->rm_blockcount)
|
|
xchk_btree_xref_set_corrupt(bs->sc,
|
|
bs->sc->sa.rmap_cur, 0);
|
|
} else {
|
|
/*
|
|
* The start of this reverse mapping does not correspond to a
|
|
* completely set region in the bitmap. The region wasn't
|
|
* fully set by walking the AG metadata, so this is a
|
|
* cross-referencing corruption.
|
|
*/
|
|
xchk_btree_xref_set_corrupt(bs->sc, bs->sc->sa.rmap_cur, 0);
|
|
}
|
|
|
|
/* Unset the region so that we can detect missing rmap records. */
|
|
return xagb_bitmap_clear(bmp, irec->rm_startblock, irec->rm_blockcount);
|
|
}
|
|
|
|
/* Scrub an rmapbt record. */
|
|
STATIC int
|
|
xchk_rmapbt_rec(
|
|
struct xchk_btree *bs,
|
|
const union xfs_btree_rec *rec)
|
|
{
|
|
struct xchk_rmap *cr = bs->private;
|
|
struct xfs_rmap_irec irec;
|
|
|
|
if (xfs_rmap_btrec_to_irec(rec, &irec) != NULL ||
|
|
xfs_rmap_check_irec(bs->cur->bc_ag.pag, &irec) != NULL) {
|
|
xchk_btree_set_corrupt(bs->sc, bs->cur, 0);
|
|
return 0;
|
|
}
|
|
|
|
xchk_rmapbt_check_unwritten_in_keyflags(bs);
|
|
xchk_rmapbt_check_mergeable(bs, cr, &irec);
|
|
xchk_rmapbt_check_overlapping(bs, cr, &irec);
|
|
xchk_rmapbt_xref(bs->sc, &irec);
|
|
|
|
return xchk_rmapbt_mark_bitmap(bs, cr, &irec);
|
|
}
|
|
|
|
/* Add an AGFL block to the rmap list. */
|
|
STATIC int
|
|
xchk_rmapbt_walk_agfl(
|
|
struct xfs_mount *mp,
|
|
xfs_agblock_t agbno,
|
|
void *priv)
|
|
{
|
|
struct xagb_bitmap *bitmap = priv;
|
|
|
|
return xagb_bitmap_set(bitmap, agbno, 1);
|
|
}
|
|
|
|
/*
|
|
* Set up bitmaps mapping all the AG metadata to compare with the rmapbt
|
|
* records.
|
|
*
|
|
* Grab our own btree cursors here if the scrub setup function didn't give us a
|
|
* btree cursor due to reports of poor health. We need to find out if the
|
|
* rmapbt disagrees with primary metadata btrees to tag the rmapbt as being
|
|
* XCORRUPT.
|
|
*/
|
|
STATIC int
|
|
xchk_rmapbt_walk_ag_metadata(
|
|
struct xfs_scrub *sc,
|
|
struct xchk_rmap *cr)
|
|
{
|
|
struct xfs_mount *mp = sc->mp;
|
|
struct xfs_buf *agfl_bp;
|
|
struct xfs_agf *agf = sc->sa.agf_bp->b_addr;
|
|
struct xfs_btree_cur *cur;
|
|
int error;
|
|
|
|
/* OWN_FS: AG headers */
|
|
error = xagb_bitmap_set(&cr->fs_owned, XFS_SB_BLOCK(mp),
|
|
XFS_AGFL_BLOCK(mp) - XFS_SB_BLOCK(mp) + 1);
|
|
if (error)
|
|
goto out;
|
|
|
|
/* OWN_LOG: Internal log */
|
|
if (xfs_ag_contains_log(mp, sc->sa.pag->pag_agno)) {
|
|
error = xagb_bitmap_set(&cr->log_owned,
|
|
XFS_FSB_TO_AGBNO(mp, mp->m_sb.sb_logstart),
|
|
mp->m_sb.sb_logblocks);
|
|
if (error)
|
|
goto out;
|
|
}
|
|
|
|
/* OWN_AG: bnobt, cntbt, rmapbt, and AGFL */
|
|
cur = sc->sa.bno_cur;
|
|
if (!cur)
|
|
cur = xfs_bnobt_init_cursor(sc->mp, sc->tp, sc->sa.agf_bp,
|
|
sc->sa.pag);
|
|
error = xagb_bitmap_set_btblocks(&cr->ag_owned, cur);
|
|
if (cur != sc->sa.bno_cur)
|
|
xfs_btree_del_cursor(cur, error);
|
|
if (error)
|
|
goto out;
|
|
|
|
cur = sc->sa.cnt_cur;
|
|
if (!cur)
|
|
cur = xfs_cntbt_init_cursor(sc->mp, sc->tp, sc->sa.agf_bp,
|
|
sc->sa.pag);
|
|
error = xagb_bitmap_set_btblocks(&cr->ag_owned, cur);
|
|
if (cur != sc->sa.cnt_cur)
|
|
xfs_btree_del_cursor(cur, error);
|
|
if (error)
|
|
goto out;
|
|
|
|
error = xagb_bitmap_set_btblocks(&cr->ag_owned, sc->sa.rmap_cur);
|
|
if (error)
|
|
goto out;
|
|
|
|
error = xfs_alloc_read_agfl(sc->sa.pag, sc->tp, &agfl_bp);
|
|
if (error)
|
|
goto out;
|
|
|
|
error = xfs_agfl_walk(sc->mp, agf, agfl_bp, xchk_rmapbt_walk_agfl,
|
|
&cr->ag_owned);
|
|
xfs_trans_brelse(sc->tp, agfl_bp);
|
|
if (error)
|
|
goto out;
|
|
|
|
/* OWN_INOBT: inobt, finobt */
|
|
cur = sc->sa.ino_cur;
|
|
if (!cur)
|
|
cur = xfs_inobt_init_cursor(sc->sa.pag, sc->tp, sc->sa.agi_bp);
|
|
error = xagb_bitmap_set_btblocks(&cr->inobt_owned, cur);
|
|
if (cur != sc->sa.ino_cur)
|
|
xfs_btree_del_cursor(cur, error);
|
|
if (error)
|
|
goto out;
|
|
|
|
if (xfs_has_finobt(sc->mp)) {
|
|
cur = sc->sa.fino_cur;
|
|
if (!cur)
|
|
cur = xfs_finobt_init_cursor(sc->sa.pag, sc->tp,
|
|
sc->sa.agi_bp);
|
|
error = xagb_bitmap_set_btblocks(&cr->inobt_owned, cur);
|
|
if (cur != sc->sa.fino_cur)
|
|
xfs_btree_del_cursor(cur, error);
|
|
if (error)
|
|
goto out;
|
|
}
|
|
|
|
/* OWN_REFC: refcountbt */
|
|
if (xfs_has_reflink(sc->mp)) {
|
|
cur = sc->sa.refc_cur;
|
|
if (!cur)
|
|
cur = xfs_refcountbt_init_cursor(sc->mp, sc->tp,
|
|
sc->sa.agf_bp, sc->sa.pag);
|
|
error = xagb_bitmap_set_btblocks(&cr->refcbt_owned, cur);
|
|
if (cur != sc->sa.refc_cur)
|
|
xfs_btree_del_cursor(cur, error);
|
|
if (error)
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
/*
|
|
* If there's an error, set XFAIL and disable the bitmap
|
|
* cross-referencing checks, but proceed with the scrub anyway.
|
|
*/
|
|
if (error)
|
|
xchk_btree_xref_process_error(sc, sc->sa.rmap_cur,
|
|
sc->sa.rmap_cur->bc_nlevels - 1, &error);
|
|
else
|
|
cr->bitmaps_complete = true;
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Check for set regions in the bitmaps; if there are any, the rmap records do
|
|
* not describe all the AG metadata.
|
|
*/
|
|
STATIC void
|
|
xchk_rmapbt_check_bitmaps(
|
|
struct xfs_scrub *sc,
|
|
struct xchk_rmap *cr)
|
|
{
|
|
struct xfs_btree_cur *cur = sc->sa.rmap_cur;
|
|
unsigned int level;
|
|
|
|
if (sc->sm->sm_flags & (XFS_SCRUB_OFLAG_CORRUPT |
|
|
XFS_SCRUB_OFLAG_XFAIL))
|
|
return;
|
|
if (!cur)
|
|
return;
|
|
level = cur->bc_nlevels - 1;
|
|
|
|
/*
|
|
* Any bitmap with bits still set indicates that the reverse mapping
|
|
* doesn't cover the entire primary structure.
|
|
*/
|
|
if (xagb_bitmap_hweight(&cr->fs_owned) != 0)
|
|
xchk_btree_xref_set_corrupt(sc, cur, level);
|
|
|
|
if (xagb_bitmap_hweight(&cr->log_owned) != 0)
|
|
xchk_btree_xref_set_corrupt(sc, cur, level);
|
|
|
|
if (xagb_bitmap_hweight(&cr->ag_owned) != 0)
|
|
xchk_btree_xref_set_corrupt(sc, cur, level);
|
|
|
|
if (xagb_bitmap_hweight(&cr->inobt_owned) != 0)
|
|
xchk_btree_xref_set_corrupt(sc, cur, level);
|
|
|
|
if (xagb_bitmap_hweight(&cr->refcbt_owned) != 0)
|
|
xchk_btree_xref_set_corrupt(sc, cur, level);
|
|
}
|
|
|
|
/* Scrub the rmap btree for some AG. */
|
|
int
|
|
xchk_rmapbt(
|
|
struct xfs_scrub *sc)
|
|
{
|
|
struct xchk_rmap *cr;
|
|
int error;
|
|
|
|
cr = kzalloc(sizeof(struct xchk_rmap), XCHK_GFP_FLAGS);
|
|
if (!cr)
|
|
return -ENOMEM;
|
|
|
|
xagb_bitmap_init(&cr->fs_owned);
|
|
xagb_bitmap_init(&cr->log_owned);
|
|
xagb_bitmap_init(&cr->ag_owned);
|
|
xagb_bitmap_init(&cr->inobt_owned);
|
|
xagb_bitmap_init(&cr->refcbt_owned);
|
|
|
|
error = xchk_rmapbt_walk_ag_metadata(sc, cr);
|
|
if (error)
|
|
goto out;
|
|
|
|
error = xchk_btree(sc, sc->sa.rmap_cur, xchk_rmapbt_rec,
|
|
&XFS_RMAP_OINFO_AG, cr);
|
|
if (error)
|
|
goto out;
|
|
|
|
xchk_rmapbt_check_bitmaps(sc, cr);
|
|
|
|
out:
|
|
xagb_bitmap_destroy(&cr->refcbt_owned);
|
|
xagb_bitmap_destroy(&cr->inobt_owned);
|
|
xagb_bitmap_destroy(&cr->ag_owned);
|
|
xagb_bitmap_destroy(&cr->log_owned);
|
|
xagb_bitmap_destroy(&cr->fs_owned);
|
|
kfree(cr);
|
|
return error;
|
|
}
|
|
|
|
/* xref check that the extent is owned only by a given owner */
|
|
void
|
|
xchk_xref_is_only_owned_by(
|
|
struct xfs_scrub *sc,
|
|
xfs_agblock_t bno,
|
|
xfs_extlen_t len,
|
|
const struct xfs_owner_info *oinfo)
|
|
{
|
|
struct xfs_rmap_matches res;
|
|
int error;
|
|
|
|
if (!sc->sa.rmap_cur || xchk_skip_xref(sc->sm))
|
|
return;
|
|
|
|
error = xfs_rmap_count_owners(sc->sa.rmap_cur, bno, len, oinfo, &res);
|
|
if (!xchk_should_check_xref(sc, &error, &sc->sa.rmap_cur))
|
|
return;
|
|
if (res.matches != 1)
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.rmap_cur, 0);
|
|
if (res.bad_non_owner_matches)
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.rmap_cur, 0);
|
|
if (res.non_owner_matches)
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.rmap_cur, 0);
|
|
}
|
|
|
|
/* xref check that the extent is not owned by a given owner */
|
|
void
|
|
xchk_xref_is_not_owned_by(
|
|
struct xfs_scrub *sc,
|
|
xfs_agblock_t bno,
|
|
xfs_extlen_t len,
|
|
const struct xfs_owner_info *oinfo)
|
|
{
|
|
struct xfs_rmap_matches res;
|
|
int error;
|
|
|
|
if (!sc->sa.rmap_cur || xchk_skip_xref(sc->sm))
|
|
return;
|
|
|
|
error = xfs_rmap_count_owners(sc->sa.rmap_cur, bno, len, oinfo, &res);
|
|
if (!xchk_should_check_xref(sc, &error, &sc->sa.rmap_cur))
|
|
return;
|
|
if (res.matches != 0)
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.rmap_cur, 0);
|
|
if (res.bad_non_owner_matches)
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.rmap_cur, 0);
|
|
}
|
|
|
|
/* xref check that the extent has no reverse mapping at all */
|
|
void
|
|
xchk_xref_has_no_owner(
|
|
struct xfs_scrub *sc,
|
|
xfs_agblock_t bno,
|
|
xfs_extlen_t len)
|
|
{
|
|
enum xbtree_recpacking outcome;
|
|
int error;
|
|
|
|
if (!sc->sa.rmap_cur || xchk_skip_xref(sc->sm))
|
|
return;
|
|
|
|
error = xfs_rmap_has_records(sc->sa.rmap_cur, bno, len, &outcome);
|
|
if (!xchk_should_check_xref(sc, &error, &sc->sa.rmap_cur))
|
|
return;
|
|
if (outcome != XBTREE_RECPACKING_EMPTY)
|
|
xchk_btree_xref_set_corrupt(sc, sc->sa.rmap_cur, 0);
|
|
}
|