// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2018 Oracle. All Rights Reserved. * Author: Darrick J. Wong */ #ifndef __XFS_SCRUB_REPAIR_H__ #define __XFS_SCRUB_REPAIR_H__ #include "xfs_btree.h" #include "xfs_btree_staging.h" #include "xfs_quota_defs.h" #include "scrub/bitmap.h" union xfs_btree_ptr; static inline int xrep_notsupported(struct xfs_scrub *sc) { return -EOPNOTSUPP; } #ifdef CONFIG_XFS_ONLINE_REPAIR /* Repair helpers */ int xrep_attempt(struct xfs_scrub *sc); void xrep_failure(struct xfs_mount *mp); int xrep_roll_ag_trans(struct xfs_scrub *sc); int xrep_roll_trans(struct xfs_scrub *sc); bool xrep_ag_has_space(struct xfs_perag *pag, xfs_extlen_t nr_blocks, enum xfs_ag_resv_type type); xfs_extlen_t xrep_calc_ag_resblks(struct xfs_scrub *sc); int xrep_alloc_ag_block(struct xfs_scrub *sc, const struct xfs_owner_info *oinfo, xfs_fsblock_t *fsbno, enum xfs_ag_resv_type resv); int xrep_init_btblock(struct xfs_scrub *sc, xfs_fsblock_t fsb, struct xfs_buf **bpp, xfs_btnum_t btnum, const struct xfs_buf_ops *ops); static inline int xrep_trans_commit( struct xfs_scrub *sc) { int error = xfs_trans_commit(sc->tp); sc->tp = NULL; return error; } struct xbitmap; int xrep_fix_freelist(struct xfs_scrub *sc, int alloc_flags); int xrep_reap_extents(struct xfs_scrub *sc, struct xbitmap *exlist, const struct xfs_owner_info *oinfo, enum xfs_ag_resv_type type); struct xrep_find_ag_btree { /* in: rmap owner of the btree we're looking for */ uint64_t rmap_owner; /* in: buffer ops */ const struct xfs_buf_ops *buf_ops; /* out: the highest btree block found and the tree height */ xfs_agblock_t root; unsigned int height; }; int xrep_find_ag_btree_roots(struct xfs_scrub *sc, struct xfs_buf *agf_bp, struct xrep_find_ag_btree *btree_info, struct xfs_buf *agfl_bp); void xrep_force_quotacheck(struct xfs_scrub *sc, xfs_dqtype_t type); int xrep_ino_dqattach(struct xfs_scrub *sc); int xrep_reset_perag_resv(struct xfs_scrub *sc); int xrep_bmap(struct xfs_scrub *sc, int whichfork, bool allow_unwritten); int xrep_metadata_inode_forks(struct xfs_scrub *sc); int xrep_setup_ag_rmapbt(struct xfs_scrub *sc); int xrep_setup_rtsummary(struct xfs_scrub *sc, unsigned int *resblks); int xrep_setup_xattr(struct xfs_scrub *sc); int xrep_setup_directory(struct xfs_scrub *sc); int xrep_xattr_reset_fork(struct xfs_scrub *sc, struct xfs_inode *ip); void xrep_ag_btcur_init(struct xfs_scrub *sc, struct xchk_ag *sa); int xrep_ag_init(struct xfs_scrub *sc, struct xfs_perag *pag, struct xchk_ag *sa); /* Metadata revalidators */ int xrep_revalidate_allocbt(struct xfs_scrub *sc); int xrep_revalidate_iallocbt(struct xfs_scrub *sc); /* Buffer cache scan context. */ struct xrep_buf_scan { /* Disk address for the buffers we want to scan. */ xfs_daddr_t daddr; /* Maximum number of addresses to walk. */ xfs_daddr_t max_daddrs; /* Increment the length by this step. */ xfs_daddr_t daddr_step; /* Internal scan state; initialize to zero. */ xfs_daddr_t __sector_count; }; xfs_daddr_t xrep_max_buf_len(struct xfs_mount *mp, xfs_extlen_t fsblocks); struct xfs_buf *xrep_buf_scan_advance(struct xfs_mount *mp, struct xrep_buf_scan *scan); /* Metadata repairers */ int xrep_probe(struct xfs_scrub *sc); int xrep_superblock(struct xfs_scrub *sc); int xrep_agf(struct xfs_scrub *sc); int xrep_agfl(struct xfs_scrub *sc); int xrep_agi(struct xfs_scrub *sc); int xrep_allocbt(struct xfs_scrub *sc); int xrep_iallocbt(struct xfs_scrub *sc); int xrep_rmapbt(struct xfs_scrub *sc); int xrep_refcountbt(struct xfs_scrub *sc); int xrep_inode(struct xfs_scrub *sc); int xrep_bmap_data(struct xfs_scrub *sc); int xrep_bmap_attr(struct xfs_scrub *sc); int xrep_symlink(struct xfs_scrub *sc); int xrep_fscounters(struct xfs_scrub *sc); int xrep_xattr(struct xfs_scrub *sc); int xrep_directory(struct xfs_scrub *sc); int xrep_parent(struct xfs_scrub *sc); #ifdef CONFIG_XFS_QUOTA int xrep_quota(struct xfs_scrub *sc); int xrep_quotacheck(struct xfs_scrub *sc); #else # define xrep_quota xrep_notsupported # define xrep_quotacheck xrep_notsupported #endif /* CONFIG_XFS_QUOTA */ #ifdef CONFIG_XFS_RT int xrep_rtsummary(struct xfs_scrub *sc); #else # define xrep_rtsummary xrep_notsupported #endif /* CONFIG_XFS_RT */ struct xrep_newbt_resv { /* Link to list of extents that we've reserved. */ struct list_head list; struct xfs_log_item *efi; /* FSB of the block we reserved. */ xfs_fsblock_t fsbno; /* Length of the reservation. */ xfs_extlen_t len; /* How much of this reservation has been used. */ xfs_extlen_t used; }; struct xfs_alloc_arg; struct xrep_newbt { struct xfs_scrub *sc; /* Custom allocation function, or NULL for xfs_alloc_vextent */ int (*alloc_vextent)(struct xfs_scrub *sc, struct xfs_alloc_arg *args); /* List of extents that we've reserved. */ struct list_head resv_list; /* Fake root for new btree. */ union { struct xbtree_afakeroot afake; struct xbtree_ifakeroot ifake; }; /* rmap owner of these blocks */ struct xfs_owner_info oinfo; /* Allocation hint */ xfs_fsblock_t alloc_hint; /* per-ag reservation type */ enum xfs_ag_resv_type resv; }; #define for_each_xrep_newbt_reservation(xnr, resv, n) \ list_for_each_entry_safe((resv), (n), &(xnr)->resv_list, list) void xrep_newbt_init_bare(struct xrep_newbt *xnr, struct xfs_scrub *sc); void xrep_newbt_init_ag(struct xrep_newbt *xnr, struct xfs_scrub *sc, const struct xfs_owner_info *oinfo, xfs_fsblock_t alloc_hint, enum xfs_ag_resv_type resv); void xrep_newbt_init_inode(struct xrep_newbt *xnr, struct xfs_scrub *sc, int whichfork, const struct xfs_owner_info *oinfo); int xrep_newbt_add_blocks(struct xrep_newbt *xnr, xfs_fsblock_t fsbno, xfs_extlen_t len); int xrep_newbt_alloc_blocks(struct xrep_newbt *xnr, uint64_t nr_blocks); void xrep_newbt_destroy(struct xrep_newbt *xnr, int error); int xrep_newbt_claim_block(struct xfs_btree_cur *cur, struct xrep_newbt *xnr, union xfs_btree_ptr *ptr); void xrep_bload_estimate_slack(struct xfs_scrub *sc, struct xfs_btree_bload *bload); int xrep_newbt_relog_efis(struct xrep_newbt *xnr); bool xrep_buf_verify_struct(struct xfs_buf *bp, const struct xfs_buf_ops *ops); xfs_ino_t xrep_dotdot_lookup(struct xfs_scrub *sc); #else static inline int xrep_attempt( struct xfs_scrub *sc) { return -EOPNOTSUPP; } static inline void xrep_failure(struct xfs_mount *mp) {} static inline xfs_extlen_t xrep_calc_ag_resblks( struct xfs_scrub *sc) { return 0; } static inline int xrep_reset_perag_resv( struct xfs_scrub *sc) { if (!(sc->flags & XREP_RESET_PERAG_RESV)) return 0; ASSERT(0); return -EOPNOTSUPP; } /* repair setup functions for CONFIG_XFS_REPAIR=n */ static inline int xrep_setup_ag_rmapbt( struct xfs_scrub *sc) { /* We don't support rmap repair, but we can still do a scan. */ return xchk_setup_ag_btree(sc, false); } static inline int xrep_setup_rtsummary(struct xfs_scrub *sc, unsigned int *resblks) { *resblks = 0; return 0; } static inline int xrep_setup_xattr( struct xfs_scrub *sc) { return 0; } #define xrep_setup_directory xrep_setup_xattr #define xrep_revalidate_allocbt (NULL) #define xrep_revalidate_iallocbt (NULL) #define xrep_probe xrep_notsupported #define xrep_superblock xrep_notsupported #define xrep_agf xrep_notsupported #define xrep_agfl xrep_notsupported #define xrep_agi xrep_notsupported #define xrep_allocbt xrep_notsupported #define xrep_iallocbt xrep_notsupported #define xrep_rmapbt xrep_notsupported #define xrep_refcountbt xrep_notsupported #define xrep_inode xrep_notsupported #define xrep_bmap_data xrep_notsupported #define xrep_bmap_attr xrep_notsupported #define xrep_symlink xrep_notsupported #define xrep_quota xrep_notsupported #define xrep_quotacheck xrep_notsupported #define xrep_fscounters xrep_notsupported #define xrep_rtsummary xrep_notsupported #define xrep_xattr xrep_notsupported #define xrep_directory xrep_notsupported #define xrep_parent xrep_notsupported #endif /* CONFIG_XFS_ONLINE_REPAIR */ #endif /* __XFS_SCRUB_REPAIR_H__ */