/* $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); }