Blame gfs2/fsck/main.c

Packit 6ef888
#include "clusterautoconfig.h"
Packit 6ef888
Packit 6ef888
#include <unistd.h>
Packit 6ef888
#include <stdio.h>
Packit 6ef888
#include <stdint.h>
Packit 6ef888
#include <stdlib.h>
Packit 6ef888
#include <libgen.h>
Packit 6ef888
#include <string.h>
Packit 6ef888
#include <stdarg.h>
Packit 6ef888
#include <ctype.h>
Packit 6ef888
#include <signal.h>
Packit 6ef888
#include <libintl.h>
Packit 6ef888
#include <locale.h>
Packit 6ef888
#include <sys/time.h>
Packit 6ef888
#include <time.h>
Packit 6ef888
#define _(String) gettext(String)
Packit 6ef888
#include <syslog.h>
Packit 6ef888
Packit 6ef888
#include <logging.h>
Packit 6ef888
#include "copyright.cf"
Packit 6ef888
#include "libgfs2.h"
Packit 6ef888
#include "fsck.h"
Packit 6ef888
#include "link.h"
Packit 6ef888
#include "osi_list.h"
Packit 6ef888
#include "metawalk.h"
Packit 6ef888
#include "util.h"
Packit 6ef888
Packit 6ef888
struct gfs2_options opts = {0};
Packit 6ef888
struct gfs2_inode *lf_dip = NULL; /* Lost and found directory inode */
Packit 6ef888
int lf_was_created = 0;
Packit 6ef888
uint64_t last_fs_block, last_reported_block = -1;
Packit 6ef888
int64_t last_reported_fblock = -1000000;
Packit 6ef888
int skip_this_pass = FALSE, fsck_abort = FALSE;
Packit 6ef888
int errors_found = 0, errors_corrected = 0;
Packit 6ef888
const char *pass = "";
Packit 6ef888
uint64_t last_data_block;
Packit 6ef888
uint64_t first_data_block;
Packit 6ef888
int preen = 0, force_check = 0;
Packit 6ef888
struct osi_root dup_blocks;
Packit 6ef888
struct osi_root dirtree;
Packit 6ef888
struct osi_root inodetree;
Packit 6ef888
int dups_found = 0, dups_found_first = 0;
Packit 6ef888
struct gfs_sb *sbd1 = NULL;
Packit 6ef888
int sb_fixed = 0;
Packit 6ef888
int print_level = MSG_NOTICE;
Packit 6ef888
Packit 6ef888
/* This function is for libgfs2's sake.                                      */
Packit 6ef888
void print_it(const char *label, const char *fmt, const char *fmt2, ...)
Packit 6ef888
{
Packit 6ef888
	va_list args;
Packit 6ef888
Packit 6ef888
	va_start(args, fmt2);
Packit 6ef888
	printf("%s: ", label);
Packit 6ef888
	vprintf(fmt, args);
Packit 6ef888
	va_end(args);
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static void usage(char *name)
Packit 6ef888
{
Packit 6ef888
	printf("Usage: %s [-afhnpqvVy] <device> \n", basename(name));
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static void version(void)
Packit 6ef888
{
Packit 6ef888
	printf( _("GFS2 fsck %s (built %s %s)\n"),
Packit 6ef888
	       VERSION, __DATE__, __TIME__);
Packit 6ef888
	printf(REDHAT_COPYRIGHT "\n");
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static int read_cmdline(int argc, char **argv, struct gfs2_options *gopts)
Packit 6ef888
{
Packit 6ef888
	int c;
Packit 6ef888
Packit 6ef888
	while ((c = getopt(argc, argv, "afhnpqvyV")) != -1) {
Packit 6ef888
		switch(c) {
Packit 6ef888
Packit 6ef888
		case 'a':
Packit 6ef888
		case 'p':
Packit 6ef888
			if (gopts->yes || gopts->no) {
Packit 6ef888
				fprintf(stderr, _("Options -p/-a, -y and -n may not be used together\n"));
Packit 6ef888
				return FSCK_USAGE;
Packit 6ef888
			}
Packit 6ef888
			preen = 1;
Packit 6ef888
			gopts->yes = 1;
Packit 6ef888
			break;
Packit 6ef888
		case 'f':
Packit 6ef888
			force_check = 1;
Packit 6ef888
			break;
Packit 6ef888
		case 'h':
Packit 6ef888
			usage(argv[0]);
Packit 6ef888
			exit(FSCK_OK);
Packit 6ef888
			break;
Packit 6ef888
		case 'n':
Packit 6ef888
			if (gopts->yes || preen) {
Packit 6ef888
				fprintf(stderr, _("Options -p/-a, -y and -n may not be used together\n"));
Packit 6ef888
				return FSCK_USAGE;
Packit 6ef888
			}
Packit 6ef888
			gopts->no = 1;
Packit 6ef888
			break;
Packit 6ef888
		case 'q':
Packit 6ef888
			decrease_verbosity();
Packit 6ef888
			break;
Packit 6ef888
		case 'v':
Packit 6ef888
			increase_verbosity();
Packit 6ef888
			break;
Packit 6ef888
		case 'V':
Packit 6ef888
			version();
Packit 6ef888
			exit(FSCK_OK);
Packit 6ef888
			break;
Packit 6ef888
		case 'y':
Packit 6ef888
			if (gopts->no || preen) {
Packit 6ef888
				fprintf(stderr, _("Options -p/-a, -y and -n may not be used together\n"));
Packit 6ef888
				return FSCK_USAGE;
Packit 6ef888
			}
Packit 6ef888
			gopts->yes = 1;
Packit 6ef888
			break;
Packit 6ef888
		case ':':
Packit 6ef888
		case '?':
Packit 6ef888
			fprintf(stderr, _("Please use '-h' for help.\n"));
Packit 6ef888
			return FSCK_USAGE;
Packit 6ef888
		default:
Packit 6ef888
			fprintf(stderr, _("Invalid option %c\n"), c);
Packit 6ef888
			return FSCK_USAGE;
Packit 6ef888
Packit 6ef888
		}
Packit 6ef888
	}
Packit 6ef888
	if (argc > optind) {
Packit 6ef888
		gopts->device = (argv[optind]);
Packit 6ef888
		if (!gopts->device) {
Packit 6ef888
			fprintf(stderr, _("Please use '-h' for help.\n"));
Packit 6ef888
			return FSCK_USAGE;
Packit 6ef888
		}
Packit 6ef888
	} else {
Packit 6ef888
		fprintf(stderr, _("No device specified (Please use '-h' for help)\n"));
Packit 6ef888
		return FSCK_USAGE;
Packit 6ef888
	}
Packit 6ef888
	return 0;
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static void interrupt(int sig)
Packit 6ef888
{
Packit 6ef888
	char response;
Packit 6ef888
	char progress[PATH_MAX];
Packit 6ef888
Packit 6ef888
	if (!last_reported_block || last_reported_block == last_fs_block)
Packit 6ef888
		sprintf(progress, _("progress unknown.\n"));
Packit 6ef888
	else
Packit 6ef888
		sprintf(progress, _("processing block %llu out of %llu\n"),
Packit 6ef888
			(unsigned long long)last_reported_block,
Packit 6ef888
			(unsigned long long)last_fs_block);
Packit 6ef888
	
Packit 6ef888
	response = generic_interrupt("fsck.gfs2", pass, progress,
Packit 6ef888
				     _("Do you want to abort fsck.gfs2, skip " \
Packit 6ef888
				     "the rest of this pass or continue " \
Packit 6ef888
				     "(a/s/c)?"), "asc");
Packit 6ef888
	if (tolower(response) == 's') {
Packit 6ef888
		skip_this_pass = TRUE;
Packit 6ef888
		return;
Packit 6ef888
	}
Packit 6ef888
	else if (tolower(response) == 'a') {
Packit 6ef888
		fsck_abort = TRUE;
Packit 6ef888
		return;
Packit 6ef888
	}
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static int check_statfs(struct gfs2_sbd *sdp)
Packit 6ef888
{
Packit 6ef888
	struct osi_node *n, *next = NULL;
Packit 6ef888
	struct rgrp_tree *rgd;
Packit 6ef888
	struct gfs2_rindex *ri;
Packit 6ef888
	struct gfs2_statfs_change sc = {0,};
Packit 6ef888
	char buf[sizeof(struct gfs2_statfs_change)];
Packit 6ef888
	int count;
Packit 6ef888
Packit 6ef888
	if (sdp->gfs1 && !sdp->md.statfs->i_di.di_size) {
Packit 6ef888
		log_info("This GFS1 file system is not using fast_statfs.\n");
Packit 6ef888
		return 0;
Packit 6ef888
	}
Packit 6ef888
	/* Read the current statfs values */
Packit 6ef888
	count = gfs2_readi(sdp->md.statfs, buf, 0,
Packit 6ef888
			   sdp->md.statfs->i_di.di_size);
Packit 6ef888
	if (count != sizeof(struct gfs2_statfs_change)) {
Packit 6ef888
		log_err(_("Failed to read statfs values (%d of %"PRIu64" read)\n"),
Packit 6ef888
		        count, (uint64_t)sdp->md.statfs->i_di.di_size);
Packit 6ef888
		return FSCK_ERROR;
Packit 6ef888
	}
Packit 6ef888
	gfs2_statfs_change_in(&sc, buf);
Packit 6ef888
	/* Calculate the real values from the rgrp information */
Packit 6ef888
	sdp->blks_total = 0;
Packit 6ef888
	sdp->blks_alloced = 0;
Packit 6ef888
	sdp->dinodes_alloced = 0;
Packit 6ef888
Packit 6ef888
	for (n = osi_first(&sdp->rgtree); n; n = next) {
Packit 6ef888
		next = osi_next(n);
Packit 6ef888
		rgd = (struct rgrp_tree *)n;
Packit 6ef888
		ri = &rgd->ri;
Packit 6ef888
		sdp->blks_total += ri->ri_data;
Packit 6ef888
		sdp->blks_alloced += (ri->ri_data - rgd->rg.rg_free);
Packit 6ef888
		sdp->dinodes_alloced += rgd->rg.rg_dinodes;
Packit 6ef888
	}
Packit 6ef888
Packit 6ef888
	/* See if they match */
Packit 6ef888
	if (sc.sc_total == sdp->blks_total &&
Packit 6ef888
	    sc.sc_free == (sdp->blks_total - sdp->blks_alloced) &&
Packit 6ef888
	    sc.sc_dinodes == sdp->dinodes_alloced) {
Packit 6ef888
		log_info( _("The statfs file is accurate.\n"));
Packit 6ef888
		return 0;
Packit 6ef888
	}
Packit 6ef888
	log_err( _("The statfs file is wrong:\n\n"));
Packit 6ef888
	log_err( _("Current statfs values:\n"));
Packit 6ef888
	log_err( _("blocks:  %lld (0x%llx)\n"),
Packit 6ef888
		 (unsigned long long)sc.sc_total,
Packit 6ef888
		 (unsigned long long)sc.sc_total);
Packit 6ef888
	log_err( _("free:    %lld (0x%llx)\n"),
Packit 6ef888
		 (unsigned long long)sc.sc_free,
Packit 6ef888
		 (unsigned long long)sc.sc_free);
Packit 6ef888
	log_err( _("dinodes: %lld (0x%llx)\n\n"),
Packit 6ef888
		 (unsigned long long)sc.sc_dinodes,
Packit 6ef888
		 (unsigned long long)sc.sc_dinodes);
Packit 6ef888
Packit 6ef888
	log_err( _("Calculated statfs values:\n"));
Packit 6ef888
	log_err( _("blocks:  %lld (0x%llx)\n"),
Packit 6ef888
		 (unsigned long long)sdp->blks_total,
Packit 6ef888
		 (unsigned long long)sdp->blks_total);
Packit 6ef888
	log_err( _("free:    %lld (0x%llx)\n"),
Packit 6ef888
		 (unsigned long long)(sdp->blks_total - sdp->blks_alloced),
Packit 6ef888
		 (unsigned long long)(sdp->blks_total - sdp->blks_alloced));
Packit 6ef888
	log_err( _("dinodes: %lld (0x%llx)\n"),
Packit 6ef888
		 (unsigned long long)sdp->dinodes_alloced,
Packit 6ef888
		 (unsigned long long)sdp->dinodes_alloced);
Packit 6ef888
Packit 6ef888
	errors_found++;
Packit 6ef888
	if (!query( _("Okay to fix the master statfs file? (y/n)"))) {
Packit 6ef888
		log_err( _("The statfs file was not fixed.\n"));
Packit 6ef888
		return 0;
Packit 6ef888
	}
Packit 6ef888
Packit 6ef888
	do_init_statfs(sdp);
Packit 6ef888
	log_err( _("The statfs file was fixed.\n"));
Packit 6ef888
	errors_corrected++;
Packit 6ef888
	return 0;
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static const struct fsck_pass passes[] = {
Packit 6ef888
	{ .name = "pass1",  .f = pass1 },
Packit 6ef888
	{ .name = "pass1b", .f = pass1b },
Packit 6ef888
	{ .name = "pass2",  .f = pass2 },
Packit 6ef888
	{ .name = "pass3",  .f = pass3 },
Packit 6ef888
	{ .name = "pass4",  .f = pass4 },
Packit 6ef888
	{ .name = "check_statfs", .f = check_statfs },
Packit 6ef888
	{ .name = NULL, }
Packit 6ef888
};
Packit 6ef888
Packit 6ef888
static int fsck_pass(const struct fsck_pass *p, struct gfs2_sbd *sdp)
Packit 6ef888
{
Packit 6ef888
	int ret;
Packit 6ef888
	struct timeval timer;
Packit 6ef888
Packit 6ef888
	if (fsck_abort)
Packit 6ef888
		return FSCK_CANCELED;
Packit 6ef888
	pass = p->name;
Packit 6ef888
Packit 6ef888
	log_notice( _("Starting %s\n"), p->name);
Packit 6ef888
	gettimeofday(&timer, NULL);
Packit 6ef888
Packit 6ef888
	ret = p->f(sdp);
Packit 6ef888
	if (ret)
Packit 6ef888
		exit(ret);
Packit 6ef888
	if (skip_this_pass || fsck_abort) {
Packit 6ef888
		skip_this_pass = 0;
Packit 6ef888
		log_notice( _("%s interrupted   \n"), p->name);
Packit 6ef888
		return FSCK_CANCELED;
Packit 6ef888
	}
Packit 6ef888
Packit 6ef888
	print_pass_duration(p->name, &timer;;
Packit 6ef888
	return 0;
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static void exitlog(int status, void *unused)
Packit 6ef888
{
Packit 6ef888
	syslog(LOG_INFO, "exit: %d", status);
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
static void startlog(int argc, char **argv)
Packit 6ef888
{
Packit 6ef888
	int i;
Packit 6ef888
	char *cmd, *p;
Packit 6ef888
	size_t len;
Packit 6ef888
Packit 6ef888
	for (len = i = 0; i < argc; i++)
Packit 6ef888
		len += strlen(argv[i]);
Packit 6ef888
	len += argc; /* Add spaces and '\0' */
Packit 6ef888
Packit 6ef888
	cmd = malloc(len);
Packit 6ef888
	if (cmd == NULL) {
Packit 6ef888
		perror(argv[0]);
Packit 6ef888
		exit(FSCK_ERROR);
Packit 6ef888
	}
Packit 6ef888
	p = cmd;
Packit 6ef888
	for (i = 0; i < argc; i++, p++) {
Packit 6ef888
		p = stpcpy(p, argv[i]);
Packit 6ef888
		*p = ' ';
Packit 6ef888
	}
Packit 6ef888
	*(--p) = '\0';
Packit 6ef888
	syslog(LOG_INFO, "started: %s", cmd);
Packit 6ef888
	free(cmd);
Packit 6ef888
}
Packit 6ef888
Packit 6ef888
int main(int argc, char **argv)
Packit 6ef888
{
Packit 6ef888
	struct gfs2_sbd sb;
Packit 6ef888
	struct gfs2_sbd *sdp = &sb;
Packit 6ef888
	int j;
Packit 6ef888
	int i;
Packit 6ef888
	int error = 0;
Packit 6ef888
	int all_clean = 0;
Packit 6ef888
	struct sigaction act = { .sa_handler = interrupt, };
Packit 6ef888
Packit 6ef888
	setlocale(LC_ALL, "");
Packit 6ef888
	textdomain("gfs2-utils");
Packit 6ef888
Packit 6ef888
	openlog("fsck.gfs2", LOG_CONS|LOG_PID, LOG_USER);
Packit 6ef888
	startlog(argc - 1, &argv[1]);
Packit 6ef888
	on_exit(exitlog, NULL);
Packit 6ef888
Packit 6ef888
	memset(sdp, 0, sizeof(*sdp));
Packit 6ef888
Packit 6ef888
	if ((error = read_cmdline(argc, argv, &opts)))
Packit 6ef888
		exit(error);
Packit 6ef888
	setbuf(stdout, NULL);
Packit 6ef888
	log_notice( _("Initializing fsck\n"));
Packit 6ef888
	if ((error = initialize(sdp, force_check, preen, &all_clean)))
Packit 6ef888
		exit(error);
Packit 6ef888
Packit 6ef888
	if (!force_check && all_clean && preen) {
Packit 6ef888
		log_err( _("%s: clean.\n"), opts.device);
Packit 6ef888
		destroy(sdp);
Packit 6ef888
		exit(FSCK_OK);
Packit 6ef888
	}
Packit 6ef888
Packit 6ef888
	sigaction(SIGINT, &act, NULL);
Packit 6ef888
Packit 6ef888
	for (i = 0; passes[i].name; i++)
Packit 6ef888
		error = fsck_pass(passes + i, sdp);
Packit 6ef888
Packit 6ef888
	/* Free up our system inodes */
Packit 6ef888
	if (!sdp->gfs1)
Packit 6ef888
		inode_put(&sdp->md.inum);
Packit 6ef888
	inode_put(&sdp->md.statfs);
Packit 6ef888
	for (j = 0; j < sdp->md.journals; j++)
Packit 6ef888
		inode_put(&sdp->md.journal[j]);
Packit 6ef888
	free(sdp->md.journal);
Packit 6ef888
	sdp->md.journal = NULL;
Packit 6ef888
	inode_put(&sdp->md.jiinode);
Packit 6ef888
	inode_put(&sdp->md.riinode);
Packit 6ef888
	inode_put(&sdp->md.qinode);
Packit 6ef888
	if (!sdp->gfs1)
Packit 6ef888
		inode_put(&sdp->md.pinode);
Packit 6ef888
	inode_put(&sdp->md.rooti);
Packit 6ef888
	if (!sdp->gfs1)
Packit 6ef888
		inode_put(&sdp->master_dir);
Packit 6ef888
	if (lf_dip)
Packit 6ef888
		inode_put(&lf_dip);
Packit 6ef888
Packit 6ef888
	if (!opts.no && errors_corrected)
Packit 6ef888
		log_notice( _("Writing changes to disk\n"));
Packit 6ef888
	fsync(sdp->device_fd);
Packit 6ef888
	link1_destroy(&nlink1map);
Packit 6ef888
	link1_destroy(&clink1map);
Packit 6ef888
	destroy(sdp);
Packit 6ef888
	if (sb_fixed)
Packit 6ef888
		log_warn(_("Superblock was reset. Use tunegfs2 to manually "
Packit 6ef888
		           "set lock table before mounting.\n"));
Packit 6ef888
	log_notice( _("fsck.gfs2 complete\n"));
Packit 6ef888
Packit 6ef888
	if (!error) {
Packit 6ef888
		if (!errors_found)
Packit 6ef888
			error = FSCK_OK;
Packit 6ef888
		else if (errors_found == errors_corrected)
Packit 6ef888
			error = FSCK_NONDESTRUCT;
Packit 6ef888
		else
Packit 6ef888
			error = FSCK_UNCORRECTED;
Packit 6ef888
	}
Packit 6ef888
	exit(error);
Packit 6ef888
}