10121b42aSDavid E. O'Brien /* 20121b42aSDavid E. O'Brien * Copyright (C) 1995, 1996, 1997 Wolfgang Solfrank 30121b42aSDavid E. O'Brien * Copyright (c) 1995 Martin Husemann 40121b42aSDavid E. O'Brien * 50121b42aSDavid E. O'Brien * Redistribution and use in source and binary forms, with or without 60121b42aSDavid E. O'Brien * modification, are permitted provided that the following conditions 70121b42aSDavid E. O'Brien * are met: 80121b42aSDavid E. O'Brien * 1. Redistributions of source code must retain the above copyright 90121b42aSDavid E. O'Brien * notice, this list of conditions and the following disclaimer. 100121b42aSDavid E. O'Brien * 2. Redistributions in binary form must reproduce the above copyright 110121b42aSDavid E. O'Brien * notice, this list of conditions and the following disclaimer in the 120121b42aSDavid E. O'Brien * documentation and/or other materials provided with the distribution. 130121b42aSDavid E. O'Brien * 3. All advertising materials mentioning features or use of this software 140121b42aSDavid E. O'Brien * must display the following acknowledgement: 150121b42aSDavid E. O'Brien * This product includes software developed by Martin Husemann 160121b42aSDavid E. O'Brien * and Wolfgang Solfrank. 170121b42aSDavid E. O'Brien * 4. Neither the name of the University nor the names of its contributors 180121b42aSDavid E. O'Brien * may be used to endorse or promote products derived from this software 190121b42aSDavid E. O'Brien * without specific prior written permission. 200121b42aSDavid E. O'Brien * 210121b42aSDavid E. O'Brien * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR 220121b42aSDavid E. O'Brien * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 230121b42aSDavid E. O'Brien * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 240121b42aSDavid E. O'Brien * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT, 250121b42aSDavid E. O'Brien * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 260121b42aSDavid E. O'Brien * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 270121b42aSDavid E. O'Brien * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 280121b42aSDavid E. O'Brien * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 290121b42aSDavid E. O'Brien * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 300121b42aSDavid E. O'Brien * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 310121b42aSDavid E. O'Brien */ 320121b42aSDavid E. O'Brien 330121b42aSDavid E. O'Brien 340121b42aSDavid E. O'Brien #include <sys/cdefs.h> 350121b42aSDavid E. O'Brien #ifndef lint 360121b42aSDavid E. O'Brien __RCSID("$NetBSD: check.c,v 1.10 2000/04/25 23:02:51 jdolecek Exp $"); 370121b42aSDavid E. O'Brien static const char rcsid[] = 380121b42aSDavid E. O'Brien "$FreeBSD$"; 390121b42aSDavid E. O'Brien #endif /* not lint */ 400121b42aSDavid E. O'Brien 410121b42aSDavid E. O'Brien #include <stdlib.h> 420121b42aSDavid E. O'Brien #include <string.h> 430121b42aSDavid E. O'Brien #include <ctype.h> 440121b42aSDavid E. O'Brien #include <stdio.h> 450121b42aSDavid E. O'Brien #include <unistd.h> 460121b42aSDavid E. O'Brien #include <fcntl.h> 470121b42aSDavid E. O'Brien 480121b42aSDavid E. O'Brien #include "ext.h" 490121b42aSDavid E. O'Brien #include "fsutil.h" 500121b42aSDavid E. O'Brien 510121b42aSDavid E. O'Brien int 52565e3e65STom Rhodes checkfilesys(const char *fname) 530121b42aSDavid E. O'Brien { 540121b42aSDavid E. O'Brien int dosfs; 550121b42aSDavid E. O'Brien struct bootblock boot; 560121b42aSDavid E. O'Brien struct fatEntry *fat = NULL; 570121b42aSDavid E. O'Brien int i, finish_dosdirsection=0; 580121b42aSDavid E. O'Brien int mod = 0; 590121b42aSDavid E. O'Brien int ret = 8; 600121b42aSDavid E. O'Brien 610121b42aSDavid E. O'Brien rdonly = alwaysno; 620121b42aSDavid E. O'Brien if (!preen) 630121b42aSDavid E. O'Brien printf("** %s", fname); 640121b42aSDavid E. O'Brien 650121b42aSDavid E. O'Brien dosfs = open(fname, rdonly ? O_RDONLY : O_RDWR, 0); 660121b42aSDavid E. O'Brien if (dosfs < 0 && !rdonly) { 670121b42aSDavid E. O'Brien dosfs = open(fname, O_RDONLY, 0); 680121b42aSDavid E. O'Brien if (dosfs >= 0) 690121b42aSDavid E. O'Brien pwarn(" (NO WRITE)\n"); 700121b42aSDavid E. O'Brien else if (!preen) 710121b42aSDavid E. O'Brien printf("\n"); 720121b42aSDavid E. O'Brien rdonly = 1; 730121b42aSDavid E. O'Brien } else if (!preen) 740121b42aSDavid E. O'Brien printf("\n"); 750121b42aSDavid E. O'Brien 760121b42aSDavid E. O'Brien if (dosfs < 0) { 770121b42aSDavid E. O'Brien perror("Can't open"); 780121b42aSDavid E. O'Brien return 8; 790121b42aSDavid E. O'Brien } 800121b42aSDavid E. O'Brien 810121b42aSDavid E. O'Brien if (readboot(dosfs, &boot) != FSOK) { 820121b42aSDavid E. O'Brien close(dosfs); 830121b42aSDavid E. O'Brien printf("\n"); 840121b42aSDavid E. O'Brien return 8; 850121b42aSDavid E. O'Brien } 860121b42aSDavid E. O'Brien 87cede1f56STom Rhodes if (checkdirty(dosfs, &boot) && !force) { 88f7bf3122SBruce Evans if (preen) 89f7bf3122SBruce Evans printf("%s: ", fname); 90cede1f56STom Rhodes printf("FILESYSTEM CLEAN; SKIPPING CHECKS\n"); 91cede1f56STom Rhodes ret = 0; 92cede1f56STom Rhodes goto out; 93cede1f56STom Rhodes } 94cede1f56STom Rhodes 950121b42aSDavid E. O'Brien if (!preen) { 960121b42aSDavid E. O'Brien if (boot.ValidFat < 0) 970121b42aSDavid E. O'Brien printf("** Phase 1 - Read and Compare FATs\n"); 980121b42aSDavid E. O'Brien else 990121b42aSDavid E. O'Brien printf("** Phase 1 - Read FAT\n"); 1000121b42aSDavid E. O'Brien } 1010121b42aSDavid E. O'Brien 1020121b42aSDavid E. O'Brien mod |= readfat(dosfs, &boot, boot.ValidFat >= 0 ? boot.ValidFat : 0, &fat); 1030121b42aSDavid E. O'Brien if (mod & FSFATAL) { 1040121b42aSDavid E. O'Brien close(dosfs); 1050121b42aSDavid E. O'Brien return 8; 1060121b42aSDavid E. O'Brien } 1070121b42aSDavid E. O'Brien 1080121b42aSDavid E. O'Brien if (boot.ValidFat < 0) 109565e3e65STom Rhodes for (i = 1; i < (int)boot.FATs; i++) { 1100121b42aSDavid E. O'Brien struct fatEntry *currentFat; 1110121b42aSDavid E. O'Brien 1120121b42aSDavid E. O'Brien mod |= readfat(dosfs, &boot, i, ¤tFat); 1130121b42aSDavid E. O'Brien 1140121b42aSDavid E. O'Brien if (mod & FSFATAL) 1150121b42aSDavid E. O'Brien goto out; 1160121b42aSDavid E. O'Brien 1170121b42aSDavid E. O'Brien mod |= comparefat(&boot, fat, currentFat, i); 1180121b42aSDavid E. O'Brien free(currentFat); 1190121b42aSDavid E. O'Brien if (mod & FSFATAL) 1200121b42aSDavid E. O'Brien goto out; 1210121b42aSDavid E. O'Brien } 1220121b42aSDavid E. O'Brien 1230121b42aSDavid E. O'Brien if (!preen) 1240121b42aSDavid E. O'Brien printf("** Phase 2 - Check Cluster Chains\n"); 1250121b42aSDavid E. O'Brien 1260121b42aSDavid E. O'Brien mod |= checkfat(&boot, fat); 1270121b42aSDavid E. O'Brien if (mod & FSFATAL) 1280121b42aSDavid E. O'Brien goto out; 1290121b42aSDavid E. O'Brien /* delay writing FATs */ 1300121b42aSDavid E. O'Brien 1310121b42aSDavid E. O'Brien if (!preen) 1320121b42aSDavid E. O'Brien printf("** Phase 3 - Checking Directories\n"); 1330121b42aSDavid E. O'Brien 1340121b42aSDavid E. O'Brien mod |= resetDosDirSection(&boot, fat); 1350121b42aSDavid E. O'Brien finish_dosdirsection = 1; 1360121b42aSDavid E. O'Brien if (mod & FSFATAL) 1370121b42aSDavid E. O'Brien goto out; 1380121b42aSDavid E. O'Brien /* delay writing FATs */ 1390121b42aSDavid E. O'Brien 1400121b42aSDavid E. O'Brien mod |= handleDirTree(dosfs, &boot, fat); 1410121b42aSDavid E. O'Brien if (mod & FSFATAL) 1420121b42aSDavid E. O'Brien goto out; 1430121b42aSDavid E. O'Brien 1440121b42aSDavid E. O'Brien if (!preen) 1450121b42aSDavid E. O'Brien printf("** Phase 4 - Checking for Lost Files\n"); 1460121b42aSDavid E. O'Brien 1470121b42aSDavid E. O'Brien mod |= checklost(dosfs, &boot, fat); 1480121b42aSDavid E. O'Brien if (mod & FSFATAL) 1490121b42aSDavid E. O'Brien goto out; 1500121b42aSDavid E. O'Brien 1510121b42aSDavid E. O'Brien /* now write the FATs */ 1520121b42aSDavid E. O'Brien if (mod & FSFATMOD) { 1530121b42aSDavid E. O'Brien if (ask(1, "Update FATs")) { 1540121b42aSDavid E. O'Brien mod |= writefat(dosfs, &boot, fat, mod & FSFIXFAT); 1550121b42aSDavid E. O'Brien if (mod & FSFATAL) 1560121b42aSDavid E. O'Brien goto out; 1570121b42aSDavid E. O'Brien } else 1580121b42aSDavid E. O'Brien mod |= FSERROR; 1590121b42aSDavid E. O'Brien } 1600121b42aSDavid E. O'Brien 1610121b42aSDavid E. O'Brien if (boot.NumBad) 1620121b42aSDavid E. O'Brien pwarn("%d files, %d free (%d clusters), %d bad (%d clusters)\n", 1630121b42aSDavid E. O'Brien boot.NumFiles, 1640121b42aSDavid E. O'Brien boot.NumFree * boot.ClusterSize / 1024, boot.NumFree, 1650121b42aSDavid E. O'Brien boot.NumBad * boot.ClusterSize / 1024, boot.NumBad); 1660121b42aSDavid E. O'Brien else 1670121b42aSDavid E. O'Brien pwarn("%d files, %d free (%d clusters)\n", 1680121b42aSDavid E. O'Brien boot.NumFiles, 1690121b42aSDavid E. O'Brien boot.NumFree * boot.ClusterSize / 1024, boot.NumFree); 1700121b42aSDavid E. O'Brien 1710121b42aSDavid E. O'Brien if (mod && (mod & FSERROR) == 0) { 1720121b42aSDavid E. O'Brien if (mod & FSDIRTY) { 1730121b42aSDavid E. O'Brien if (ask(1, "MARK FILE SYSTEM CLEAN") == 0) 1740121b42aSDavid E. O'Brien mod &= ~FSDIRTY; 1750121b42aSDavid E. O'Brien 1760121b42aSDavid E. O'Brien if (mod & FSDIRTY) { 1770121b42aSDavid E. O'Brien pwarn("MARKING FILE SYSTEM CLEAN\n"); 1780121b42aSDavid E. O'Brien mod |= writefat(dosfs, &boot, fat, 1); 1790121b42aSDavid E. O'Brien } else { 1800121b42aSDavid E. O'Brien pwarn("\n***** FILE SYSTEM IS LEFT MARKED AS DIRTY *****\n"); 1810121b42aSDavid E. O'Brien mod |= FSERROR; /* file system not clean */ 1820121b42aSDavid E. O'Brien } 1830121b42aSDavid E. O'Brien } 1840121b42aSDavid E. O'Brien } 1850121b42aSDavid E. O'Brien 1860121b42aSDavid E. O'Brien if (mod & (FSFATAL | FSERROR)) 1870121b42aSDavid E. O'Brien goto out; 1880121b42aSDavid E. O'Brien 1890121b42aSDavid E. O'Brien ret = 0; 1900121b42aSDavid E. O'Brien 1910121b42aSDavid E. O'Brien out: 1920121b42aSDavid E. O'Brien if (finish_dosdirsection) 1930121b42aSDavid E. O'Brien finishDosDirSection(); 1940121b42aSDavid E. O'Brien free(fat); 1950121b42aSDavid E. O'Brien close(dosfs); 1960121b42aSDavid E. O'Brien 1970121b42aSDavid E. O'Brien if (mod & (FSFATMOD|FSDIRMOD)) 1980121b42aSDavid E. O'Brien pwarn("\n***** FILE SYSTEM WAS MODIFIED *****\n"); 1990121b42aSDavid E. O'Brien 2000121b42aSDavid E. O'Brien return ret; 2010121b42aSDavid E. O'Brien } 202cede1f56STom Rhodes 203f7bf3122SBruce Evans int 204f7bf3122SBruce Evans checkdirty(int fs, struct bootblock *boot) 205cede1f56STom Rhodes { 206cede1f56STom Rhodes off_t off; 207cede1f56STom Rhodes u_char *buffer; 208cede1f56STom Rhodes int ret = 0; 209cede1f56STom Rhodes 210cede1f56STom Rhodes if (boot->ClustMask == CLUST12_MASK) 211cede1f56STom Rhodes return 0; 212cede1f56STom Rhodes 213cede1f56STom Rhodes off = boot->ResSectors; 214cede1f56STom Rhodes off *= boot->BytesPerSec; 215cede1f56STom Rhodes 216cede1f56STom Rhodes buffer = malloc(boot->BytesPerSec); 217cede1f56STom Rhodes if (buffer == NULL) { 218cede1f56STom Rhodes perror("No space for FAT"); 219cede1f56STom Rhodes return 1; 220cede1f56STom Rhodes } 221cede1f56STom Rhodes 222cede1f56STom Rhodes if (lseek(fs, off, SEEK_SET) != off) { 223cede1f56STom Rhodes perror("Unable to read FAT"); 224cede1f56STom Rhodes goto err; 225cede1f56STom Rhodes } 226cede1f56STom Rhodes 227f7bf3122SBruce Evans if (read(fs, buffer, boot->BytesPerSec) != boot->BytesPerSec) { 228cede1f56STom Rhodes perror("Unable to read FAT"); 229cede1f56STom Rhodes goto err; 230cede1f56STom Rhodes } 231cede1f56STom Rhodes 232f7bf3122SBruce Evans if (buffer[0] == boot->Media && buffer[1] == 0xff && 233f7bf3122SBruce Evans buffer[2] == 0xff && 234f7bf3122SBruce Evans ((boot->ClustMask == CLUST16_MASK && buffer[3] == 0x7f) || 235f7bf3122SBruce Evans (boot->ClustMask == CLUST32_MASK && buffer[3] == 0x0f && 236f7bf3122SBruce Evans buffer[4] == 0xff && buffer[5] == 0xff && buffer[6] == 0xff && 237f7bf3122SBruce Evans buffer[7] == 0x07))) 238cede1f56STom Rhodes ret = 0; 239cede1f56STom Rhodes else 240cede1f56STom Rhodes ret = 1; 241cede1f56STom Rhodes 242cede1f56STom Rhodes err: 243cede1f56STom Rhodes free(buffer); 244cede1f56STom Rhodes return ret; 245cede1f56STom Rhodes } 246