Plumb in the pieces necessary to check the free space btrees. Signed-off-by: Darrick J. Wong <darrick.wong@xxxxxxxxxx> --- libxfs/libxfs_priv.h | 1 libxfs/xfs_alloc.c | 98 ++++++++++++++++++++++++++++++++++++++++++++++ libxfs/xfs_alloc.h | 3 + libxfs/xfs_alloc_btree.c | 51 +++++++++++++++++++++--- 4 files changed, 147 insertions(+), 6 deletions(-) diff --git a/libxfs/libxfs_priv.h b/libxfs/libxfs_priv.h index 527bd49..dabdafc 100644 --- a/libxfs/libxfs_priv.h +++ b/libxfs/libxfs_priv.h @@ -518,5 +518,6 @@ int libxfs_zero_extent(struct xfs_inode *ip, xfs_fsblock_t start_fsb, bool xfs_log_check_lsn(struct xfs_mount *, xfs_lsn_t); #define xfs_always_cow (false) +#define xfs_err(...) ((void)0) #endif /* __LIBXFS_INTERNAL_XFS_H__ */ diff --git a/libxfs/xfs_alloc.c b/libxfs/xfs_alloc.c index 7ab05e3..7219057 100644 --- a/libxfs/xfs_alloc.c +++ b/libxfs/xfs_alloc.c @@ -35,6 +35,7 @@ #include "xfs_trans.h" #include "xfs_ag_resv.h" #include "xfs_refcount_btree.h" +#include "xfs_scrub.h" struct workqueue_struct *xfs_alloc_wq; @@ -2953,3 +2954,100 @@ xfs_alloc_record_exists( *is_freesp = (fbno <= bno && fbno + flen >= bno + len); return 0; } + +STATIC int +xfs_allocbt_scrub_rmap_check( + struct xfs_btree_cur *cur, + struct xfs_rmap_irec *rec, + void *priv) +{ + xfs_err(cur->bc_mp, "%s: freespace in rmapbt! %u/%u %u %lld %lld %x", + __func__, cur->bc_private.a.agno, rec->rm_startblock, + rec->rm_blockcount, rec->rm_owner, rec->rm_offset, + rec->rm_flags); + return XFS_BTREE_QUERY_RANGE_ABORT; +} + +STATIC int +xfs_allocbt_scrub_helper( + struct xfs_btree_scrub *bs, + union xfs_btree_rec *rec) +{ + struct xfs_mount *mp = bs->cur->bc_mp; + xfs_agblock_t bno; + xfs_extlen_t len; + struct xfs_rmap_irec low; + struct xfs_rmap_irec high; + bool no_rmap; + int error; + + bno = be32_to_cpu(rec->alloc.ar_startblock); + len = be32_to_cpu(rec->alloc.ar_blockcount); + + XFS_BTREC_SCRUB_CHECK(bs, bno <= mp->m_sb.sb_agblocks); + XFS_BTREC_SCRUB_CHECK(bs, bno < bno + len); + XFS_BTREC_SCRUB_CHECK(bs, (unsigned long long)bno + len <= + mp->m_sb.sb_agblocks); + + /* if rmapbt, make sure there's no record */ + if (!bs->rmap_cur) + return 0; + + memset(&low, 0, sizeof(low)); + low.rm_startblock = bno; + memset(&high, 0xFF, sizeof(high)); + high.rm_startblock = bno + len - 1; + + error = xfs_rmapbt_query_range(bs->rmap_cur, &low, &high, + &xfs_allocbt_scrub_rmap_check, NULL); + if (error && error != XFS_BTREE_QUERY_RANGE_ABORT) + goto err; + no_rmap = error == 0; + XFS_BTREC_SCRUB_CHECK(bs, no_rmap); +err: + return error; +} + +/* Scrub the freespace btrees for some AG. */ +STATIC int +xfs_allocbt_scrub( + struct xfs_mount *mp, + xfs_agnumber_t agno, + int which) +{ + struct xfs_btree_scrub bs; + int error; + + error = xfs_alloc_read_agf(mp, NULL, agno, 0, &bs.agf_bp); + if (error) + return error; + + bs.cur = xfs_allocbt_init_cursor(mp, NULL, bs.agf_bp, agno, which); + bs.scrub_rec = xfs_allocbt_scrub_helper; + xfs_rmap_ag_owner(&bs.oinfo, XFS_RMAP_OWN_AG); + error = xfs_btree_scrub(&bs); + xfs_btree_del_cursor(bs.cur, + error ? XFS_BTREE_ERROR : XFS_BTREE_NOERROR); + xfs_trans_brelse(NULL, bs.agf_bp); + + if (!error && bs.error) + error = bs.error; + + return error; +} + +int +xfs_bnobt_scrub( + struct xfs_mount *mp, + xfs_agnumber_t agno) +{ + return xfs_allocbt_scrub(mp, agno, XFS_BTNUM_BNO); +} + +int +xfs_cntbt_scrub( + struct xfs_mount *mp, + xfs_agnumber_t agno) +{ + return xfs_allocbt_scrub(mp, agno, XFS_BTNUM_CNT); +} diff --git a/libxfs/xfs_alloc.h b/libxfs/xfs_alloc.h index 4f2ce38..f1fcc7e 100644 --- a/libxfs/xfs_alloc.h +++ b/libxfs/xfs_alloc.h @@ -213,4 +213,7 @@ xfs_extlen_t xfs_prealloc_blocks(struct xfs_mount *mp); int xfs_alloc_record_exists(struct xfs_btree_cur *cur, xfs_agblock_t bno, xfs_extlen_t len, bool *is_freesp); +int xfs_bnobt_scrub(struct xfs_mount *mp, xfs_agnumber_t agno); +int xfs_cntbt_scrub(struct xfs_mount *mp, xfs_agnumber_t agno); + #endif /* __XFS_ALLOC_H__ */ diff --git a/libxfs/xfs_alloc_btree.c b/libxfs/xfs_alloc_btree.c index ff4bae4..1794791 100644 --- a/libxfs/xfs_alloc_btree.c +++ b/libxfs/xfs_alloc_btree.c @@ -254,6 +254,26 @@ xfs_allocbt_key_diff( return (__int64_t)be32_to_cpu(kp->ar_startblock) - rec->ar_startblock; } +STATIC __int64_t +xfs_bnobt_diff_two_keys( + struct xfs_btree_cur *cur, + union xfs_btree_key *k1, + union xfs_btree_key *k2) +{ + return (__int64_t)be32_to_cpu(k2->alloc.ar_startblock) - + be32_to_cpu(k1->alloc.ar_startblock); +} + +STATIC __int64_t +xfs_cntbt_diff_two_keys( + struct xfs_btree_cur *cur, + union xfs_btree_key *k1, + union xfs_btree_key *k2) +{ + return (__int64_t)be32_to_cpu(k2->alloc.ar_blockcount) - + be32_to_cpu(k1->alloc.ar_blockcount); +} + static bool xfs_allocbt_verify( struct xfs_buf *bp) @@ -342,7 +362,6 @@ const struct xfs_buf_ops xfs_allocbt_buf_ops = { }; -#if defined(DEBUG) || defined(XFS_WARN) STATIC int xfs_allocbt_keys_inorder( struct xfs_btree_cur *cur, @@ -379,9 +398,29 @@ xfs_allocbt_recs_inorder( be32_to_cpu(r2->alloc.ar_startblock)); } } -#endif /* DEBUG */ -static const struct xfs_btree_ops xfs_allocbt_ops = { +static const struct xfs_btree_ops xfs_bnobt_ops = { + .rec_len = sizeof(xfs_alloc_rec_t), + .key_len = sizeof(xfs_alloc_key_t), + + .dup_cursor = xfs_allocbt_dup_cursor, + .set_root = xfs_allocbt_set_root, + .alloc_block = xfs_allocbt_alloc_block, + .free_block = xfs_allocbt_free_block, + .update_lastrec = xfs_allocbt_update_lastrec, + .get_minrecs = xfs_allocbt_get_minrecs, + .get_maxrecs = xfs_allocbt_get_maxrecs, + .init_key_from_rec = xfs_allocbt_init_key_from_rec, + .init_rec_from_cur = xfs_allocbt_init_rec_from_cur, + .init_ptr_from_cur = xfs_allocbt_init_ptr_from_cur, + .key_diff = xfs_allocbt_key_diff, + .buf_ops = &xfs_allocbt_buf_ops, + .diff_two_keys = xfs_bnobt_diff_two_keys, + .keys_inorder = xfs_allocbt_keys_inorder, + .recs_inorder = xfs_allocbt_recs_inorder, +}; + +static const struct xfs_btree_ops xfs_cntbt_ops = { .rec_len = sizeof(xfs_alloc_rec_t), .key_len = sizeof(xfs_alloc_key_t), @@ -397,10 +436,9 @@ static const struct xfs_btree_ops xfs_allocbt_ops = { .init_ptr_from_cur = xfs_allocbt_init_ptr_from_cur, .key_diff = xfs_allocbt_key_diff, .buf_ops = &xfs_allocbt_buf_ops, -#if defined(DEBUG) || defined(XFS_WARN) + .diff_two_keys = xfs_cntbt_diff_two_keys, .keys_inorder = xfs_allocbt_keys_inorder, .recs_inorder = xfs_allocbt_recs_inorder, -#endif }; /* @@ -425,12 +463,13 @@ xfs_allocbt_init_cursor( cur->bc_mp = mp; cur->bc_btnum = btnum; cur->bc_blocklog = mp->m_sb.sb_blocklog; - cur->bc_ops = &xfs_allocbt_ops; if (btnum == XFS_BTNUM_CNT) { + cur->bc_ops = &xfs_cntbt_ops; cur->bc_nlevels = be32_to_cpu(agf->agf_levels[XFS_BTNUM_CNT]); cur->bc_flags = XFS_BTREE_LASTREC_UPDATE; } else { + cur->bc_ops = &xfs_bnobt_ops; cur->bc_nlevels = be32_to_cpu(agf->agf_levels[XFS_BTNUM_BNO]); } _______________________________________________ xfs mailing list xfs@xxxxxxxxxxx http://oss.sgi.com/mailman/listinfo/xfs