}
void
-scanfunc_bno(
+scanfunc_allocbt(
struct xfs_btree_block *block,
int level,
xfs_agblock_t bno,
xfs_agnumber_t agno,
int suspect,
- int isroot
- )
+ int isroot,
+ __uint32_t magic)
{
+ const char *name;
xfs_agblock_t b, e;
int i;
xfs_alloc_ptr_t *pp;
int numrecs;
int state;
- if (be32_to_cpu(block->bb_magic) != XFS_ABTB_MAGIC) {
- do_warn(_("bad magic # %#x in btbno block %d/%d\n"),
- be32_to_cpu(block->bb_magic), agno, bno);
+ assert(magic == XFS_ABTB_MAGIC || magic == XFS_ABTC_MAGIC);
+
+ name = (magic == XFS_ABTB_MAGIC) ? "bno" : "cnt";
+
+ if (be32_to_cpu(block->bb_magic) != magic) {
+ do_warn(_("bad magic # %#x in bt%s block %d/%d\n"),
+ be32_to_cpu(block->bb_magic), name, agno, bno);
hdr_errors++;
if (suspect)
return;
}
if (be16_to_cpu(block->bb_level) != level) {
- do_warn(_("expected level %d got %d in btbno block %d/%d\n"),
- level, be16_to_cpu(block->bb_level), agno, bno);
+ do_warn(_("expected level %d got %d in bt%s block %d/%d\n"),
+ level, be16_to_cpu(block->bb_level), name, agno, bno);
hdr_errors++;
if (suspect)
return;
default:
set_agbno_state(mp, agno, bno, XR_E_MULT);
do_warn(
-_("bno freespace btree block claimed (state %d), agno %d, bno %d, suspect %d\n"),
- state, agno, bno, suspect);
+_("%s freespace btree block claimed (state %d), agno %d, bno %d, suspect %d\n"),
+ name, state, agno, bno, suspect);
return;
}
continue;
for (b = be32_to_cpu(rp[i].ar_startblock);
b < e; b++) {
- if (get_agbno_state(mp, agno, b)
- == XR_E_UNKNOWN)
+ state = get_agbno_state(mp, agno, b);
+ switch (state) {
+ case XR_E_UNKNOWN:
set_agbno_state(mp, agno, b,
XR_E_FREE1);
- else {
+ break;
+ case XR_E_FREE1:
+ /*
+ * no warning messages -- we'll catch
+ * FREE1 blocks later
+ */
+ if (magic == XFS_ABTC_MAGIC) {
+ set_agbno_state(mp, agno, b,
+ XR_E_FREE);
+ break;
+ }
+ default:
do_warn(
- _("block (%d,%d) multiply claimed by bno space tree, state - %d\n"),
- agno, b,
- get_agbno_state(mp, agno, b));
+ _("block (%d,%d) multiply claimed by %s space tree, state - %d\n"),
+ agno, b, name, state);
+ break;
}
}
}
*/
if (be32_to_cpu(pp[i]) != 0 && verify_agbno(mp, agno,
be32_to_cpu(pp[i])))
- scan_sbtree(be32_to_cpu(pp[i]), level, agno,
- suspect, scanfunc_bno, 0);
+ scan_sbtree(be32_to_cpu(pp[i]), level, agno, suspect,
+ (magic == XFS_ABTB_MAGIC) ?
+ scanfunc_bno : scanfunc_cnt, 0);
}
}
+void
+scanfunc_bno(
+ struct xfs_btree_block *block,
+ int level,
+ xfs_agblock_t bno,
+ xfs_agnumber_t agno,
+ int suspect,
+ int isroot)
+{
+ return scanfunc_allocbt(block, level, bno, agno,
+ suspect, isroot, XFS_ABTB_MAGIC);
+}
+
void
scanfunc_cnt(
struct xfs_btree_block *block,
int isroot
)
{
- xfs_alloc_ptr_t *pp;
- xfs_alloc_rec_t *rp;
- xfs_agblock_t b, e;
- int i;
- int hdr_errors;
- int numrecs;
- int state;
-
- hdr_errors = 0;
-
- if (be32_to_cpu(block->bb_magic) != XFS_ABTC_MAGIC) {
- do_warn(_("bad magic # %#x in btcnt block %d/%d\n"),
- be32_to_cpu(block->bb_magic), agno, bno);
- hdr_errors++;
- if (suspect)
- return;
- }
- if (be16_to_cpu(block->bb_level) != level) {
- do_warn(_("expected level %d got %d in btcnt block %d/%d\n"),
- level, be16_to_cpu(block->bb_level), agno, bno);
- hdr_errors++;
- if (suspect)
- return;
- }
-
- /*
- * check for btree blocks multiply claimed
- */
- state = get_agbno_state(mp, agno, bno);
-
- switch (state) {
- case XR_E_UNKNOWN:
- set_agbno_state(mp, agno, bno, XR_E_FS_MAP);
- break;
- default:
- set_agbno_state(mp, agno, bno, XR_E_MULT);
- do_warn(
-_("bcnt freespace btree block claimed (state %d), agno %d, bno %d, suspect %d\n"),
- state, agno, bno, suspect);
- return;
- }
-
- numrecs = be16_to_cpu(block->bb_numrecs);
-
- if (level == 0) {
- if (numrecs > mp->m_alloc_mxr[0]) {
- numrecs = mp->m_alloc_mxr[0];
- hdr_errors++;
- }
- if (isroot == 0 && numrecs < mp->m_alloc_mnr[0]) {
- numrecs = mp->m_alloc_mnr[0];
- hdr_errors++;
- }
-
- if (hdr_errors)
- suspect++;
-
- rp = XFS_ALLOC_REC_ADDR(mp, block, 1);
- for (i = 0; i < numrecs; i++) {
- if (be32_to_cpu(rp[i].ar_blockcount) == 0 ||
- be32_to_cpu(rp[i].ar_startblock) == 0 ||
- !verify_agbno(mp, agno, be32_to_cpu(
- rp[i].ar_startblock)) ||
- be32_to_cpu(rp[i].ar_blockcount) >
- MAXEXTLEN)
- continue;
-
- e = be32_to_cpu(rp[i].ar_startblock) +
- be32_to_cpu(rp[i].ar_blockcount);
- if (!verify_agbno(mp, agno, e - 1))
- continue;
- for (b = be32_to_cpu(rp[i].ar_startblock); b < e; b++) {
- state = get_agbno_state(mp, agno, b);
- /*
- * no warning messages -- we'll catch
- * FREE1 blocks later
- */
- switch (state) {
- case XR_E_FREE1:
- set_agbno_state(mp, agno, b, XR_E_FREE);
- break;
- case XR_E_UNKNOWN:
- set_agbno_state(mp, agno, b,
- XR_E_FREE1);
- break;
- default:
- do_warn(
- _("block (%d,%d) already used, state %d\n"),
- agno, b, state);
- break;
- }
- }
- }
- return;
- }
-
- /*
- * interior record
- */
- pp = XFS_ALLOC_PTR_ADDR(mp, block, 1, mp->m_alloc_mxr[1]);
-
- if (numrecs > mp->m_alloc_mxr[1]) {
- numrecs = mp->m_alloc_mxr[1];
- hdr_errors++;
- }
- if (isroot == 0 && numrecs < mp->m_alloc_mnr[1]) {
- numrecs = mp->m_alloc_mnr[1];
- hdr_errors++;
- }
-
- /*
- * don't pass bogus tree flag down further if this block
- * looked ok. bail out if two levels in a row look bad.
- */
-
- if (suspect && !hdr_errors)
- suspect = 0;
-
- if (hdr_errors) {
- if (suspect)
- return;
- else suspect++;
- }
-
- for (i = 0; i < numrecs; i++) {
- if (be32_to_cpu(pp[i]) != 0 && verify_agbno(mp, agno,
- be32_to_cpu(pp[i])))
- scan_sbtree(be32_to_cpu(pp[i]), level, agno,
- suspect, scanfunc_cnt, 0);
- }
+ return scanfunc_allocbt(block, level, bno, agno,
+ suspect, isroot, XFS_ABTC_MAGIC);
}
/*