[BACK]Return to phase7.c CVS log [TXT][DIR] Up to [Development] / xfs-cmds / xfsprogs / repair

File: [Development] / xfs-cmds / xfsprogs / repair / phase7.c (download)

Revision 1.13, Fri Sep 29 17:16:48 2006 UTC (11 years ago) by mvalluri
Branch: MAIN
Changes since 1.12: +5 -0 lines

Progress reporting for lengthy runs.
Invoke reporting functions to track progress.

/*
 * Copyright (c) 2000-2001,2005 Silicon Graphics, Inc.
 * All Rights Reserved.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it would be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write the Free Software Foundation,
 * Inc.,  51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 */

#include <libxfs.h>
#include "avl.h"
#include "globals.h"
#include "agheader.h"
#include "incore.h"
#include "protos.h"
#include "err_protos.h"
#include "dinode.h"
#include "versions.h"
#include "prefetch.h"
#include "progress.h"
#include "threads.h"

/* dinoc is a pointer to the IN-CORE dinode core */
void
set_nlinks(xfs_dinode_core_t	*dinoc,
		xfs_ino_t	ino,
		__uint32_t	nrefs,
		int		*dirty)
{
	if (!no_modify)  {
		if (dinoc->di_nlink != nrefs)  {
			*dirty = 1;
			do_warn(
		_("resetting inode %llu nlinks from %d to %d\n"),
				ino, dinoc->di_nlink, nrefs);

			if (nrefs > XFS_MAXLINK_1)  {
				ASSERT(fs_inode_nlink);
				do_warn(
_("nlinks %d will overflow v1 ino, ino %llu will be converted to version 2\n"),
					nrefs, ino);

			}
			dinoc->di_nlink = nrefs;
		}
	} else  {
		if (dinoc->di_nlink != nrefs)
			do_warn(
			_("would have reset inode %llu nlinks from %d to %d\n"),
				ino, dinoc->di_nlink, nrefs);
	}
}

void
phase7_alt_function(xfs_mount_t *mp, xfs_agnumber_t agno)
{
	register ino_tree_node_t *irec;
	int			j;
	int			dirty;
	xfs_ino_t		ino;
	__uint32_t		nrefs;
	xfs_agblock_t		agbno;
	xfs_dinode_t		*dip;
	ino_tree_node_t 	*ino_ra;
	xfs_buf_t		*bp;

	if (verbose)
		do_log(_("        - agno = %d\n"), agno);

	ino_ra = prefetch_inode_chunks(mp, agno, NULL);

	/*
	 * read on-disk inodes in chunks. then,
	 * look at each on-disk inode 1 at a time.
	 * if the number of links is bad, reset it.
	 */

	irec = findfirst_inode_rec(agno);

	while (irec != NULL)  {

		if (ino_ra && (irec->ino_startnum >= ino_ra->ino_startnum))
			ino_ra = prefetch_inode_chunks(mp, agno, ino_ra);

		agbno = XFS_AGINO_TO_AGBNO(mp, irec->ino_startnum);
		bp = libxfs_readbuf(mp->m_dev,
				XFS_AGB_TO_DADDR(mp, agno, agbno),
				XFS_FSB_TO_BB(mp, XFS_IALLOC_BLOCKS(mp)), 0);
		if (bp == NULL) {
			if (!no_modify) {
				do_error(
	_("cannot read inode %llu, disk block %lld, cnt %d\n"),
				XFS_AGINO_TO_INO(mp, agno, irec->ino_startnum),
				XFS_AGB_TO_DADDR(mp, agno, agbno),
				(int)XFS_FSB_TO_BB(mp, XFS_IALLOC_BLOCKS(mp)));
				/* NOT REACHED */
			}
			do_warn(
	_("cannot read inode %llu, disk block %lld, cnt %d\n"),
				XFS_AGINO_TO_INO(mp, agno, irec->ino_startnum),
				XFS_AGB_TO_DADDR(mp, agno, agbno),
				(int)XFS_FSB_TO_BB(mp, XFS_IALLOC_BLOCKS(mp)));

			irec = next_ino_rec(irec);
			continue;	/* while */
		}
		dirty = 0;
		for (j = 0; j < XFS_INODES_PER_CHUNK; j++)  {
			assert(is_inode_confirmed(irec, j));

			if (is_inode_free(irec, j))
				continue;

			assert(no_modify || is_inode_reached(irec, j));
			assert(no_modify ||
					is_inode_referenced(irec, j));

			nrefs = num_inode_references(irec, j);

			ino = XFS_AGINO_TO_INO(mp, agno,
				irec->ino_startnum + j);

			dip = (xfs_dinode_t *)(XFS_BUF_PTR(bp) +
					(j << mp->m_sb.sb_inodelog));

			/* Swap the fields we care about to native format */
			dip->di_core.di_magic = INT_GET(dip->di_core.di_magic, ARCH_CONVERT);
			dip->di_core.di_nlink = INT_GET(dip->di_core.di_nlink, ARCH_CONVERT);

			if (dip->di_core.di_magic != XFS_DINODE_MAGIC) {
				if (!no_modify) {
					do_error(
	_("ino: %llu, bad d_inode magic saw: (0x%x) expecting (0x%x)\n"),
				ino, dip->di_core.di_magic, XFS_DINODE_MAGIC);
					/* NOT REACHED */
				}
				do_warn(
	_("ino: %llu, bad d_inode magic saw: (0x%x) expecting (0x%x)\n"),
				ino, dip->di_core.di_magic, XFS_DINODE_MAGIC);
				continue;
			}
			/*
			 * compare and set links for all inodes
			 * but the lost+found inode.  we keep
			 * that correct as we go.
			 */
			if (dip->di_core.di_nlink != nrefs) {
				if (ino != orphanage_ino) {
					set_nlinks(&dip->di_core, ino,
							nrefs, &dirty);
				}
			}

			/* Swap the fields back */
			dip->di_core.di_magic = INT_GET(dip->di_core.di_magic, ARCH_CONVERT);
			dip->di_core.di_nlink = INT_GET(dip->di_core.di_nlink, ARCH_CONVERT);
		}

		if (dirty)
			libxfs_writebuf(bp, 0);
		else
			libxfs_putbuf(bp);

		irec = next_ino_rec(irec);
		PROG_RPT_INC(prog_rpt_done[agno], XFS_INODES_PER_CHUNK);
	}
}

void
phase7_alt(xfs_mount_t *mp)
{
	int		i;

	set_progress_msg(no_modify ? PROGRESS_FMT_VRFY_LINK : PROGRESS_FMT_CORR_LINK,
		(__uint64_t) mp->m_sb.sb_icount);
	libxfs_bcache_purge();

	for (i = 0; i < glob_agcount; i++)  {
		queue_work(phase7_alt_function, mp, i);
	}
	wait_for_workers();
	print_final_rpt();
}

void
phase7(xfs_mount_t *mp)
{
	ino_tree_node_t		*irec;
	xfs_inode_t		*ip;
	xfs_trans_t		*tp;
	int			i;
	int			j;
	int			error;
	int			dirty;
	xfs_ino_t		ino;
	__uint32_t		nrefs;

	if (!no_modify)
		do_log(_("Phase 7 - verify and correct link counts...\n"));
	else
		do_log(_("Phase 7 - verify link counts...\n"));


	if (do_prefetch) {
		phase7_alt(mp);
		return;
	}

	tp = libxfs_trans_alloc(mp, XFS_TRANS_REMOVE);

	error = libxfs_trans_reserve(tp, (no_modify ? 0 : 10),
			XFS_REMOVE_LOG_RES(mp), 0, XFS_TRANS_PERM_LOG_RES,
			XFS_REMOVE_LOG_COUNT);

	ASSERT(error == 0);

	/*
	 * for each ag, look at each inode 1 at a time using the
	 * sim code.  if the number of links is bad, reset it,
	 * log the inode core, commit the transaction, and
	 * allocate a new transaction
	 */
	for (i = 0; i < glob_agcount; i++)  {
		irec = findfirst_inode_rec(i);

		while (irec != NULL)  {
			for (j = 0; j < XFS_INODES_PER_CHUNK; j++)  {
				ASSERT(is_inode_confirmed(irec, j));

				if (is_inode_free(irec, j))
					continue;

				ASSERT(no_modify || is_inode_reached(irec, j));
				ASSERT(no_modify ||
						is_inode_referenced(irec, j));

				nrefs = num_inode_references(irec, j);

				ino = XFS_AGINO_TO_INO(mp, i,
					irec->ino_startnum + j);

				error = libxfs_trans_iget(mp, tp, ino, 0, 0, &ip);

				if (error)  {
					if (!no_modify)
						do_error(
				_("couldn't map inode %llu, err = %d\n"),
							ino, error);
					else  {
						do_warn(
	_("couldn't map inode %llu, err = %d, can't compare link counts\n"),
							ino, error);
						continue;
					}
				}

				dirty = 0;

				/*
				 * compare and set links for all inodes
				 * but the lost+found inode.  we keep
				 * that correct as we go.
				 */
				if (ino != orphanage_ino)
					set_nlinks(&ip->i_d, ino, nrefs,
							&dirty);

				if (!dirty)  {
					libxfs_trans_iput(tp, ip, 0);
				} else  {
					libxfs_trans_log_inode(tp, ip,
							XFS_ILOG_CORE);
					/*
					 * no need to do a bmap finish since
					 * we're not allocating anything
					 */
					ASSERT(error == 0);
					error = libxfs_trans_commit(tp,
						XFS_TRANS_RELEASE_LOG_RES|
						XFS_TRANS_SYNC, NULL);

					ASSERT(error == 0);

					tp = libxfs_trans_alloc(mp,
							XFS_TRANS_REMOVE);

					error = libxfs_trans_reserve(tp,
						(no_modify ? 0 : 10),
						XFS_REMOVE_LOG_RES(mp),
						0, XFS_TRANS_PERM_LOG_RES,
						XFS_REMOVE_LOG_COUNT);
					ASSERT(error == 0);
				}
			}
			irec = next_ino_rec(irec);
		}
	}

	/*
	 * always have one unfinished transaction coming out
	 * of the loop.  cancel it.
	 */
	libxfs_trans_cancel(tp, XFS_TRANS_RELEASE_LOG_RES);
}