11de7b4b8SPedro F. Giffuni /*- 21de7b4b8SPedro F. Giffuni * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 31de7b4b8SPedro F. Giffuni * 40121b42aSDavid E. O'Brien * Copyright (C) 1995, 1996, 1997 Wolfgang Solfrank 50121b42aSDavid E. O'Brien * Copyright (c) 1995 Martin Husemann 60121b42aSDavid E. O'Brien * 70121b42aSDavid E. O'Brien * Redistribution and use in source and binary forms, with or without 80121b42aSDavid E. O'Brien * modification, are permitted provided that the following conditions 90121b42aSDavid E. O'Brien * are met: 100121b42aSDavid E. O'Brien * 1. Redistributions of source code must retain the above copyright 110121b42aSDavid E. O'Brien * notice, this list of conditions and the following disclaimer. 120121b42aSDavid E. O'Brien * 2. Redistributions in binary form must reproduce the above copyright 130121b42aSDavid E. O'Brien * notice, this list of conditions and the following disclaimer in the 140121b42aSDavid E. O'Brien * documentation and/or other materials provided with the distribution. 150121b42aSDavid E. O'Brien * 160121b42aSDavid E. O'Brien * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR 170121b42aSDavid E. O'Brien * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 180121b42aSDavid E. O'Brien * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 190121b42aSDavid E. O'Brien * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT, 200121b42aSDavid E. O'Brien * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 210121b42aSDavid E. O'Brien * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 220121b42aSDavid E. O'Brien * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 230121b42aSDavid E. O'Brien * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 240121b42aSDavid E. O'Brien * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 250121b42aSDavid E. O'Brien * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 260121b42aSDavid E. O'Brien */ 270121b42aSDavid E. O'Brien 280121b42aSDavid E. O'Brien 290121b42aSDavid E. O'Brien #include <sys/cdefs.h> 300121b42aSDavid E. O'Brien #ifndef lint 316cf357bcSUlrich Spörlein __RCSID("$NetBSD: check.c,v 1.14 2006/06/05 16:51:18 christos Exp $"); 320121b42aSDavid E. O'Brien static const char rcsid[] = 330121b42aSDavid E. O'Brien "$FreeBSD$"; 340121b42aSDavid E. O'Brien #endif /* not lint */ 350121b42aSDavid E. O'Brien 36*9e4029ffSXin LI #ifdef HAVE_LIBUTIL_H 37*9e4029ffSXin LI #include <libutil.h> 38*9e4029ffSXin LI #endif 390121b42aSDavid E. O'Brien #include <stdlib.h> 400121b42aSDavid E. O'Brien #include <string.h> 410121b42aSDavid E. O'Brien #include <stdio.h> 420121b42aSDavid E. O'Brien #include <unistd.h> 430121b42aSDavid E. O'Brien #include <fcntl.h> 440121b42aSDavid E. O'Brien 450121b42aSDavid E. O'Brien #include "ext.h" 460121b42aSDavid E. O'Brien #include "fsutil.h" 470121b42aSDavid E. O'Brien 480121b42aSDavid E. O'Brien int 49565e3e65STom Rhodes checkfilesys(const char *fname) 500121b42aSDavid E. O'Brien { 510121b42aSDavid E. O'Brien int dosfs; 520121b42aSDavid E. O'Brien struct bootblock boot; 539708ba9fSXin LI struct fat_descriptor *fat = NULL; 546069db97SKonstantin Belousov int finish_dosdirsection=0; 550121b42aSDavid E. O'Brien int mod = 0; 560121b42aSDavid E. O'Brien int ret = 8; 570121b42aSDavid E. O'Brien 580121b42aSDavid E. O'Brien rdonly = alwaysno; 590121b42aSDavid E. O'Brien if (!preen) 600121b42aSDavid E. O'Brien printf("** %s", fname); 610121b42aSDavid E. O'Brien 620121b42aSDavid E. O'Brien dosfs = open(fname, rdonly ? O_RDONLY : O_RDWR, 0); 630121b42aSDavid E. O'Brien if (dosfs < 0 && !rdonly) { 640121b42aSDavid E. O'Brien dosfs = open(fname, O_RDONLY, 0); 650121b42aSDavid E. O'Brien if (dosfs >= 0) 660121b42aSDavid E. O'Brien pwarn(" (NO WRITE)\n"); 670121b42aSDavid E. O'Brien else if (!preen) 680121b42aSDavid E. O'Brien printf("\n"); 690121b42aSDavid E. O'Brien rdonly = 1; 700121b42aSDavid E. O'Brien } else if (!preen) 710121b42aSDavid E. O'Brien printf("\n"); 720121b42aSDavid E. O'Brien 730121b42aSDavid E. O'Brien if (dosfs < 0) { 746cf357bcSUlrich Spörlein perr("Can't open `%s'", fname); 752923ae8aSJaakko Heinonen printf("\n"); 760121b42aSDavid E. O'Brien return 8; 770121b42aSDavid E. O'Brien } 780121b42aSDavid E. O'Brien 793e855e9cSXin LI if (readboot(dosfs, &boot) == FSFATAL) { 800121b42aSDavid E. O'Brien close(dosfs); 810121b42aSDavid E. O'Brien printf("\n"); 820121b42aSDavid E. O'Brien return 8; 830121b42aSDavid E. O'Brien } 840121b42aSDavid E. O'Brien 85ae62d940SBruce Evans if (skipclean && preen && checkdirty(dosfs, &boot)) { 86f7bf3122SBruce Evans printf("%s: ", fname); 87cede1f56STom Rhodes printf("FILESYSTEM CLEAN; SKIPPING CHECKS\n"); 88cede1f56STom Rhodes ret = 0; 89cede1f56STom Rhodes goto out; 90cede1f56STom Rhodes } 91cede1f56STom Rhodes 920121b42aSDavid E. O'Brien if (!preen) { 939708ba9fSXin LI printf("** Phase 1 - Read FAT and checking connectivity\n"); 940121b42aSDavid E. O'Brien } 950121b42aSDavid E. O'Brien 969708ba9fSXin LI mod |= readfat(dosfs, &boot, &fat); 970121b42aSDavid E. O'Brien if (mod & FSFATAL) { 980121b42aSDavid E. O'Brien close(dosfs); 990121b42aSDavid E. O'Brien return 8; 1000121b42aSDavid E. O'Brien } 1010121b42aSDavid E. O'Brien 10273db93b8SXin LI if (!preen) 1039708ba9fSXin LI printf("** Phase 2 - Checking Directories\n"); 10473db93b8SXin LI 1059708ba9fSXin LI mod |= resetDosDirSection(fat); 1060121b42aSDavid E. O'Brien finish_dosdirsection = 1; 1070121b42aSDavid E. O'Brien if (mod & FSFATAL) 1080121b42aSDavid E. O'Brien goto out; 1090121b42aSDavid E. O'Brien /* delay writing FATs */ 1100121b42aSDavid E. O'Brien 1119708ba9fSXin LI mod |= handleDirTree(fat); 1120121b42aSDavid E. O'Brien if (mod & FSFATAL) 1130121b42aSDavid E. O'Brien goto out; 1140121b42aSDavid E. O'Brien 1150121b42aSDavid E. O'Brien if (!preen) 1169708ba9fSXin LI printf("** Phase 3 - Checking for Lost Files\n"); 1170121b42aSDavid E. O'Brien 1189708ba9fSXin LI mod |= checklost(fat); 1190121b42aSDavid E. O'Brien if (mod & FSFATAL) 1200121b42aSDavid E. O'Brien goto out; 1210121b42aSDavid E. O'Brien 1220121b42aSDavid E. O'Brien /* now write the FATs */ 1239708ba9fSXin LI if (mod & FSFATMOD) { 1240121b42aSDavid E. O'Brien if (ask(1, "Update FATs")) { 1259708ba9fSXin LI mod |= writefat(fat); 1260121b42aSDavid E. O'Brien if (mod & FSFATAL) 1270121b42aSDavid E. O'Brien goto out; 1280121b42aSDavid E. O'Brien } else 1290121b42aSDavid E. O'Brien mod |= FSERROR; 1300121b42aSDavid E. O'Brien } 1310121b42aSDavid E. O'Brien 132*9e4029ffSXin LI #ifdef HAVE_LIBUTIL_H 133*9e4029ffSXin LI char freestr[7], badstr[7]; 134*9e4029ffSXin LI 135*9e4029ffSXin LI int64_t freebytes = boot.NumFree * boot.ClusterSize; 136*9e4029ffSXin LI humanize_number(freestr, sizeof(freestr), freebytes, "", 137*9e4029ffSXin LI HN_AUTOSCALE, HN_DECIMAL | HN_IEC_PREFIXES); 138*9e4029ffSXin LI if (boot.NumBad) { 139*9e4029ffSXin LI int64_t badbytes = boot.NumBad * boot.ClusterSize; 140*9e4029ffSXin LI 141*9e4029ffSXin LI humanize_number(badstr, sizeof(badstr), badbytes, "", 142*9e4029ffSXin LI HN_AUTOSCALE, HN_B | HN_DECIMAL | HN_IEC_PREFIXES); 143*9e4029ffSXin LI 144*9e4029ffSXin LI pwarn("%d files, %sB free (%d clusters), %sB bad (%d clusters)\n", 145*9e4029ffSXin LI boot.NumFiles, 146*9e4029ffSXin LI freestr, boot.NumFree, 147*9e4029ffSXin LI badstr, boot.NumBad); 148*9e4029ffSXin LI } else { 149*9e4029ffSXin LI pwarn("%d files, %sB free (%d clusters)\n", 150*9e4029ffSXin LI boot.NumFiles, 151*9e4029ffSXin LI freestr, boot.NumFree); 152*9e4029ffSXin LI } 153*9e4029ffSXin LI #else 1540121b42aSDavid E. O'Brien if (boot.NumBad) 155*9e4029ffSXin LI pwarn("%d files, %d KiB free (%d clusters), %d KiB bad (%d clusters)\n", 1560121b42aSDavid E. O'Brien boot.NumFiles, 1570121b42aSDavid E. O'Brien boot.NumFree * boot.ClusterSize / 1024, boot.NumFree, 1580121b42aSDavid E. O'Brien boot.NumBad * boot.ClusterSize / 1024, boot.NumBad); 1590121b42aSDavid E. O'Brien else 160*9e4029ffSXin LI pwarn("%d files, %d KiB free (%d clusters)\n", 1610121b42aSDavid E. O'Brien boot.NumFiles, 1620121b42aSDavid E. O'Brien boot.NumFree * boot.ClusterSize / 1024, boot.NumFree); 163*9e4029ffSXin LI #endif 1640121b42aSDavid E. O'Brien 1650121b42aSDavid E. O'Brien if (mod && (mod & FSERROR) == 0) { 1660121b42aSDavid E. O'Brien if (mod & FSDIRTY) { 1670121b42aSDavid E. O'Brien if (ask(1, "MARK FILE SYSTEM CLEAN") == 0) 1680121b42aSDavid E. O'Brien mod &= ~FSDIRTY; 1690121b42aSDavid E. O'Brien 1700121b42aSDavid E. O'Brien if (mod & FSDIRTY) { 1710121b42aSDavid E. O'Brien pwarn("MARKING FILE SYSTEM CLEAN\n"); 1729708ba9fSXin LI mod |= writefat(fat); 1730121b42aSDavid E. O'Brien } else { 1740121b42aSDavid E. O'Brien pwarn("\n***** FILE SYSTEM IS LEFT MARKED AS DIRTY *****\n"); 1750121b42aSDavid E. O'Brien mod |= FSERROR; /* file system not clean */ 1760121b42aSDavid E. O'Brien } 1770121b42aSDavid E. O'Brien } 1780121b42aSDavid E. O'Brien } 1790121b42aSDavid E. O'Brien 1800121b42aSDavid E. O'Brien if (mod & (FSFATAL | FSERROR)) 1810121b42aSDavid E. O'Brien goto out; 1820121b42aSDavid E. O'Brien 1830121b42aSDavid E. O'Brien ret = 0; 1840121b42aSDavid E. O'Brien 1850121b42aSDavid E. O'Brien out: 1860121b42aSDavid E. O'Brien if (finish_dosdirsection) 1870121b42aSDavid E. O'Brien finishDosDirSection(); 1880121b42aSDavid E. O'Brien free(fat); 1890121b42aSDavid E. O'Brien close(dosfs); 1900121b42aSDavid E. O'Brien 1910121b42aSDavid E. O'Brien if (mod & (FSFATMOD|FSDIRMOD)) 1920121b42aSDavid E. O'Brien pwarn("\n***** FILE SYSTEM WAS MODIFIED *****\n"); 1930121b42aSDavid E. O'Brien 1940121b42aSDavid E. O'Brien return ret; 1950121b42aSDavid E. O'Brien } 196