NetBSD/sbin/fsck_v7fs/fsck_v7fs.c

227 lines
5.2 KiB
C

/* $NetBSD: fsck_v7fs.c,v 1.2 2017/01/10 20:54:10 christos Exp $ */
/*-
* Copyright (c) 2004, 2011 The NetBSD Foundation, Inc.
* All rights reserved.
*
* This code is derived from software contributed to The NetBSD Foundation
* by UCHIYAMA Yasushi.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include <sys/cdefs.h>
#ifndef lint
__RCSID("$NetBSD: fsck_v7fs.c,v 1.2 2017/01/10 20:54:10 christos Exp $");
#endif /* not lint */
#include <sys/types.h>
#include <sys/disklabel.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <fcntl.h>
#include <err.h>
#include <fs/v7fs/v7fs.h>
#include "v7fs_impl.h"
#include "fsck_v7fs.h"
#include "progress.h"
static void usage(void) __dead;
static void catopt(char **, const char *);
enum fsck_operate fsck_operate;
bool verbose = true;
#define VPRINTF(fmt, args...) { if (verbose) printf(fmt, ##args); }
int
main(int argc, char **argv)
{
const char *device;
struct disklabel d;
struct partition *p;
struct stat st;
int Fflag = 0;
int part;
int fd, ch;
int endian = _BYTE_ORDER;
int openflags = O_RDWR;
size_t part_sectors;
int fsck_flags = 0;
char *options = 0;
bool progress_bar_enable = false;
fsck_operate = ASK;
if (argc < 2)
usage();
while ((ch = getopt(argc, argv, "pPqynfx:dFB:o:")) != -1) {
switch (ch) {
/*
* generic fsck options
*/
case 'd': /* Not supported */
break;
case 'f': /* Always forced */
break;
case 'p':
fsck_operate = PREEN;
break;
case 'y':
fsck_operate = ALWAYS_YES;
break;
case 'n':
fsck_operate = ALWAYS_NO;
openflags = O_RDONLY;
break;
case 'P':
progress_bar_enable = true;
break;
case 'q': /* Not supported */
break;
case 'x': /* Not supported */
break;
/*
* v7fs fsck options
*/
case 'F':
Fflag = 1;
break;
case 'B':
switch (optarg[0]) {
case 'l':
endian = _LITTLE_ENDIAN;
break;
case 'b':
endian = _BIG_ENDIAN;
break;
case 'p':
endian = _PDP_ENDIAN;
break;
}
break;
case 'o': /* datablock, freeblock duplication check */
if (*optarg)
catopt(&options, optarg);
break;
default:
usage();
/*NOTREACHED*/
}
}
argc -= optind;
argv += optind;
if (argc != 1)
usage();
device = argv[0];
if (options) {
if (strstr(options, "data"))
fsck_flags |= V7FS_FSCK_DATABLOCK_DUP;
if (strstr(options, "free"))
fsck_flags |= V7FS_FSCK_FREEBLOCK_DUP;
}
if (Fflag) {
if ((fd = open(device, openflags)) == -1) {
pfatal("%s", device);
}
if (fstat(fd, &st)) {
pfatal("stat");
}
part_sectors = st.st_size >> V7FS_BSHIFT;
setcdevname(device, fsck_operate == PREEN);
} else {
/* blockcheck sets 'hot' */
device = blockcheck(device);
setcdevname(device, fsck_operate == PREEN);
if ((fd = open(device, openflags)) == -1) {
pfatal("%s", device);
}
part = DISKPART(st.st_rdev);
if (ioctl(fd, DIOCGDINFO, &d) == -1) {
pfatal("DIOCGDINFO");
}
p = &d.d_partitions[part];
part_sectors = p->p_size;
VPRINTF("partition=%d size=%d offset=%d fstype=%d secsize=%d\n",
part, p->p_size, p->p_offset, p->p_fstype, d.d_secsize);
if (p->p_fstype != FS_V7) {
pfatal("not a Version 7 partition.");
}
}
progress_switch(progress_bar_enable);
progress_init();
progress(&(struct progress_arg){ .cdev = device });
struct v7fs_mount_device mount;
mount.device.fd = fd;
mount.endian = endian;
mount.sectors = part_sectors;
int error = v7fs_fsck(&mount, fsck_flags);
close(fd);
return error;
}
static void
catopt(char **sp, const char *o)
{
char *s, *n;
s = *sp;
if (s) {
if (asprintf(&n, "%s,%s", s, o) < 0)
err(1, "asprintf");
free(s);
s = n;
} else
s = strdup(o);
*sp = s;
}
static void
usage(void)
{
(void)fprintf(stderr, "usage: %s [-ynpP] [-o options] [-B endian] "
"special-device\n",
getprogname());
(void)fprintf(stderr, "usage: %s -F [-ynpP] [-o options] [-B endian] "
"file\n",
getprogname());
exit(FSCK_EXIT_USAGE);
}