|  | // SPDX-License-Identifier: GPL-2.0-or-later | 
|  | /* | 
|  | * Copyright (C) 2020-2023 Oracle.  All Rights Reserved. | 
|  | * Author: Darrick J. Wong <djwong@kernel.org> | 
|  | */ | 
|  | #include "xfs.h" | 
|  | #include "xfs_fs.h" | 
|  | #include "xfs_shared.h" | 
|  | #include "xfs_format.h" | 
|  | #include "xfs_trans_resv.h" | 
|  | #include "xfs_mount.h" | 
|  | #include "xfs_btree.h" | 
|  | #include "xfs_log_format.h" | 
|  | #include "xfs_trans.h" | 
|  | #include "xfs_inode.h" | 
|  | #include "xfs_bit.h" | 
|  | #include "xfs_bmap.h" | 
|  | #include "xfs_bmap_btree.h" | 
|  | #include "scrub/scrub.h" | 
|  | #include "scrub/common.h" | 
|  | #include "scrub/trace.h" | 
|  | #include "scrub/repair.h" | 
|  | #include "scrub/xfile.h" | 
|  | #include "scrub/rtbitmap.h" | 
|  |  | 
|  | /* Set up to repair the realtime bitmap file metadata. */ | 
|  | int | 
|  | xrep_setup_rtbitmap( | 
|  | struct xfs_scrub	*sc, | 
|  | struct xchk_rtbitmap	*rtb) | 
|  | { | 
|  | struct xfs_mount	*mp = sc->mp; | 
|  | unsigned long long	blocks = 0; | 
|  |  | 
|  | /* | 
|  | * Reserve enough blocks to write out a completely new bmbt for a | 
|  | * maximally fragmented bitmap file.  We do not hold the rtbitmap | 
|  | * ILOCK yet, so this is entirely speculative. | 
|  | */ | 
|  | blocks = xfs_bmbt_calc_size(mp, mp->m_sb.sb_rbmblocks); | 
|  | if (blocks > UINT_MAX) | 
|  | return -EOPNOTSUPP; | 
|  |  | 
|  | rtb->resblks += blocks; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Make sure that the given range of the data fork of the realtime file is | 
|  | * mapped to written blocks.  The caller must ensure that the inode is joined | 
|  | * to the transaction. | 
|  | */ | 
|  | STATIC int | 
|  | xrep_rtbitmap_data_mappings( | 
|  | struct xfs_scrub	*sc, | 
|  | xfs_filblks_t		len) | 
|  | { | 
|  | struct xfs_bmbt_irec	map; | 
|  | xfs_fileoff_t		off = 0; | 
|  | int			error; | 
|  |  | 
|  | ASSERT(sc->ip != NULL); | 
|  |  | 
|  | while (off < len) { | 
|  | int		nmaps = 1; | 
|  |  | 
|  | /* | 
|  | * If we have a real extent mapping this block then we're | 
|  | * in ok shape. | 
|  | */ | 
|  | error = xfs_bmapi_read(sc->ip, off, len - off, &map, &nmaps, | 
|  | XFS_DATA_FORK); | 
|  | if (error) | 
|  | return error; | 
|  | if (nmaps == 0) { | 
|  | ASSERT(nmaps != 0); | 
|  | return -EFSCORRUPTED; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Written extents are ok.  Holes are not filled because we | 
|  | * do not know the freespace information. | 
|  | */ | 
|  | if (xfs_bmap_is_written_extent(&map) || | 
|  | map.br_startblock == HOLESTARTBLOCK) { | 
|  | off = map.br_startoff + map.br_blockcount; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * If we find a delalloc reservation then something is very | 
|  | * very wrong.  Bail out. | 
|  | */ | 
|  | if (map.br_startblock == DELAYSTARTBLOCK) | 
|  | return -EFSCORRUPTED; | 
|  |  | 
|  | /* Make sure we're really converting an unwritten extent. */ | 
|  | if (map.br_state != XFS_EXT_UNWRITTEN) { | 
|  | ASSERT(map.br_state == XFS_EXT_UNWRITTEN); | 
|  | return -EFSCORRUPTED; | 
|  | } | 
|  |  | 
|  | /* Make sure this block has a real zeroed extent mapped. */ | 
|  | nmaps = 1; | 
|  | error = xfs_bmapi_write(sc->tp, sc->ip, map.br_startoff, | 
|  | map.br_blockcount, | 
|  | XFS_BMAPI_CONVERT | XFS_BMAPI_ZERO, | 
|  | 0, &map, &nmaps); | 
|  | if (error) | 
|  | return error; | 
|  | if (nmaps != 1) | 
|  | return -EFSCORRUPTED; | 
|  |  | 
|  | /* Commit new extent and all deferred work. */ | 
|  | error = xrep_defer_finish(sc); | 
|  | if (error) | 
|  | return error; | 
|  |  | 
|  | off = map.br_startoff + map.br_blockcount; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Fix broken rt volume geometry. */ | 
|  | STATIC int | 
|  | xrep_rtbitmap_geometry( | 
|  | struct xfs_scrub	*sc, | 
|  | struct xchk_rtbitmap	*rtb) | 
|  | { | 
|  | struct xfs_mount	*mp = sc->mp; | 
|  | struct xfs_trans	*tp = sc->tp; | 
|  |  | 
|  | /* Superblock fields */ | 
|  | if (mp->m_sb.sb_rextents != rtb->rextents) | 
|  | xfs_trans_mod_sb(sc->tp, XFS_TRANS_SB_REXTENTS, | 
|  | rtb->rextents - mp->m_sb.sb_rextents); | 
|  |  | 
|  | if (mp->m_sb.sb_rbmblocks != rtb->rbmblocks) | 
|  | xfs_trans_mod_sb(tp, XFS_TRANS_SB_RBMBLOCKS, | 
|  | rtb->rbmblocks - mp->m_sb.sb_rbmblocks); | 
|  |  | 
|  | if (mp->m_sb.sb_rextslog != rtb->rextslog) | 
|  | xfs_trans_mod_sb(tp, XFS_TRANS_SB_REXTSLOG, | 
|  | rtb->rextslog - mp->m_sb.sb_rextslog); | 
|  |  | 
|  | /* Fix broken isize */ | 
|  | sc->ip->i_disk_size = roundup_64(sc->ip->i_disk_size, | 
|  | mp->m_sb.sb_blocksize); | 
|  |  | 
|  | if (sc->ip->i_disk_size < XFS_FSB_TO_B(mp, rtb->rbmblocks)) | 
|  | sc->ip->i_disk_size = XFS_FSB_TO_B(mp, rtb->rbmblocks); | 
|  |  | 
|  | xfs_trans_log_inode(sc->tp, sc->ip, XFS_ILOG_CORE); | 
|  | return xrep_roll_trans(sc); | 
|  | } | 
|  |  | 
|  | /* Repair the realtime bitmap file metadata. */ | 
|  | int | 
|  | xrep_rtbitmap( | 
|  | struct xfs_scrub	*sc) | 
|  | { | 
|  | struct xchk_rtbitmap	*rtb = sc->buf; | 
|  | struct xfs_mount	*mp = sc->mp; | 
|  | unsigned long long	blocks = 0; | 
|  | int			error; | 
|  |  | 
|  | /* Impossibly large rtbitmap means we can't touch the filesystem. */ | 
|  | if (rtb->rbmblocks > U32_MAX) | 
|  | return 0; | 
|  |  | 
|  | /* | 
|  | * If the size of the rt bitmap file is larger than what we reserved, | 
|  | * figure out if we need to adjust the block reservation in the | 
|  | * transaction. | 
|  | */ | 
|  | blocks = xfs_bmbt_calc_size(mp, rtb->rbmblocks); | 
|  | if (blocks > UINT_MAX) | 
|  | return -EOPNOTSUPP; | 
|  | if (blocks > rtb->resblks) { | 
|  | error = xfs_trans_reserve_more(sc->tp, blocks, 0); | 
|  | if (error) | 
|  | return error; | 
|  |  | 
|  | rtb->resblks += blocks; | 
|  | } | 
|  |  | 
|  | /* Fix inode core and forks. */ | 
|  | error = xrep_metadata_inode_forks(sc); | 
|  | if (error) | 
|  | return error; | 
|  |  | 
|  | xfs_trans_ijoin(sc->tp, sc->ip, 0); | 
|  |  | 
|  | /* Ensure no unwritten extents. */ | 
|  | error = xrep_rtbitmap_data_mappings(sc, rtb->rbmblocks); | 
|  | if (error) | 
|  | return error; | 
|  |  | 
|  | /* Fix inconsistent bitmap geometry */ | 
|  | return xrep_rtbitmap_geometry(sc, rtb); | 
|  | } |