1 /*- 2 * Copyright (c) 1980, 1988, 1991, 1993 3 * The Regents of the University of California. All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions 7 * are met: 8 * 1. Redistributions of source code must retain the above copyright 9 * notice, this list of conditions and the following disclaimer. 10 * 2. Redistributions in binary form must reproduce the above copyright 11 * notice, this list of conditions and the following disclaimer in the 12 * documentation and/or other materials provided with the distribution. 13 * 3. All advertising materials mentioning features or use of this software 14 * must display the following acknowledgement: 15 * This product includes software developed by the University of 16 * California, Berkeley and its contributors. 17 * 4. Neither the name of the University nor the names of its contributors 18 * may be used to endorse or promote products derived from this software 19 * without specific prior written permission. 20 * 21 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 22 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 24 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 25 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 26 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 27 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 28 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 29 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 30 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 31 * SUCH DAMAGE. 32 */ 33 34 #ifndef lint 35 #if 0 36 static char sccsid[] = "@(#)traverse.c 8.7 (Berkeley) 6/15/95"; 37 #endif 38 static const char rcsid[] = 39 "$FreeBSD$"; 40 #endif /* not lint */ 41 42 #include <sys/param.h> 43 #include <sys/stat.h> 44 45 #include <ufs/ufs/dir.h> 46 #include <ufs/ufs/dinode.h> 47 #include <ufs/ffs/fs.h> 48 49 #include <protocols/dumprestore.h> 50 51 #include <ctype.h> 52 #include <stdio.h> 53 #include <errno.h> 54 #include <string.h> 55 #include <unistd.h> 56 #include <stdlib.h> 57 58 #include "dump.h" 59 60 union dinode { 61 struct ufs1_dinode dp1; 62 struct ufs2_dinode dp2; 63 }; 64 #define DIP(dp, field) \ 65 ((sblock->fs_magic == FS_UFS1_MAGIC) ? \ 66 (dp)->dp1.field : (dp)->dp2.field) 67 68 #define HASDUMPEDFILE 0x1 69 #define HASSUBDIRS 0x2 70 71 static int dirindir(ino_t ino, ufs2_daddr_t blkno, int level, long *size, 72 long *tapesize, int nodump); 73 static void dmpindir(ino_t ino, ufs2_daddr_t blk, int level, off_t *size); 74 static int searchdir(ino_t ino, ufs2_daddr_t blkno, long size, long filesize, 75 long *tapesize, int nodump); 76 77 /* 78 * This is an estimation of the number of TP_BSIZE blocks in the file. 79 * It estimates the number of blocks in files with holes by assuming 80 * that all of the blocks accounted for by di_blocks are data blocks 81 * (when some of the blocks are usually used for indirect pointers); 82 * hence the estimate may be high. 83 */ 84 long 85 blockest(union dinode *dp) 86 { 87 long blkest, sizeest; 88 89 /* 90 * dp->di_size is the size of the file in bytes. 91 * dp->di_blocks stores the number of sectors actually in the file. 92 * If there are more sectors than the size would indicate, this just 93 * means that there are indirect blocks in the file or unused 94 * sectors in the last file block; we can safely ignore these 95 * (blkest = sizeest below). 96 * If the file is bigger than the number of sectors would indicate, 97 * then the file has holes in it. In this case we must use the 98 * block count to estimate the number of data blocks used, but 99 * we use the actual size for estimating the number of indirect 100 * dump blocks (sizeest vs. blkest in the indirect block 101 * calculation). 102 */ 103 blkest = howmany(dbtob(DIP(dp, di_blocks)), TP_BSIZE); 104 sizeest = howmany(DIP(dp, di_size), TP_BSIZE); 105 if (blkest > sizeest) 106 blkest = sizeest; 107 if (DIP(dp, di_size) > sblock->fs_bsize * NDADDR) { 108 /* calculate the number of indirect blocks on the dump tape */ 109 blkest += 110 howmany(sizeest - NDADDR * sblock->fs_bsize / TP_BSIZE, 111 TP_NINDIR); 112 } 113 return (blkest + 1); 114 } 115 116 /* Auxiliary macro to pick up files changed since previous dump. */ 117 #define CHANGEDSINCE(dp, t) \ 118 (DIP(dp, di_mtime) >= (t) || DIP(dp, di_ctime) >= (t)) 119 120 /* The WANTTODUMP macro decides whether a file should be dumped. */ 121 #ifdef UF_NODUMP 122 #define WANTTODUMP(dp) \ 123 (CHANGEDSINCE(dp, spcl.c_ddate) && \ 124 (nonodump || (DIP(dp, di_flags) & UF_NODUMP) != UF_NODUMP)) 125 #else 126 #define WANTTODUMP(dp) CHANGEDSINCE(dp, spcl.c_ddate) 127 #endif 128 129 /* 130 * Dump pass 1. 131 * 132 * Walk the inode list for a filesystem to find all allocated inodes 133 * that have been modified since the previous dump time. Also, find all 134 * the directories in the filesystem. 135 */ 136 int 137 mapfiles(ino_t maxino, long *tapesize) 138 { 139 int mode; 140 ino_t ino; 141 union dinode *dp; 142 int anydirskipped = 0; 143 144 for (ino = ROOTINO; ino < maxino; ino++) { 145 dp = getino(ino, &mode); 146 if (mode == 0) 147 continue; 148 /* 149 * Everything must go in usedinomap so that a check 150 * for "in dumpdirmap but not in usedinomap" to detect 151 * dirs with nodump set has a chance of succeeding 152 * (this is used in mapdirs()). 153 */ 154 SETINO(ino, usedinomap); 155 if (mode == IFDIR) 156 SETINO(ino, dumpdirmap); 157 if (WANTTODUMP(dp)) { 158 SETINO(ino, dumpinomap); 159 if (mode != IFREG && mode != IFDIR && mode != IFLNK) 160 *tapesize += 1; 161 else 162 *tapesize += blockest(dp); 163 continue; 164 } 165 if (mode == IFDIR) { 166 if (!nonodump && (DIP(dp, di_flags) & UF_NODUMP)) 167 CLRINO(ino, usedinomap); 168 anydirskipped = 1; 169 } 170 } 171 /* 172 * Restore gets very upset if the root is not dumped, 173 * so ensure that it always is dumped. 174 */ 175 SETINO(ROOTINO, dumpinomap); 176 return (anydirskipped); 177 } 178 179 /* 180 * Dump pass 2. 181 * 182 * Scan each directory on the filesystem to see if it has any modified 183 * files in it. If it does, and has not already been added to the dump 184 * list (because it was itself modified), then add it. If a directory 185 * has not been modified itself, contains no modified files and has no 186 * subdirectories, then it can be deleted from the dump list and from 187 * the list of directories. By deleting it from the list of directories, 188 * its parent may now qualify for the same treatment on this or a later 189 * pass using this algorithm. 190 */ 191 int 192 mapdirs(ino_t maxino, long *tapesize) 193 { 194 union dinode *dp; 195 int i, isdir, nodump; 196 char *map; 197 ino_t ino; 198 union dinode di; 199 long filesize; 200 int ret, change = 0; 201 202 isdir = 0; /* XXX just to get gcc to shut up */ 203 for (map = dumpdirmap, ino = 1; ino < maxino; ino++) { 204 if (((ino - 1) % NBBY) == 0) /* map is offset by 1 */ 205 isdir = *map++; 206 else 207 isdir >>= 1; 208 /* 209 * If a directory has been removed from usedinomap, it 210 * either has the nodump flag set, or has inherited 211 * it. Although a directory can't be in dumpinomap if 212 * it isn't in usedinomap, we have to go through it to 213 * propagate the nodump flag. 214 */ 215 nodump = !nonodump && (TSTINO(ino, usedinomap) == 0); 216 if ((isdir & 1) == 0 || (TSTINO(ino, dumpinomap) && !nodump)) 217 continue; 218 dp = getino(ino, &i); 219 /* 220 * inode buf may change in searchdir(). 221 */ 222 if (sblock->fs_magic == FS_UFS1_MAGIC) 223 di.dp1 = dp->dp1; 224 else 225 di.dp2 = dp->dp2; 226 filesize = DIP(&di, di_size); 227 for (ret = 0, i = 0; filesize > 0 && i < NDADDR; i++) { 228 if (DIP(&di, di_db[i]) != 0) 229 ret |= searchdir(ino, DIP(&di, di_db[i]), 230 (long)sblksize(sblock, DIP(dp, di_size), i), 231 filesize, tapesize, nodump); 232 if (ret & HASDUMPEDFILE) 233 filesize = 0; 234 else 235 filesize -= sblock->fs_bsize; 236 } 237 for (i = 0; filesize > 0 && i < NIADDR; i++) { 238 if (DIP(&di, di_ib[i]) == 0) 239 continue; 240 ret |= dirindir(ino, DIP(&di, di_ib[i]), i, &filesize, 241 tapesize, nodump); 242 } 243 if (ret & HASDUMPEDFILE) { 244 SETINO(ino, dumpinomap); 245 *tapesize += blockest(dp); 246 change = 1; 247 continue; 248 } 249 if (nodump) { 250 if (ret & HASSUBDIRS) 251 change = 1; /* subdirs inherit nodump */ 252 CLRINO(ino, dumpdirmap); 253 } else if ((ret & HASSUBDIRS) == 0) 254 if (!TSTINO(ino, dumpinomap)) { 255 CLRINO(ino, dumpdirmap); 256 change = 1; 257 } 258 } 259 return (change); 260 } 261 262 /* 263 * Read indirect blocks, and pass the data blocks to be searched 264 * as directories. Quit as soon as any entry is found that will 265 * require the directory to be dumped. 266 */ 267 static int 268 dirindir( 269 ino_t ino, 270 ufs2_daddr_t blkno, 271 int ind_level, 272 long *filesize, 273 long *tapesize, 274 int nodump) 275 { 276 int ret = 0; 277 int i; 278 static caddr_t idblk; 279 280 if (idblk == NULL && (idblk = malloc(sblock->fs_bsize)) == NULL) 281 quit("dirindir: cannot allocate indirect memory.\n"); 282 bread(fsbtodb(sblock, blkno), idblk, (int)sblock->fs_bsize); 283 if (ind_level <= 0) { 284 for (i = 0; *filesize > 0 && i < NINDIR(sblock); i++) { 285 if (sblock->fs_magic == FS_UFS1_MAGIC) 286 blkno = ((ufs1_daddr_t *)idblk)[i]; 287 else 288 blkno = ((ufs2_daddr_t *)idblk)[i]; 289 if (blkno != 0) 290 ret |= searchdir(ino, blkno, sblock->fs_bsize, 291 *filesize, tapesize, nodump); 292 if (ret & HASDUMPEDFILE) 293 *filesize = 0; 294 else 295 *filesize -= sblock->fs_bsize; 296 } 297 return (ret); 298 } 299 ind_level--; 300 for (i = 0; *filesize > 0 && i < NINDIR(sblock); i++) { 301 if (sblock->fs_magic == FS_UFS1_MAGIC) 302 blkno = ((ufs1_daddr_t *)idblk)[i]; 303 else 304 blkno = ((ufs2_daddr_t *)idblk)[i]; 305 if (blkno != 0) 306 ret |= dirindir(ino, blkno, ind_level, filesize, 307 tapesize, nodump); 308 } 309 return (ret); 310 } 311 312 /* 313 * Scan a disk block containing directory information looking to see if 314 * any of the entries are on the dump list and to see if the directory 315 * contains any subdirectories. 316 */ 317 static int 318 searchdir( 319 ino_t ino, 320 ufs2_daddr_t blkno, 321 long size, 322 long filesize, 323 long *tapesize, 324 int nodump) 325 { 326 int mode; 327 struct direct *dp; 328 union dinode *ip; 329 long loc, ret = 0; 330 static caddr_t dblk; 331 332 if (dblk == NULL && (dblk = malloc(sblock->fs_bsize)) == NULL) 333 quit("searchdir: cannot allocate indirect memory.\n"); 334 bread(fsbtodb(sblock, blkno), dblk, (int)size); 335 if (filesize < size) 336 size = filesize; 337 for (loc = 0; loc < size; ) { 338 dp = (struct direct *)(dblk + loc); 339 if (dp->d_reclen == 0) { 340 msg("corrupted directory, inumber %d\n", ino); 341 break; 342 } 343 loc += dp->d_reclen; 344 if (dp->d_ino == 0) 345 continue; 346 if (dp->d_name[0] == '.') { 347 if (dp->d_name[1] == '\0') 348 continue; 349 if (dp->d_name[1] == '.' && dp->d_name[2] == '\0') 350 continue; 351 } 352 if (nodump) { 353 ip = getino(dp->d_ino, &mode); 354 if (TSTINO(dp->d_ino, dumpinomap)) { 355 CLRINO(dp->d_ino, dumpinomap); 356 *tapesize -= blockest(ip); 357 } 358 /* 359 * Add back to dumpdirmap and remove from usedinomap 360 * to propagate nodump. 361 */ 362 if (mode == IFDIR) { 363 SETINO(dp->d_ino, dumpdirmap); 364 CLRINO(dp->d_ino, usedinomap); 365 ret |= HASSUBDIRS; 366 } 367 } else { 368 if (TSTINO(dp->d_ino, dumpinomap)) { 369 ret |= HASDUMPEDFILE; 370 if (ret & HASSUBDIRS) 371 break; 372 } 373 if (TSTINO(dp->d_ino, dumpdirmap)) { 374 ret |= HASSUBDIRS; 375 if (ret & HASDUMPEDFILE) 376 break; 377 } 378 } 379 } 380 return (ret); 381 } 382 383 /* 384 * Dump passes 3 and 4. 385 * 386 * Dump the contents of an inode to tape. 387 */ 388 void 389 dumpino(union dinode *dp, ino_t ino) 390 { 391 struct ufs1_dinode *dp1; 392 struct ufs2_dinode *dp2; 393 int ind_level, cnt; 394 off_t size; 395 char buf[TP_BSIZE]; 396 397 if (newtape) { 398 newtape = 0; 399 dumpmap(dumpinomap, TS_BITS, ino); 400 } 401 CLRINO(ino, dumpinomap); 402 if (sblock->fs_magic == FS_UFS1_MAGIC) { 403 spcl.c_mode = dp->dp1.di_mode; 404 spcl.c_size = dp->dp1.di_size; 405 spcl.c_atime = _time32_to_time(dp->dp1.di_atime); 406 spcl.c_atimensec = dp->dp1.di_atimensec; 407 spcl.c_mtime = _time32_to_time(dp->dp1.di_mtime); 408 spcl.c_mtimensec = dp->dp1.di_mtimensec; 409 spcl.c_createtime = 0; 410 spcl.c_createtimensec = 0; 411 spcl.c_rdev = dp->dp1.di_rdev; 412 spcl.c_file_flags = dp->dp1.di_flags; 413 spcl.c_uid = dp->dp1.di_uid; 414 spcl.c_gid = dp->dp1.di_gid; 415 } else { 416 spcl.c_mode = dp->dp2.di_mode; 417 spcl.c_size = dp->dp2.di_size; 418 spcl.c_atime = _time64_to_time(dp->dp2.di_atime); 419 spcl.c_atimensec = dp->dp2.di_atimensec; 420 spcl.c_mtime = _time64_to_time(dp->dp2.di_mtime); 421 spcl.c_mtimensec = dp->dp2.di_mtimensec; 422 spcl.c_createtime = _time64_to_time(dp->dp2.di_createtime); 423 spcl.c_createtimensec = dp->dp2.di_creatensec; 424 spcl.c_rdev = dp->dp2.di_rdev; 425 spcl.c_file_flags = dp->dp2.di_flags; 426 spcl.c_uid = dp->dp2.di_uid; 427 spcl.c_gid = dp->dp2.di_gid; 428 } 429 spcl.c_type = TS_INODE; 430 spcl.c_count = 0; 431 switch (DIP(dp, di_mode) & S_IFMT) { 432 433 case 0: 434 /* 435 * Freed inode. 436 */ 437 return; 438 439 case S_IFLNK: 440 /* 441 * Check for short symbolic link. 442 */ 443 #ifdef FS_44INODEFMT 444 if (DIP(dp, di_size) > 0 && 445 DIP(dp, di_size) < sblock->fs_maxsymlinklen) { 446 spcl.c_addr[0] = 1; 447 spcl.c_count = 1; 448 writeheader(ino); 449 if (sblock->fs_magic == FS_UFS1_MAGIC) 450 memmove(buf, (caddr_t)dp->dp1.di_db, 451 (u_long)DIP(dp, di_size)); 452 else 453 memmove(buf, (caddr_t)dp->dp2.di_db, 454 (u_long)DIP(dp, di_size)); 455 buf[DIP(dp, di_size)] = '\0'; 456 writerec(buf, 0); 457 return; 458 } 459 #endif 460 /* fall through */ 461 462 case S_IFDIR: 463 case S_IFREG: 464 if (DIP(dp, di_size) > 0) 465 break; 466 /* fall through */ 467 468 case S_IFIFO: 469 case S_IFSOCK: 470 case S_IFCHR: 471 case S_IFBLK: 472 writeheader(ino); 473 return; 474 475 default: 476 msg("Warning: undefined file type 0%o\n", 477 DIP(dp, di_mode) & IFMT); 478 return; 479 } 480 if (DIP(dp, di_size) > NDADDR * sblock->fs_bsize) 481 cnt = NDADDR * sblock->fs_frag; 482 else 483 cnt = howmany(DIP(dp, di_size), sblock->fs_fsize); 484 if (sblock->fs_magic == FS_UFS1_MAGIC) 485 ufs1_blksout(&dp->dp1.di_db[0], cnt, ino); 486 else 487 ufs2_blksout(&dp->dp2.di_db[0], cnt, ino); 488 if ((size = DIP(dp, di_size) - NDADDR * sblock->fs_bsize) <= 0) 489 return; 490 for (ind_level = 0; ind_level < NIADDR; ind_level++) { 491 dmpindir(ino, DIP(dp, di_ib[ind_level]), ind_level, &size); 492 if (size <= 0) 493 return; 494 } 495 } 496 497 /* 498 * Read indirect blocks, and pass the data blocks to be dumped. 499 */ 500 static void 501 dmpindir(ino_t ino, ufs2_daddr_t blk, int ind_level, off_t *size) 502 { 503 int i, cnt; 504 static caddr_t idblk; 505 506 if (idblk == NULL && (idblk = malloc(sblock->fs_bsize)) == NULL) 507 quit("dmpindir: cannot allocate indirect memory.\n"); 508 if (blk != 0) 509 bread(fsbtodb(sblock, blk), idblk, (int) sblock->fs_bsize); 510 else 511 memset(idblk, 0, (int)sblock->fs_bsize); 512 if (ind_level <= 0) { 513 if (*size < NINDIR(sblock) * sblock->fs_bsize) 514 cnt = howmany(*size, sblock->fs_fsize); 515 else 516 cnt = NINDIR(sblock) * sblock->fs_frag; 517 *size -= NINDIR(sblock) * sblock->fs_bsize; 518 if (sblock->fs_magic == FS_UFS1_MAGIC) 519 ufs1_blksout((ufs1_daddr_t *)idblk, cnt, ino); 520 else 521 ufs2_blksout((ufs2_daddr_t *)idblk, cnt, ino); 522 return; 523 } 524 ind_level--; 525 for (i = 0; i < NINDIR(sblock); i++) { 526 if (sblock->fs_magic == FS_UFS1_MAGIC) 527 dmpindir(ino, ((ufs1_daddr_t *)idblk)[i], ind_level, 528 size); 529 else 530 dmpindir(ino, ((ufs2_daddr_t *)idblk)[i], ind_level, 531 size); 532 if (*size <= 0) 533 return; 534 } 535 } 536 537 /* 538 * Collect up the data into tape record sized buffers and output them. 539 */ 540 void 541 ufs1_blksout(ufs1_daddr_t *blkp, int frags, ino_t ino) 542 { 543 ufs1_daddr_t *bp; 544 int i, j, count, blks, tbperdb; 545 546 blks = howmany(frags * sblock->fs_fsize, TP_BSIZE); 547 tbperdb = sblock->fs_bsize >> tp_bshift; 548 for (i = 0; i < blks; i += TP_NINDIR) { 549 if (i + TP_NINDIR > blks) 550 count = blks; 551 else 552 count = i + TP_NINDIR; 553 for (j = i; j < count; j++) 554 if (blkp[j / tbperdb] != 0) 555 spcl.c_addr[j - i] = 1; 556 else 557 spcl.c_addr[j - i] = 0; 558 spcl.c_count = count - i; 559 writeheader(ino); 560 bp = &blkp[i / tbperdb]; 561 for (j = i; j < count; j += tbperdb, bp++) 562 if (*bp != 0) { 563 if (j + tbperdb <= count) 564 dumpblock(*bp, (int)sblock->fs_bsize); 565 else 566 dumpblock(*bp, (count - j) * TP_BSIZE); 567 } 568 spcl.c_type = TS_ADDR; 569 } 570 } 571 572 /* 573 * Collect up the data into tape record sized buffers and output them. 574 */ 575 void 576 ufs2_blksout(ufs2_daddr_t *blkp, int frags, ino_t ino) 577 { 578 ufs2_daddr_t *bp; 579 int i, j, count, blks, tbperdb; 580 581 blks = howmany(frags * sblock->fs_fsize, TP_BSIZE); 582 tbperdb = sblock->fs_bsize >> tp_bshift; 583 for (i = 0; i < blks; i += TP_NINDIR) { 584 if (i + TP_NINDIR > blks) 585 count = blks; 586 else 587 count = i + TP_NINDIR; 588 for (j = i; j < count; j++) 589 if (blkp[j / tbperdb] != 0) 590 spcl.c_addr[j - i] = 1; 591 else 592 spcl.c_addr[j - i] = 0; 593 spcl.c_count = count - i; 594 writeheader(ino); 595 bp = &blkp[i / tbperdb]; 596 for (j = i; j < count; j += tbperdb, bp++) 597 if (*bp != 0) { 598 if (j + tbperdb <= count) 599 dumpblock(*bp, (int)sblock->fs_bsize); 600 else 601 dumpblock(*bp, (count - j) * TP_BSIZE); 602 } 603 spcl.c_type = TS_ADDR; 604 } 605 } 606 607 /* 608 * Dump a map to the tape. 609 */ 610 void 611 dumpmap(char *map, int type, ino_t ino) 612 { 613 int i; 614 char *cp; 615 616 spcl.c_type = type; 617 spcl.c_count = howmany(mapsize * sizeof(char), TP_BSIZE); 618 writeheader(ino); 619 for (i = 0, cp = map; i < spcl.c_count; i++, cp += TP_BSIZE) 620 writerec(cp, 0); 621 } 622 623 /* 624 * Write a header record to the dump tape. 625 */ 626 void 627 writeheader(ino_t ino) 628 { 629 int32_t sum, cnt, *lp; 630 631 spcl.c_inumber = ino; 632 spcl.c_magic = FS_UFS2_MAGIC; 633 spcl.c_checksum = 0; 634 lp = (int32_t *)&spcl; 635 sum = 0; 636 cnt = sizeof(union u_spcl) / (4 * sizeof(int32_t)); 637 while (--cnt >= 0) { 638 sum += *lp++; 639 sum += *lp++; 640 sum += *lp++; 641 sum += *lp++; 642 } 643 spcl.c_checksum = CHECKSUM - sum; 644 writerec((char *)&spcl, 1); 645 } 646 647 union dinode * 648 getino(ino_t inum, int *modep) 649 { 650 static ino_t minino, maxino; 651 static caddr_t inoblock; 652 struct ufs1_dinode *dp1; 653 struct ufs2_dinode *dp2; 654 655 if (inoblock == NULL && (inoblock = malloc(sblock->fs_bsize)) == NULL) 656 quit("cannot allocate inode memory.\n"); 657 curino = inum; 658 if (inum >= minino && inum < maxino) 659 goto gotit; 660 bread(fsbtodb(sblock, ino_to_fsba(sblock, inum)), inoblock, 661 (int)sblock->fs_bsize); 662 minino = inum - (inum % INOPB(sblock)); 663 maxino = minino + INOPB(sblock); 664 gotit: 665 if (sblock->fs_magic == FS_UFS1_MAGIC) { 666 dp1 = &((struct ufs1_dinode *)inoblock)[inum - minino]; 667 *modep = (dp1->di_mode & IFMT); 668 return ((union dinode *)dp1); 669 } 670 dp2 = &((struct ufs2_dinode *)inoblock)[inum - minino]; 671 *modep = (dp2->di_mode & IFMT); 672 return ((union dinode *)dp2); 673 } 674 675 /* 676 * Read a chunk of data from the disk. 677 * Try to recover from hard errors by reading in sector sized pieces. 678 * Error recovery is attempted at most BREADEMAX times before seeking 679 * consent from the operator to continue. 680 */ 681 int breaderrors = 0; 682 #define BREADEMAX 32 683 684 void 685 bread(ufs2_daddr_t blkno, char *buf, int size) 686 { 687 int cnt, i; 688 689 loop: 690 if ((cnt = pread(diskfd, buf, size, ((off_t)blkno << dev_bshift))) == 691 size) 692 return; 693 if (blkno + (size / dev_bsize) > fsbtodb(sblock, sblock->fs_size)) { 694 /* 695 * Trying to read the final fragment. 696 * 697 * NB - dump only works in TP_BSIZE blocks, hence 698 * rounds `dev_bsize' fragments up to TP_BSIZE pieces. 699 * It should be smarter about not actually trying to 700 * read more than it can get, but for the time being 701 * we punt and scale back the read only when it gets 702 * us into trouble. (mkm 9/25/83) 703 */ 704 size -= dev_bsize; 705 goto loop; 706 } 707 if (cnt == -1) 708 msg("read error from %s: %s: [block %d]: count=%d\n", 709 disk, strerror(errno), blkno, size); 710 else 711 msg("short read error from %s: [block %d]: count=%d, got=%d\n", 712 disk, blkno, size, cnt); 713 if (++breaderrors > BREADEMAX) { 714 msg("More than %d block read errors from %d\n", 715 BREADEMAX, disk); 716 broadcast("DUMP IS AILING!\n"); 717 msg("This is an unrecoverable error.\n"); 718 if (!query("Do you want to attempt to continue?")){ 719 dumpabort(0); 720 /*NOTREACHED*/ 721 } else 722 breaderrors = 0; 723 } 724 /* 725 * Zero buffer, then try to read each sector of buffer separately. 726 */ 727 memset(buf, 0, size); 728 for (i = 0; i < size; i += dev_bsize, buf += dev_bsize, blkno++) { 729 if ((cnt = pread(diskfd, buf, (int)dev_bsize, 730 ((off_t)blkno << dev_bshift))) == dev_bsize) 731 continue; 732 if (cnt == -1) { 733 msg("read error from %s: %s: [sector %d]: count=%d\n", 734 disk, strerror(errno), blkno, dev_bsize); 735 continue; 736 } 737 msg("short read error from %s: [sector %d]: count=%d, got=%d\n", 738 disk, blkno, dev_bsize, cnt); 739 } 740 } 741