From patchwork Sat Jan 21 08:04:29 2017 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: "Darrick J. Wong" X-Patchwork-Id: 9529963 Return-Path: Received: from mail.wl.linuxfoundation.org (pdx-wl-mail.web.codeaurora.org [172.30.200.125]) by pdx-korg-patchwork.web.codeaurora.org (Postfix) with ESMTP id 0B3C46020B for ; Sat, 21 Jan 2017 08:04:35 +0000 (UTC) Received: from mail.wl.linuxfoundation.org (localhost [127.0.0.1]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id EDF8F28427 for ; Sat, 21 Jan 2017 08:04:34 +0000 (UTC) Received: by mail.wl.linuxfoundation.org (Postfix, from userid 486) id E2FBD28698; Sat, 21 Jan 2017 08:04:34 +0000 (UTC) X-Spam-Checker-Version: SpamAssassin 3.3.1 (2010-03-16) on pdx-wl-mail.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-6.9 required=2.0 tests=BAYES_00, RCVD_IN_DNSWL_HI, UNPARSEABLE_RELAY autolearn=unavailable version=3.3.1 Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by mail.wl.linuxfoundation.org (Postfix) with ESMTP id 3A63C28427 for ; Sat, 21 Jan 2017 08:04:34 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1751185AbdAUIEc (ORCPT ); Sat, 21 Jan 2017 03:04:32 -0500 Received: from aserp1040.oracle.com ([141.146.126.69]:19879 "EHLO aserp1040.oracle.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1750894AbdAUIEc (ORCPT ); Sat, 21 Jan 2017 03:04:32 -0500 Received: from aserv0021.oracle.com (aserv0021.oracle.com [141.146.126.233]) by aserp1040.oracle.com (Sentrion-MTA-4.3.2/Sentrion-MTA-4.3.2) with ESMTP id v0L84U0p014449 (version=TLSv1 cipher=DHE-RSA-AES256-SHA bits=256 verify=OK); Sat, 21 Jan 2017 08:04:31 GMT Received: from aserv0122.oracle.com (aserv0122.oracle.com [141.146.126.236]) by aserv0021.oracle.com (8.13.8/8.14.4) with ESMTP id v0L84Ug7013989 (version=TLSv1/SSLv3 cipher=DHE-RSA-AES256-SHA bits=256 verify=OK); Sat, 21 Jan 2017 08:04:30 GMT Received: from abhmp0014.oracle.com (abhmp0014.oracle.com [141.146.116.20]) by aserv0122.oracle.com (8.14.4/8.14.4) with ESMTP id v0L84UeV031060; Sat, 21 Jan 2017 08:04:30 GMT Received: from localhost (/24.21.211.40) by default (Oracle Beehive Gateway v4.0) with ESMTP ; Sat, 21 Jan 2017 00:04:30 -0800 Subject: [PATCH 38/55] xfs: cross-reference refcount btree during scrub From: "Darrick J. Wong" To: darrick.wong@oracle.com Cc: linux-xfs@vger.kernel.org, linux-fsdevel@vger.kernel.org Date: Sat, 21 Jan 2017 00:04:29 -0800 Message-ID: <148498586908.15323.6778945753466221760.stgit@birch.djwong.org> In-Reply-To: <148498561504.15323.8531512066874274553.stgit@birch.djwong.org> References: <148498561504.15323.8531512066874274553.stgit@birch.djwong.org> User-Agent: StGit/0.17.1-dirty MIME-Version: 1.0 X-Source-IP: aserv0021.oracle.com [141.146.126.233] Sender: linux-fsdevel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-fsdevel@vger.kernel.org X-Virus-Scanned: ClamAV using ClamSMTP During metadata btree scrub, we should cross-reference with the reference counts. Signed-off-by: Darrick J. Wong --- fs/xfs/libxfs/xfs_refcount.c | 19 +++++++++++++ fs/xfs/libxfs/xfs_refcount.h | 3 ++ fs/xfs/scrub/agheader.c | 52 ++++++++++++++++++++++++++++++++++++ fs/xfs/scrub/alloc.c | 10 +++++++ fs/xfs/scrub/bmap.c | 57 ++++++++++++++++++++++++++++++++++++++++ fs/xfs/scrub/ialloc.c | 10 +++++++ fs/xfs/scrub/rmap.c | 60 ++++++++++++++++++++++++++++++++++++++++++ 7 files changed, 211 insertions(+) -- To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html diff --git a/fs/xfs/libxfs/xfs_refcount.c b/fs/xfs/libxfs/xfs_refcount.c index b177ef3..c6c875d 100644 --- a/fs/xfs/libxfs/xfs_refcount.c +++ b/fs/xfs/libxfs/xfs_refcount.c @@ -1696,3 +1696,22 @@ xfs_refcount_recover_cow_leftovers( xfs_trans_cancel(tp); goto out_free; } + +/* Is there a record covering a given extent? */ +int +xfs_refcount_has_record( + struct xfs_btree_cur *cur, + xfs_agblock_t bno, + xfs_extlen_t len, + bool *exists) +{ + union xfs_btree_irec low; + union xfs_btree_irec high; + + memset(&low, 0, sizeof(low)); + low.rc.rc_startblock = bno; + memset(&high, 0xFF, sizeof(high)); + high.rc.rc_startblock = bno + len - 1; + + return xfs_btree_has_record(cur, &low, &high, exists); +} diff --git a/fs/xfs/libxfs/xfs_refcount.h b/fs/xfs/libxfs/xfs_refcount.h index 098dc66..78cb142 100644 --- a/fs/xfs/libxfs/xfs_refcount.h +++ b/fs/xfs/libxfs/xfs_refcount.h @@ -67,4 +67,7 @@ extern int xfs_refcount_free_cow_extent(struct xfs_mount *mp, extern int xfs_refcount_recover_cow_leftovers(struct xfs_mount *mp, xfs_agnumber_t agno); +extern int xfs_refcount_has_record(struct xfs_btree_cur *cur, + xfs_agblock_t bno, xfs_extlen_t len, bool *exists); + #endif /* __XFS_REFCOUNT_H__ */ diff --git a/fs/xfs/scrub/agheader.c b/fs/xfs/scrub/agheader.c index cf0093a..d94129c 100644 --- a/fs/xfs/scrub/agheader.c +++ b/fs/xfs/scrub/agheader.c @@ -34,6 +34,7 @@ #include "xfs_alloc.h" #include "xfs_ialloc.h" #include "xfs_rmap.h" +#include "xfs_refcount.h" #include "scrub/common.h" /* Set us up to check an AG header. */ @@ -166,6 +167,7 @@ xfs_scrub_superblock( bool is_freesp; bool has_inodes; bool has_rmap; + bool has_refcount; int error; int err2; @@ -336,6 +338,14 @@ xfs_scrub_superblock( XFS_SCRUB_SB_XCHECK(has_rmap); } + /* Cross-reference with the refcountbt. */ + if (psa->refc_cur) { + err2 = xfs_refcount_has_record(psa->refc_cur, XFS_SB_BLOCK(mp), + 1, &has_refcount); + if (xfs_scrub_should_xref(sc, err2, &psa->refc_cur)) + XFS_SCRUB_SB_XCHECK(!has_refcount); + } + out: return error; } @@ -388,6 +398,7 @@ xfs_scrub_agf( bool is_freesp; bool has_inodes; bool has_rmap; + bool has_refcount; int have; int level; int error = 0; @@ -554,6 +565,20 @@ xfs_scrub_agf( agf->agf_btreeblks)); } + /* Cross-reference with the refcountbt. */ + if (psa->refc_cur) { + err2 = xfs_refcount_has_record(psa->refc_cur, XFS_AGF_BLOCK(mp), + 1, &has_refcount); + if (xfs_scrub_should_xref(sc, err2, &psa->refc_cur)) + XFS_SCRUB_AGF_XCHECK(!has_refcount); + } + if (psa->refc_cur) { + err2 = xfs_btree_count_blocks(psa->refc_cur, &blocks); + if (xfs_scrub_should_xref(sc, err2, &psa->refc_cur)) + XFS_SCRUB_AGF_XCHECK(blocks == be32_to_cpu( + agf->agf_refcount_blocks)); + } + out: return error; } @@ -586,6 +611,7 @@ xfs_scrub_agfl_block( bool is_freesp; bool has_inodes; bool has_rmap; + bool has_refcount; int err2; XFS_SCRUB_AGFL_CHECK(agbno > XFS_AGI_BLOCK(mp)); @@ -628,6 +654,14 @@ xfs_scrub_agfl_block( XFS_SCRUB_AGFL_XCHECK(has_rmap); } + /* Cross-reference with the refcountbt. */ + if (sc->sa.refc_cur) { + err2 = xfs_refcount_has_record(sc->sa.refc_cur, agbno, 1, + &has_refcount); + if (xfs_scrub_should_xref(sc, err2, &sc->sa.refc_cur)) + XFS_SCRUB_AGFL_XCHECK(!has_refcount); + } + return 0; } @@ -645,6 +679,7 @@ xfs_scrub_agfl( bool is_freesp; bool has_inodes; bool has_rmap; + bool has_refcount; int error; int err2; @@ -691,6 +726,14 @@ xfs_scrub_agfl( XFS_SCRUB_AGFL_XCHECK(has_rmap); } + /* Set up cross-reference with refcountbt. */ + if (sc->sa.refc_cur) { + err2 = xfs_refcount_has_record(sc->sa.refc_cur, + XFS_AGFL_BLOCK(mp), 1, &has_refcount); + if (xfs_scrub_should_xref(sc, err2, &sc->sa.refc_cur)) + XFS_SCRUB_AGFL_XCHECK(!has_refcount); + } + /* Check the blocks in the AGFL. */ xfs_rmap_ag_owner(&sagfl.oinfo, XFS_RMAP_OWN_AG); return xfs_scrub_walk_agfl(sc, xfs_scrub_agfl_block, &sagfl); @@ -732,6 +775,7 @@ xfs_scrub_agi( bool is_freesp; bool has_inodes; bool has_rmap; + bool has_refcount; int i; int level; int error = 0; @@ -852,6 +896,14 @@ xfs_scrub_agi( XFS_SCRUB_AGI_XCHECK(has_rmap); } + /* Cross-reference with the refcountbt. */ + if (psa->refc_cur) { + err2 = xfs_refcount_has_record(psa->refc_cur, XFS_AGI_BLOCK(mp), + 1, &has_refcount); + if (xfs_scrub_should_xref(sc, err2, &psa->refc_cur)) + XFS_SCRUB_AGI_XCHECK(!has_refcount); + } + out: return error; } diff --git a/fs/xfs/scrub/alloc.c b/fs/xfs/scrub/alloc.c index 2cad0e4..8d81875 100644 --- a/fs/xfs/scrub/alloc.c +++ b/fs/xfs/scrub/alloc.c @@ -33,6 +33,7 @@ #include "xfs_rmap.h" #include "xfs_alloc.h" #include "xfs_ialloc.h" +#include "xfs_refcount.h" #include "scrub/common.h" #include "scrub/btree.h" @@ -75,6 +76,7 @@ xfs_scrub_allocbt_helper( xfs_extlen_t len; bool has_rmap; bool has_inodes; + bool has_refcount; int has_otherrec; int error = 0; int err2; @@ -142,6 +144,14 @@ xfs_scrub_allocbt_helper( XFS_SCRUB_BTREC_XCHECK(bs, !has_rmap); } + /* Cross-reference with the refcountbt. */ + if (psa->refc_cur) { + err2 = xfs_refcount_has_record(psa->refc_cur, bno, len, + &has_refcount); + if (xfs_scrub_btree_should_xref(bs, err2, &psa->refc_cur)) + XFS_SCRUB_BTREC_XCHECK(bs, !has_refcount); + } + out: return error; } diff --git a/fs/xfs/scrub/bmap.c b/fs/xfs/scrub/bmap.c index 843fb3c..953f77b 100644 --- a/fs/xfs/scrub/bmap.c +++ b/fs/xfs/scrub/bmap.c @@ -38,6 +38,7 @@ #include "xfs_rmap.h" #include "xfs_alloc.h" #include "xfs_ialloc.h" +#include "xfs_refcount.h" #include "scrub/common.h" #include "scrub/btree.h" @@ -109,12 +110,17 @@ xfs_scrub_bmap_extent( xfs_agnumber_t agno; xfs_fsblock_t bno; struct xfs_rmap_irec rmap; + struct xfs_refcount_irec rc; uint64_t owner; xfs_fileoff_t offset; + xfs_agblock_t fbno; + xfs_extlen_t flen; bool is_freesp; bool has_inodes; + bool has_cowflag; unsigned int rflags; int has_rmap; + int has_refcount; int error = 0; int err2 = 0; @@ -271,6 +277,57 @@ xfs_scrub_bmap_extent( ; } + /* + * If this is a non-shared file on a reflink filesystem, + * check the refcountbt to see if the flag is wrong. + */ + if (sa.refc_cur) { + if (info->whichfork == XFS_COW_FORK) { + /* Check this CoW staging extent. */ + err2 = xfs_refcount_lookup_le(sa.refc_cur, + bno + XFS_REFC_COW_START, + &has_refcount); + if (xfs_scrub_should_xref(info->sc, err2, + &sa.refc_cur)) { + XFS_SCRUB_BMAP_XGOTO(has_refcount, + skip_refc_xref); + } else + goto skip_refc_xref; + + err2 = xfs_refcount_get_rec(sa.refc_cur, &rc, + &has_refcount); + if (xfs_scrub_should_xref(info->sc, err2, + &sa.refc_cur)) { + XFS_SCRUB_BMAP_XGOTO(has_refcount, + skip_refc_xref); + } else + goto skip_refc_xref; + + has_cowflag = !!(rc.rc_startblock & XFS_REFC_COW_START); + XFS_SCRUB_BMAP_XCHECK( + (rc.rc_refcount == 1 && has_cowflag) || + (rc.rc_refcount != 1 && !has_cowflag)); + rc.rc_startblock &= ~XFS_REFC_COW_START; + XFS_SCRUB_BMAP_XCHECK(rc.rc_startblock <= bno); + XFS_SCRUB_BMAP_XCHECK(rc.rc_startblock < + rc.rc_startblock + rc.rc_blockcount); + XFS_SCRUB_BMAP_XCHECK(bno + irec->br_blockcount <= + rc.rc_startblock + rc.rc_blockcount); + XFS_SCRUB_BMAP_XCHECK(rc.rc_refcount == 1); + } else { + /* If this is shared, the inode flag must be set. */ + err2 = xfs_refcount_find_shared(sa.refc_cur, bno, + irec->br_blockcount, &fbno, &flen, + false); + if (xfs_scrub_should_xref(info->sc, err2, + &sa.refc_cur)) + XFS_SCRUB_BMAP_XCHECK(flen == 0 || + xfs_is_reflink_inode(ip)); + } +skip_refc_xref: + ; + } + xfs_scrub_ag_free(&sa); out: info->lastoff = irec->br_startoff + irec->br_blockcount; diff --git a/fs/xfs/scrub/ialloc.c b/fs/xfs/scrub/ialloc.c index 772d97b..7fa4ebf 100644 --- a/fs/xfs/scrub/ialloc.c +++ b/fs/xfs/scrub/ialloc.c @@ -38,6 +38,7 @@ #include "xfs_log.h" #include "xfs_trans_priv.h" #include "xfs_alloc.h" +#include "xfs_refcount.h" #include "scrub/common.h" #include "scrub/btree.h" @@ -92,6 +93,7 @@ xfs_scrub_iallocbt_chunk( bool is_freesp; bool has_inodes; bool has_rmap; + bool has_refcount; int error = 0; int err2; @@ -148,6 +150,14 @@ xfs_scrub_iallocbt_chunk( XFS_SCRUB_BTREC_XCHECK(bs, has_rmap); } + /* Cross-reference with the refcountbt. */ + if (psa->refc_cur) { + err2 = xfs_refcount_has_record(psa->refc_cur, bno, + len, &has_refcount); + if (xfs_scrub_btree_should_xref(bs, err2, &psa->refc_cur)) + XFS_SCRUB_BTREC_XCHECK(bs, !has_refcount); + } + out: return error; } diff --git a/fs/xfs/scrub/rmap.c b/fs/xfs/scrub/rmap.c index 30ec492..9b7ed8b 100644 --- a/fs/xfs/scrub/rmap.c +++ b/fs/xfs/scrub/rmap.c @@ -33,6 +33,7 @@ #include "xfs_rmap.h" #include "xfs_alloc.h" #include "xfs_ialloc.h" +#include "xfs_refcount.h" #include "scrub/common.h" #include "scrub/btree.h" @@ -48,13 +49,18 @@ xfs_scrub_rmapbt_helper( struct xfs_agf *agf; struct xfs_scrub_ag *psa; struct xfs_rmap_irec irec; + struct xfs_refcount_irec crec; xfs_agblock_t eoag; + xfs_agblock_t fbno; + xfs_extlen_t flen; bool is_freesp; bool non_inode; bool is_unwritten; bool is_bmbt; bool is_attr; bool has_inodes; + bool has_cowflag; + int has_refcount; int error = 0; int err2; @@ -144,6 +150,60 @@ xfs_scrub_rmapbt_helper( !has_inodes); } + /* Cross-reference with the refcount btree. */ + if (psa->refc_cur) { + if (irec.rm_owner == XFS_RMAP_OWN_COW) { + /* Check this CoW staging extent. */ + err2 = xfs_refcount_lookup_le(psa->refc_cur, + irec.rm_startblock + XFS_REFC_COW_START, + &has_refcount); + if (xfs_scrub_btree_should_xref(bs, err2, + &psa->refc_cur)) { + XFS_SCRUB_BTREC_XGOTO(bs, has_refcount, + skip_refc_xref); + } else + goto skip_refc_xref; + + err2 = xfs_refcount_get_rec(psa->refc_cur, &crec, + &has_refcount); + if (xfs_scrub_btree_should_xref(bs, err2, + &psa->refc_cur)) { + XFS_SCRUB_BTREC_XGOTO(bs, has_refcount, + skip_refc_xref); + } else + goto skip_refc_xref; + + has_cowflag = !!(crec.rc_startblock & XFS_REFC_COW_START); + XFS_SCRUB_BTREC_XCHECK(bs, + (crec.rc_refcount == 1 && has_cowflag) || + (crec.rc_refcount != 1 && !has_cowflag)); + crec.rc_startblock &= ~XFS_REFC_COW_START; + XFS_SCRUB_BTREC_XCHECK(bs, crec.rc_startblock <= + irec.rm_startblock); + XFS_SCRUB_BTREC_XCHECK(bs, crec.rc_startblock + + crec.rc_blockcount > + crec.rc_startblock); + XFS_SCRUB_BTREC_XCHECK(bs, crec.rc_startblock + + crec.rc_blockcount >= + irec.rm_startblock + + irec.rm_blockcount); + XFS_SCRUB_BTREC_XCHECK(bs, + crec.rc_refcount == 1); + } else { + /* If this is shared, the inode flag must be set. */ + err2 = xfs_refcount_find_shared(psa->refc_cur, + irec.rm_startblock, irec.rm_blockcount, + &fbno, &flen, false); + if (xfs_scrub_btree_should_xref(bs, err2, + &psa->refc_cur)) + XFS_SCRUB_BTREC_XCHECK(bs, flen == 0 || + (!non_inode && !is_attr && + !is_bmbt && !is_unwritten)); + } +skip_refc_xref: + ; + } + out: return error; }