// SPDX-License-Identifier: GPL-2.0-or-later /* * Copyright (C) 2022 Oracle. All Rights Reserved. * Author: Darrick J. Wong */ #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_rtalloc.h" #include "xfs_inode.h" #include "xfs_bit.h" #include "xfs_bmap.h" #include "xfs_bmap_btree.h" #include "xfs_swapext.h" #include "xfs_rtbitmap.h" #include "scrub/scrub.h" #include "scrub/common.h" #include "scrub/trace.h" #include "scrub/repair.h" #include "scrub/tempfile.h" #include "scrub/tempswap.h" #include "scrub/reap.h" #include "scrub/xfile.h" #include "scrub/rtsummary.h" struct xrep_rtsummary { /* suminfo position of xfile as we write buffers to disk. */ xfs_rtsumoff_t prep_wordoff; }; /* Set us up to repair the rtsummary file. */ int xrep_setup_rtsummary( struct xfs_scrub *sc, unsigned int *resblks, size_t *bufsize) { struct xfs_mount *mp = sc->mp; unsigned long long blocks; int error; *bufsize = max(*bufsize, sizeof(struct xrep_tempswap)); error = xrep_tempfile_create(sc, S_IFREG); if (error) return error; /* * If we're doing a repair, we reserve enough blocks to write out a * completely new summary file, plus twice as many blocks as we would * need if we can only allocate one block per data fork mapping. This * should cover the preallocation of the temporary file and swapping * the extent mappings. * * We cannot use xfs_swapext_estimate because we have not yet * constructed the replacement rtsummary and therefore do not know how * many extents it will use. By the time we do, we will have a dirty * transaction (which we cannot drop because we cannot drop the * rtsummary ILOCK) and cannot ask for more reservation. */ blocks = XFS_B_TO_FSB(mp, mp->m_rsumsize); blocks += xfs_bmbt_calc_size(mp, blocks) * 2; if (blocks > UINT_MAX) return -EOPNOTSUPP; *resblks += blocks; /* * Grab support for atomic extent swapping before we allocate any * transactions or grab ILOCKs. */ return xrep_tempswap_grab_log_assist(sc); } static int xrep_rtsummary_prep_buf( struct xfs_scrub *sc, struct xfs_buf *bp, void *data) { struct xrep_rtsummary *rs = data; struct xfs_mount *mp = sc->mp; int error; bp->b_ops = &xfs_rtbuf_ops; error = xfsum_copyout(sc, rs->prep_wordoff, xfs_rsumblock_infoptr(bp, 0), mp->m_blockwsize); if (error) return error; rs->prep_wordoff += mp->m_blockwsize; xfs_trans_buf_set_type(sc->tp, bp, XFS_BLFT_RTSUMMARY_BUF); return 0; } /* Repair the realtime summary. */ int xrep_rtsummary( struct xfs_scrub *sc) { struct xrep_rtsummary rs = { .prep_wordoff = 0, }; struct xrep_tempswap *ti = NULL; xfs_filblks_t rsumblocks; int error; /* We require the rmapbt to rebuild anything. */ if (!xfs_has_rmapbt(sc->mp)) return -EOPNOTSUPP; /* Make sure any problems with the fork are fixed. */ error = xrep_metadata_inode_forks(sc); if (error) return error; /* * Try to take ILOCK_EXCL of the temporary file. We had better be the * only ones holding onto this inode, but we can't block while holding * the rtsummary file's ILOCK_EXCL. */ while (!xrep_tempfile_ilock_nowait(sc)) { if (xchk_should_terminate(sc, &error)) return error; delay(1); } /* Make sure we have space allocated for the entire summary file. */ rsumblocks = XFS_B_TO_FSB(sc->mp, sc->mp->m_rsumsize); xfs_trans_ijoin(sc->tp, sc->ip, 0); xfs_trans_ijoin(sc->tp, sc->tempip, 0); error = xrep_tempfile_prealloc(sc, 0, rsumblocks); if (error) return error; /* Last chance to abort before we start committing fixes. */ if (xchk_should_terminate(sc, &error)) return error; /* Copy the rtsummary file that we generated. */ error = xrep_tempfile_copyin(sc, 0, rsumblocks, xrep_rtsummary_prep_buf, &rs); if (error) return error; error = xrep_tempfile_set_isize(sc, sc->mp->m_rsumsize); if (error) return error; /* * Now swap the extents. Nothing in repair uses the temporary buffer, * so we can reuse it for the tempfile swapext information. */ ti = sc->buf; error = xrep_tempswap_trans_reserve(sc, XFS_DATA_FORK, ti); if (error) return error; error = xrep_tempswap_contents(sc, ti); if (error) return error; ti = NULL; /* Free the old rtsummary blocks if they're not in use. */ return xrep_reap_ifork(sc, sc->tempip, XFS_DATA_FORK); }