1 /* $NetBSD: check.c,v 1.13 2005/01/19 20:00:45 xtraeme Exp $ */ 2 3 /* 4 * Copyright (C) 1995, 1996, 1997 Wolfgang Solfrank 5 * Copyright (c) 1995 Martin Husemann 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions and the following disclaimer. 12 * 2. Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 3. All advertising materials mentioning features or use of this software 16 * must display the following acknowledgement: 17 * This product includes software developed by Martin Husemann 18 * and Wolfgang Solfrank. 19 * 4. Neither the name of the University nor the names of its contributors 20 * may be used to endorse or promote products derived from this software 21 * without specific prior written permission. 22 * 23 * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ``AS IS'' AND ANY EXPRESS OR 24 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 25 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 26 * IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY DIRECT, INDIRECT, 27 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT 28 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 29 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 30 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 31 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 32 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 33 */ 34 35 36 #include <sys/cdefs.h> 37 #ifndef lint 38 __RCSID("$NetBSD: check.c,v 1.13 2005/01/19 20:00:45 xtraeme Exp $"); 39 #endif /* not lint */ 40 41 #include <stdlib.h> 42 #include <string.h> 43 #include <ctype.h> 44 #include <stdio.h> 45 #include <unistd.h> 46 #include <fcntl.h> 47 48 #include "ext.h" 49 #include "fsutil.h" 50 51 int 52 checkfilesys(const char *filename) 53 { 54 int dosfs; 55 struct bootblock boot; 56 struct fatEntry *fat = NULL; 57 int i, finish_dosdirsection=0; 58 int mod = 0; 59 int ret = 8; 60 61 rdonly = alwaysno; 62 if (!preen) 63 printf("** %s", filename); 64 65 dosfs = open(filename, rdonly ? O_RDONLY : O_RDWR, 0); 66 if (dosfs < 0 && !rdonly) { 67 dosfs = open(filename, O_RDONLY, 0); 68 if (dosfs >= 0) 69 pwarn(" (NO WRITE)\n"); 70 else if (!preen) 71 printf("\n"); 72 rdonly = 1; 73 } else if (!preen) 74 printf("\n"); 75 76 if (dosfs < 0) { 77 perror("Can't open"); 78 return 8; 79 } 80 81 if (readboot(dosfs, &boot) != FSOK) { 82 close(dosfs); 83 printf("\n"); 84 return 8; 85 } 86 87 if (!preen) { 88 if (boot.ValidFat < 0) 89 printf("** Phase 1 - Read and Compare FATs\n"); 90 else 91 printf("** Phase 1 - Read FAT\n"); 92 } 93 94 mod |= readfat(dosfs, &boot, boot.ValidFat >= 0 ? boot.ValidFat : 0, &fat); 95 if (mod & FSFATAL) { 96 close(dosfs); 97 return 8; 98 } 99 100 if (boot.ValidFat < 0) 101 for (i = 1; i < boot.FATs; i++) { 102 struct fatEntry *currentFat; 103 104 mod |= readfat(dosfs, &boot, i, ¤tFat); 105 106 if (mod & FSFATAL) 107 goto out; 108 109 mod |= comparefat(&boot, fat, currentFat, i); 110 free(currentFat); 111 if (mod & FSFATAL) 112 goto out; 113 } 114 115 if (!preen) 116 printf("** Phase 2 - Check Cluster Chains\n"); 117 118 mod |= checkfat(&boot, fat); 119 if (mod & FSFATAL) 120 goto out; 121 /* delay writing FATs */ 122 123 if (!preen) 124 printf("** Phase 3 - Checking Directories\n"); 125 126 mod |= resetDosDirSection(&boot, fat); 127 finish_dosdirsection = 1; 128 if (mod & FSFATAL) 129 goto out; 130 /* delay writing FATs */ 131 132 mod |= handleDirTree(dosfs, &boot, fat); 133 if (mod & FSFATAL) 134 goto out; 135 136 if (!preen) 137 printf("** Phase 4 - Checking for Lost Files\n"); 138 139 mod |= checklost(dosfs, &boot, fat); 140 if (mod & FSFATAL) 141 goto out; 142 143 /* now write the FATs */ 144 if (mod & FSFATMOD) { 145 if (ask(1, "Update FATs")) { 146 mod |= writefat(dosfs, &boot, fat, mod & FSFIXFAT); 147 if (mod & FSFATAL) 148 goto out; 149 } else 150 mod |= FSERROR; 151 } 152 153 if (boot.NumBad) 154 pwarn("%d files, %d free (%d clusters), %d bad (%d clusters)\n", 155 boot.NumFiles, 156 boot.NumFree * boot.ClusterSize / 1024, boot.NumFree, 157 boot.NumBad * boot.ClusterSize / 1024, boot.NumBad); 158 else 159 pwarn("%d files, %d free (%d clusters)\n", 160 boot.NumFiles, 161 boot.NumFree * boot.ClusterSize / 1024, boot.NumFree); 162 163 if (mod && (mod & FSERROR) == 0) { 164 if (mod & FSDIRTY) { 165 if (ask(1, "MARK FILE SYSTEM CLEAN") == 0) 166 mod &= ~FSDIRTY; 167 168 if (mod & FSDIRTY) { 169 pwarn("MARKING FILE SYSTEM CLEAN\n"); 170 mod |= writefat(dosfs, &boot, fat, 1); 171 } else { 172 pwarn("\n***** FILE SYSTEM IS LEFT MARKED AS DIRTY *****\n"); 173 mod |= FSERROR; /* file system not clean */ 174 } 175 } 176 } 177 178 if (mod & (FSFATAL | FSERROR)) 179 goto out; 180 181 ret = 0; 182 183 out: 184 if (finish_dosdirsection) 185 finishDosDirSection(); 186 free(fat); 187 close(dosfs); 188 189 if (mod & (FSFATMOD|FSDIRMOD)) 190 pwarn("\n***** FILE SYSTEM WAS MODIFIED *****\n"); 191 192 return ret; 193 } 194