xref: /openbmc/linux/fs/xfs/scrub/refcount.c (revision 52dc4b44af74196ded6413304542ead0257b5cda)
1edc09b52SDarrick J. Wong /*
2edc09b52SDarrick J. Wong  * Copyright (C) 2017 Oracle.  All Rights Reserved.
3edc09b52SDarrick J. Wong  *
4edc09b52SDarrick J. Wong  * Author: Darrick J. Wong <darrick.wong@oracle.com>
5edc09b52SDarrick J. Wong  *
6edc09b52SDarrick J. Wong  * This program is free software; you can redistribute it and/or
7edc09b52SDarrick J. Wong  * modify it under the terms of the GNU General Public License
8edc09b52SDarrick J. Wong  * as published by the Free Software Foundation; either version 2
9edc09b52SDarrick J. Wong  * of the License, or (at your option) any later version.
10edc09b52SDarrick J. Wong  *
11edc09b52SDarrick J. Wong  * This program is distributed in the hope that it would be useful,
12edc09b52SDarrick J. Wong  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13edc09b52SDarrick J. Wong  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14edc09b52SDarrick J. Wong  * GNU General Public License for more details.
15edc09b52SDarrick J. Wong  *
16edc09b52SDarrick J. Wong  * You should have received a copy of the GNU General Public License
17edc09b52SDarrick J. Wong  * along with this program; if not, write the Free Software Foundation,
18edc09b52SDarrick J. Wong  * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301, USA.
19edc09b52SDarrick J. Wong  */
20edc09b52SDarrick J. Wong #include "xfs.h"
21edc09b52SDarrick J. Wong #include "xfs_fs.h"
22edc09b52SDarrick J. Wong #include "xfs_shared.h"
23edc09b52SDarrick J. Wong #include "xfs_format.h"
24edc09b52SDarrick J. Wong #include "xfs_trans_resv.h"
25edc09b52SDarrick J. Wong #include "xfs_mount.h"
26edc09b52SDarrick J. Wong #include "xfs_defer.h"
27edc09b52SDarrick J. Wong #include "xfs_btree.h"
28edc09b52SDarrick J. Wong #include "xfs_bit.h"
29edc09b52SDarrick J. Wong #include "xfs_log_format.h"
30edc09b52SDarrick J. Wong #include "xfs_trans.h"
31edc09b52SDarrick J. Wong #include "xfs_sb.h"
32edc09b52SDarrick J. Wong #include "xfs_alloc.h"
33edc09b52SDarrick J. Wong #include "xfs_rmap.h"
34edc09b52SDarrick J. Wong #include "scrub/xfs_scrub.h"
35edc09b52SDarrick J. Wong #include "scrub/scrub.h"
36edc09b52SDarrick J. Wong #include "scrub/common.h"
37edc09b52SDarrick J. Wong #include "scrub/btree.h"
38edc09b52SDarrick J. Wong #include "scrub/trace.h"
39edc09b52SDarrick J. Wong 
40edc09b52SDarrick J. Wong /*
41edc09b52SDarrick J. Wong  * Set us up to scrub reference count btrees.
42edc09b52SDarrick J. Wong  */
43edc09b52SDarrick J. Wong int
44edc09b52SDarrick J. Wong xfs_scrub_setup_ag_refcountbt(
45edc09b52SDarrick J. Wong 	struct xfs_scrub_context	*sc,
46edc09b52SDarrick J. Wong 	struct xfs_inode		*ip)
47edc09b52SDarrick J. Wong {
48edc09b52SDarrick J. Wong 	return xfs_scrub_setup_ag_btree(sc, ip, false);
49edc09b52SDarrick J. Wong }
50edc09b52SDarrick J. Wong 
51edc09b52SDarrick J. Wong /* Reference count btree scrubber. */
52edc09b52SDarrick J. Wong 
53166d7641SDarrick J. Wong /* Cross-reference with the other btrees. */
54166d7641SDarrick J. Wong STATIC void
55166d7641SDarrick J. Wong xfs_scrub_refcountbt_xref(
56166d7641SDarrick J. Wong 	struct xfs_scrub_context	*sc,
57166d7641SDarrick J. Wong 	xfs_agblock_t			agbno,
58166d7641SDarrick J. Wong 	xfs_extlen_t			len,
59166d7641SDarrick J. Wong 	xfs_nlink_t			refcount)
60166d7641SDarrick J. Wong {
61166d7641SDarrick J. Wong 	if (sc->sm->sm_flags & XFS_SCRUB_OFLAG_CORRUPT)
62166d7641SDarrick J. Wong 		return;
63*52dc4b44SDarrick J. Wong 
64*52dc4b44SDarrick J. Wong 	xfs_scrub_xref_is_used_space(sc, agbno, len);
65166d7641SDarrick J. Wong }
66166d7641SDarrick J. Wong 
67edc09b52SDarrick J. Wong /* Scrub a refcountbt record. */
68edc09b52SDarrick J. Wong STATIC int
69edc09b52SDarrick J. Wong xfs_scrub_refcountbt_rec(
70edc09b52SDarrick J. Wong 	struct xfs_scrub_btree		*bs,
71edc09b52SDarrick J. Wong 	union xfs_btree_rec		*rec)
72edc09b52SDarrick J. Wong {
73edc09b52SDarrick J. Wong 	struct xfs_mount		*mp = bs->cur->bc_mp;
74edc09b52SDarrick J. Wong 	xfs_agnumber_t			agno = bs->cur->bc_private.a.agno;
75edc09b52SDarrick J. Wong 	xfs_agblock_t			bno;
76edc09b52SDarrick J. Wong 	xfs_extlen_t			len;
77edc09b52SDarrick J. Wong 	xfs_nlink_t			refcount;
78edc09b52SDarrick J. Wong 	bool				has_cowflag;
79edc09b52SDarrick J. Wong 	int				error = 0;
80edc09b52SDarrick J. Wong 
81edc09b52SDarrick J. Wong 	bno = be32_to_cpu(rec->refc.rc_startblock);
82edc09b52SDarrick J. Wong 	len = be32_to_cpu(rec->refc.rc_blockcount);
83edc09b52SDarrick J. Wong 	refcount = be32_to_cpu(rec->refc.rc_refcount);
84edc09b52SDarrick J. Wong 
85edc09b52SDarrick J. Wong 	/* Only CoW records can have refcount == 1. */
86edc09b52SDarrick J. Wong 	has_cowflag = (bno & XFS_REFC_COW_START);
87edc09b52SDarrick J. Wong 	if ((refcount == 1 && !has_cowflag) || (refcount != 1 && has_cowflag))
88edc09b52SDarrick J. Wong 		xfs_scrub_btree_set_corrupt(bs->sc, bs->cur, 0);
89edc09b52SDarrick J. Wong 
90edc09b52SDarrick J. Wong 	/* Check the extent. */
91edc09b52SDarrick J. Wong 	bno &= ~XFS_REFC_COW_START;
92edc09b52SDarrick J. Wong 	if (bno + len <= bno ||
93edc09b52SDarrick J. Wong 	    !xfs_verify_agbno(mp, agno, bno) ||
94edc09b52SDarrick J. Wong 	    !xfs_verify_agbno(mp, agno, bno + len - 1))
95edc09b52SDarrick J. Wong 		xfs_scrub_btree_set_corrupt(bs->sc, bs->cur, 0);
96edc09b52SDarrick J. Wong 
97edc09b52SDarrick J. Wong 	if (refcount == 0)
98edc09b52SDarrick J. Wong 		xfs_scrub_btree_set_corrupt(bs->sc, bs->cur, 0);
99edc09b52SDarrick J. Wong 
100166d7641SDarrick J. Wong 	xfs_scrub_refcountbt_xref(bs->sc, bno, len, refcount);
101166d7641SDarrick J. Wong 
102edc09b52SDarrick J. Wong 	return error;
103edc09b52SDarrick J. Wong }
104edc09b52SDarrick J. Wong 
105edc09b52SDarrick J. Wong /* Scrub the refcount btree for some AG. */
106edc09b52SDarrick J. Wong int
107edc09b52SDarrick J. Wong xfs_scrub_refcountbt(
108edc09b52SDarrick J. Wong 	struct xfs_scrub_context	*sc)
109edc09b52SDarrick J. Wong {
110edc09b52SDarrick J. Wong 	struct xfs_owner_info		oinfo;
111edc09b52SDarrick J. Wong 
112edc09b52SDarrick J. Wong 	xfs_rmap_ag_owner(&oinfo, XFS_RMAP_OWN_REFC);
113edc09b52SDarrick J. Wong 	return xfs_scrub_btree(sc, sc->sa.refc_cur, xfs_scrub_refcountbt_rec,
114edc09b52SDarrick J. Wong 			&oinfo, NULL);
115edc09b52SDarrick J. Wong }
116