8354 sync regcomp(3C) with upstream (fix make catalog)
[unleashed/tickless.git] / usr / src / cmd / backup / dump / dumptraverse.c
blob4e28e563470bb27637eeabd3af82eb29d376b4aa
1 /*
2 * Copyright 2006 Sun Microsystems, Inc. All rights reserved.
3 * Use is subject to license terms.
4 */
6 /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
7 /* All Rights Reserved */
9 /*
10 * Copyright (c) 1980 Regents of the University of California.
11 * All rights reserved. The Berkeley software License Agreement
12 * specifies the terms and conditions for redistribution.
15 #pragma ident "%Z%%M% %I% %E% SMI"
17 #include "dump.h"
18 #include <sys/file.h>
19 #include <sys/mman.h>
21 #ifdef __STDC__
22 static void lf_dmpindir(daddr32_t, int, u_offset_t *);
23 static void indir(daddr32_t, int, u_offset_t *);
24 static void lf_blksout(daddr32_t *, u_offset_t);
25 static void lf_dumpinode(struct dinode *);
26 static void dsrch(daddr32_t, ulong_t, u_offset_t);
27 void lf_dump(struct dinode *);
28 #else
29 static void lf_dmpindir();
30 static void indir();
31 static void lf_blksout();
32 static void dsrch();
33 void lf_dump();
34 #endif
36 static char msgbuf[256];
38 void
39 pass(fn, map)
40 void (*fn)(struct dinode *);
41 uchar_t *map;
43 int bits;
44 ino_t maxino;
46 maxino = (unsigned)(sblock->fs_ipg * sblock->fs_ncg - 1);
48 * Handle pass restarts. We don't check for UFSROOTINO just in
49 * case we need to restart on the root inode.
51 if (ino != 0) {
52 bits = ~0;
53 if (map != NULL) {
54 /* LINTED: lint seems to think map is signed */
55 map += (ino / NBBY);
56 bits = *map++;
58 bits >>= (ino % NBBY);
59 resetino(ino);
60 goto restart;
62 while (ino < maxino) {
63 if ((ino % NBBY) == 0) {
64 bits = ~0;
65 if (map != NULL)
66 bits = *map++;
68 restart:
69 ino++;
71 * Ignore any inode less than UFSROOTINO and inodes that
72 * we have already done on a previous pass.
74 if ((ino >= UFSROOTINO) && (bits & 1)) {
76 * The following test is merely an optimization
77 * for common case where "add" will just return.
79 if (!(fn == add && BIT(ino, nodmap)))
80 (*fn)(getino(ino));
82 bits >>= 1;
86 void
87 mark(ip)
88 struct dinode *ip;
90 int f;
92 f = ip->di_mode & IFMT;
93 if (f == 0 || ip->di_nlink <= 0) {
94 /* LINTED: 32-bit to 8-bit assignment ok */
95 BIC(ino, clrmap);
96 return;
98 /* LINTED: 32-bit to 8-bit assignment ok */
99 BIS(ino, clrmap);
100 if (f == IFDIR || f == IFATTRDIR) {
101 /* LINTED: 32-bit to 8-bit assignment ok */
102 BIS(ino, dirmap);
104 if (ip->di_ctime >= spcl.c_ddate) {
105 if (f == IFSHAD)
106 return;
107 /* LINTED: 32-bit to 8-bit assignment ok */
108 BIS(ino, nodmap);
109 /* attribute changes impact the root */
110 if (f == IFATTRDIR)
111 BIS(UFSROOTINO, nodmap);
112 if (f != IFREG && f != IFDIR && f != IFATTRDIR && f != IFLNK) {
113 o_esize += 1;
114 return;
116 est(ip);
120 void
121 active_mark(ip)
122 struct dinode *ip;
124 int f;
126 f = ip->di_mode & IFMT;
127 if (f == 0 || ip->di_nlink <= 0) {
128 /* LINTED: 32-bit to 8-bit assignment ok */
129 BIC(ino, clrmap);
130 return;
132 /* LINTED: 32-bit to 8-bit assignment ok */
133 BIS(ino, clrmap);
134 if (f == IFDIR || f == IFATTRDIR) {
135 /* LINTED: 32-bit to 8-bit assignment ok */
136 BIS(ino, dirmap);
138 if (BIT(ino, activemap)) {
139 /* LINTED: 32-bit to 8-bit assignment ok */
140 BIS(ino, nodmap);
141 /* attribute changes impact the root */
142 if (f == IFATTRDIR)
143 BIS(UFSROOTINO, nodmap);
144 if (f != IFREG && f != IFDIR && f != IFATTRDIR && f != IFLNK) {
145 o_esize += 1;
146 return;
148 est(ip);
152 static struct shcount {
153 struct shcount *higher, *lower;
154 ino_t ino;
155 unsigned long count;
156 } shcounts = {
157 NULL, NULL,
161 static struct shcount *shc = NULL;
163 void
164 markshad(ip)
165 struct dinode *ip;
167 ino_t shadow;
169 if (ip->di_shadow == 0)
170 return;
171 if (shc == NULL)
172 shc = &shcounts;
174 shadow = (ino_t)(unsigned)(ip->di_shadow);
175 while ((shadow > shc->ino) && (shc->higher))
176 shc = shc->higher;
177 while ((shadow < shc->ino) && (shc->lower))
178 shc = shc->lower;
179 if (shadow != shc->ino) {
180 struct shcount *new;
182 new = (struct shcount *)xcalloc(1, sizeof (*new));
183 new->higher = shc->higher;
184 if (shc->higher != NULL)
185 shc->higher->lower = new;
186 shc->higher = new;
187 new->lower = shc;
188 shc = new;
189 shc->ino = shadow;
192 /* LINTED: 32-bit to 8-bit assignment ok */
193 BIS(shadow, shamap);
194 shc->count++;
197 void
198 estshad(ip)
199 struct dinode *ip;
201 u_offset_t esizeprime;
202 u_offset_t tmpesize;
204 if (ip->di_size <= sizeof (union u_shadow))
205 return;
207 while ((ino > shc->ino) && (shc->higher))
208 shc = shc->higher;
209 while ((ino < shc->ino) && (shc->lower))
210 shc = shc->lower;
211 if (ino != shc->ino)
212 return; /* xxx panic? complain? */
214 tmpesize = (o_esize + f_esize);
215 esizeprime = tmpesize;
216 est(ip);
217 esizeprime = tmpesize - esizeprime;
218 esizeprime *= shc->count - 1;
219 f_esize += esizeprime;
222 void
223 freeshad()
225 if (shc == NULL)
226 return;
228 while (shc->higher)
229 shc = shc->higher;
230 while (shc->lower) {
231 shc = shc->lower;
232 if (shc->higher) /* else panic? */
233 (void) free(shc->higher);
236 * This should be unnecessary, but do it just to be safe.
237 * Note that shc might be malloc'd or static, so can't free().
239 bzero(shc, sizeof (*shc));
242 void
243 add(ip)
244 struct dinode *ip;
246 int i;
247 u_offset_t filesize;
249 if (BIT(ino, nodmap))
250 return;
251 if ((ip->di_mode & IFMT) != IFDIR &&
252 (ip->di_mode & IFMT) != IFATTRDIR) {
253 (void) snprintf(msgbuf, sizeof (msgbuf), gettext(
254 "Warning - directory at inode `%lu' vanished!\n"), ino);
255 msg(msgbuf);
256 /* LINTED: 32-bit to 8-bit assignment ok */
257 BIC(ino, dirmap);
258 return;
260 nsubdir = 0;
261 dadded = 0;
262 filesize = ip->di_size;
263 for (i = 0; i < NDADDR; i++) {
264 if (ip->di_db[i] != 0)
265 /* LINTED dblksize/blkoff does a safe cast here */
266 dsrch(ip->di_db[i], (ulong_t)dblksize(sblock, ip, i),
267 filesize);
268 filesize -= (unsigned)(sblock->fs_bsize);
270 for (i = 0; i < NIADDR; i++) {
271 if (ip->di_ib[i] != 0)
272 indir(ip->di_ib[i], i, &filesize);
274 if (dadded) {
275 nadded++;
276 if (!BIT(ino, nodmap)) {
277 /* LINTED: 32-bit to 8-bit assignment ok */
278 BIS(ino, nodmap);
279 if ((ip->di_mode & IFMT) == IFATTRDIR) {
280 /* attribute changes "auto-percolate" to root */
281 BIS(UFSROOTINO, nodmap);
283 est(ip);
286 if (nsubdir == 0) {
287 if (!BIT(ino, nodmap)) {
288 /* LINTED: 32-bit to 8-bit assignment ok */
289 BIC(ino, dirmap);
294 static void
295 indir(d, n, filesize)
296 daddr32_t d;
297 int n;
298 u_offset_t *filesize;
300 int i;
301 daddr32_t idblk[MAXNINDIR];
303 if ((unsigned)(sblock->fs_bsize) > sizeof (idblk)) {
304 msg(gettext(
305 "Inconsistency detected: filesystem block size larger than valid maximum.\n"));
306 dumpabort();
307 /*NOTREACHED*/
310 if ((unsigned)NINDIR(sblock) > MAXNINDIR) {
311 /*CSTYLED*/
312 msg(gettext(
313 "Inconsistency detected: inode has more indirect \
314 blocks than valid maximum.\n"));
315 dumpabort();
316 /*NOTREACHED*/
319 if (dadded || *filesize == 0)
320 return;
322 #ifdef lint
323 idblk[0] = '\0';
324 #endif /* lint */
326 /* xxx sanity check sblock contents before trusting them */
327 bread(fsbtodb(sblock, d), (uchar_t *)idblk, (size_t)sblock->fs_bsize);
328 if (n <= 0) {
329 for (i = 0; i < NINDIR(sblock); i++) {
330 d = idblk[i];
331 if (d != 0)
332 dsrch(d, (ulong_t)(uint32_t)sblock->fs_bsize,
333 *filesize);
334 *filesize -= (unsigned)(sblock->fs_bsize);
336 } else {
337 n--;
338 for (i = 0; i < NINDIR(sblock); i++) {
339 d = idblk[i];
340 if (d != 0)
341 indir(d, n, filesize);
346 void
347 dirdump(ip)
348 struct dinode *ip;
350 /* watchout for dir inodes deleted and maybe reallocated */
351 if (((ip->di_mode & IFMT) != IFDIR &&
352 (ip->di_mode & IFMT) != IFATTRDIR) || ip->di_nlink < 2) {
353 (void) snprintf(msgbuf, sizeof (msgbuf), gettext(
354 "Warning - directory at inode `%lu' vanished!\n"),
355 ino);
356 msg(msgbuf);
357 return;
359 lf_dump(ip);
362 static u_offset_t loffset; /* current offset in file (ufsdump) */
364 static void
365 lf_dumpmeta(ip)
366 struct dinode *ip;
368 if ((ip->di_shadow == 0) || shortmeta)
369 return;
371 lf_dumpinode(getino((ino_t)(unsigned)(ip->di_shadow)));
375 hasshortmeta(ip)
376 struct dinode **ip;
378 ino_t savino;
379 int rc;
381 if ((*ip)->di_shadow == 0)
382 return (0);
383 savino = ino;
384 *ip = getino((ino_t)(unsigned)((*ip)->di_shadow));
385 rc = ((*ip)->di_size <= sizeof (union u_shadow));
386 *ip = getino(ino = savino);
387 return (rc);
390 void
391 lf_dumpinode(ip)
392 struct dinode *ip;
394 int i;
395 u_offset_t size;
397 i = ip->di_mode & IFMT;
399 if (i == 0 || ip->di_nlink <= 0)
400 return;
402 spcl.c_dinode = *ip;
403 spcl.c_count = 0;
405 if ((i != IFDIR && i != IFATTRDIR && i != IFREG && i != IFLNK &&
406 i != IFSHAD) || ip->di_size == 0) {
407 toslave(dospcl, ino);
408 return;
411 size = NDADDR * (unsigned)(sblock->fs_bsize);
412 if (size > ip->di_size)
413 size = ip->di_size;
415 lf_blksout(&ip->di_db[0], size);
417 size = ip->di_size - size;
418 if (size > 0) {
419 for (i = 0; i < NIADDR; i++) {
420 lf_dmpindir(ip->di_ib[i], i, &size);
421 if (size == 0)
422 break;
427 void
428 lf_dump(ip)
429 struct dinode *ip;
432 if ((!BIT(ino, nodmap)) && (!BIT(ino, shamap)))
433 return;
435 shortmeta = hasshortmeta(&ip);
436 if (shortmeta) {
437 ip = getino((ino_t)(unsigned)(ip->di_shadow));
438 /* assume spcl.c_shadow is smaller than 1 block */
439 bread(fsbtodb(sblock, ip->di_db[0]),
440 (uchar_t *)spcl.c_shadow.c_shadow, sizeof (spcl.c_shadow));
441 spcl.c_flags |= DR_HASMETA;
442 } else {
443 spcl.c_flags &= ~DR_HASMETA;
445 ip = getino(ino);
447 loffset = 0;
449 if (newtape) {
450 spcl.c_type = TS_TAPE;
451 } else if (pos)
452 spcl.c_type = TS_ADDR;
453 else
454 spcl.c_type = TS_INODE;
456 newtape = 0;
457 lf_dumpinode(ip);
458 lf_dumpmeta(ip);
459 pos = 0;
462 static void
463 lf_dmpindir(blk, lvl, size)
464 daddr32_t blk;
465 int lvl;
466 u_offset_t *size;
468 int i;
469 u_offset_t cnt;
470 daddr32_t idblk[MAXNINDIR];
472 if ((unsigned)(sblock->fs_bsize) > sizeof (idblk)) {
473 msg(gettext(
474 "Inconsistency detected: filesystem block size larger than valid maximum.\n"));
475 dumpabort();
476 /*NOTREACHED*/
479 if ((unsigned)NINDIR(sblock) > MAXNINDIR) {
480 msg(gettext(
481 "Inconsistency detected: inode has more indirect \
482 blocks than valid maximum.\n"));
483 dumpabort();
484 /*NOTREACHED*/
487 if (blk != 0)
488 bread(fsbtodb(sblock, blk), (uchar_t *)idblk,
489 (size_t)sblock->fs_bsize);
490 else
491 bzero((char *)idblk, (size_t)sblock->fs_bsize);
492 if (lvl <= 0) {
493 cnt = (u_offset_t)(unsigned)NINDIR(sblock) *
494 (u_offset_t)(unsigned)(sblock->fs_bsize);
495 if (cnt > *size)
496 cnt = *size;
497 *size -= cnt;
498 lf_blksout(&idblk[0], cnt);
499 return;
501 lvl--;
502 for (i = 0; i < NINDIR(sblock); i++) {
503 lf_dmpindir(idblk[i], lvl, size);
504 if (*size == 0)
505 return;
509 static void
510 lf_blksout(blkp, bytes)
511 daddr32_t *blkp;
512 u_offset_t bytes;
514 u_offset_t i;
515 u_offset_t tbperfsb = (unsigned)(sblock->fs_bsize / tp_bsize);
517 u_offset_t j, k, count;
519 u_offset_t bytepos, diff;
520 u_offset_t bytecnt = 0;
521 off_t byteoff = 0; /* bytes to skip within first f/s block */
522 off_t fragoff = 0; /* frags to skip within first f/s block */
524 u_offset_t tpblkoff = 0; /* tape blocks to skip in first f/s block */
525 u_offset_t tpblkskip = 0; /* total tape blocks to skip */
526 u_offset_t skip; /* tape blocks to skip this pass */
528 if (pos) {
530 * We get here if a slave throws a signal to the
531 * master indicating a partially dumped file.
532 * Begin by figuring out what was undone.
534 bytepos = (offset_t)pos * tp_bsize;
536 if ((loffset + bytes) <= bytepos) {
537 /* This stuff was dumped already, forget it. */
538 loffset += (u_offset_t)tp_bsize *
539 /* LINTED: spurious complaint on sign-extending */
540 d_howmany(bytes, (u_offset_t)tp_bsize);
541 return;
544 if (loffset < bytepos) {
546 * Some of this was dumped, some wasn't.
547 * Figure out what was done and skip it.
549 diff = bytepos - loffset;
550 /* LINTED: spurious complaint on sign-extending */
551 tpblkskip = d_howmany(diff, (u_offset_t)tp_bsize);
552 /* LINTED room after EOT is only a few MB */
553 blkp += (int)(diff / sblock->fs_bsize);
555 bytecnt = diff % (unsigned)(sblock->fs_bsize);
556 /* LINTED: result fits, due to modulus */
557 byteoff = bytecnt % (off_t)(sblock->fs_fsize);
558 /* LINTED: spurious complaint on sign-extending */
559 tpblkoff = d_howmany(bytecnt,
560 (u_offset_t)(unsigned)tp_bsize);
561 /* LINTED: result fits, due to modulus */
562 fragoff = bytecnt / (off_t)(sblock->fs_fsize);
563 bytecnt = (unsigned)(sblock->fs_bsize) - bytecnt;
567 loffset += bytes;
569 while (bytes > 0) {
570 if (bytes < TP_NINDIR*tp_bsize)
571 /* LINTED: spurious complaint on sign-extending */
572 count = d_howmany(bytes, (u_offset_t)tp_bsize);
573 else
574 count = TP_NINDIR;
575 if (tpblkskip) {
576 if (tpblkskip < TP_NINDIR) {
577 bytes -= (tpblkskip * (u_offset_t)tp_bsize);
578 skip = tpblkskip;
579 tpblkskip = 0;
580 } else {
581 bytes -= (offset_t)TP_NINDIR*tp_bsize;
582 tpblkskip -= TP_NINDIR;
583 continue;
585 } else
586 skip = 0;
587 assert(tbperfsb >= tpblkoff);
588 assert((count - skip) <= TP_NINDIR);
589 for (j = 0, k = 0; j < count - skip; j++, k++) {
590 spcl.c_addr[j] = (blkp[k] != 0);
591 for (i = tbperfsb - tpblkoff; --i > 0; j++)
592 spcl.c_addr[j+1] = spcl.c_addr[j];
593 tpblkoff = 0;
595 /* LINTED (count - skip) will always fit into an int32_t */
596 spcl.c_count = count - skip;
597 toslave(dospcl, ino);
598 bytecnt = MIN(bytes, bytecnt ?
599 bytecnt : (unsigned)(sblock->fs_bsize));
600 j = 0;
601 while (j < count - skip) {
602 if (*blkp != 0) {
603 /* LINTED: fragoff fits into 32 bits */
604 dmpblk(*blkp+(int32_t)fragoff,
605 /* LINTED: bytecnt fits into 32 bits */
606 (size_t)bytecnt, byteoff);
608 blkp++;
609 bytes -= bytecnt;
610 /* LINTED: spurious complaint on sign-extending */
611 j += d_howmany(bytecnt, (u_offset_t)tp_bsize);
612 bytecnt = MIN(bytes, (unsigned)(sblock->fs_bsize));
613 byteoff = 0;
614 fragoff = 0;
616 spcl.c_type = TS_ADDR;
617 bytecnt = 0;
619 pos = 0;
622 void
623 bitmap(map, typ)
624 uchar_t *map;
625 int typ;
627 int i;
628 u_offset_t count;
629 uchar_t *cp;
631 if (!newtape)
632 spcl.c_type = typ;
633 else
634 newtape = 0;
635 for (i = 0; i < TP_NINDIR; i++)
636 spcl.c_addr[i] = 1;
637 /* LINTED: spurious complaint on sign-extending */
638 count = d_howmany(msiz * sizeof (map[0]), tp_bsize) - pos;
639 for (cp = &map[pos * tp_bsize]; count > 0;
640 count -= (u_offset_t)(unsigned)spcl.c_count) {
641 if (leftover) {
642 spcl.c_count = leftover;
643 leftover = 0;
644 } else {
645 /* LINTED value always less than INT32_MAX */
646 spcl.c_count = count > TP_NINDIR ? TP_NINDIR : count;
648 spclrec();
649 for (i = 0; i < spcl.c_count; i++, cp += tp_bsize)
650 taprec(cp, 0, tp_bsize);
651 spcl.c_type = TS_ADDR;
655 static void
656 dsrch(d, size, filesize)
657 daddr32_t d;
658 ulong_t size; /* block size */
659 u_offset_t filesize;
661 struct direct *dp;
662 struct dinode *ip;
663 ulong_t loc;
664 char dblk[MAXBSIZE];
666 if (dadded || filesize == 0)
667 return;
668 if (filesize > (u_offset_t)size)
669 filesize = (u_offset_t)size;
670 if (sizeof (dblk) < roundup(filesize, DEV_BSIZE)) {
671 msg(gettext(
672 "Inconsistency detected: filesystem block size larger than valid maximum.\n"));
673 dumpabort();
674 /*NOTREACHED*/
677 #ifdef lint
678 dblk[0] = '\0';
679 #endif /* lint */
681 /* LINTED ufs disk addresses always fit into 32 bits */
682 bread(fsbtodb(sblock, d), (uchar_t *)dblk,
683 /* LINTED from sizeof check above, roundup() <= max(size_t) */
684 (size_t)(roundup(filesize, DEV_BSIZE)));
685 loc = 0;
686 while ((u_offset_t)loc < filesize) {
687 /*LINTED [dblk is char[], loc (dp->d_reclen) % 4 == 0]*/
688 dp = (struct direct *)(dblk + loc);
689 if (dp->d_reclen == 0) {
690 (void) snprintf(msgbuf, sizeof (msgbuf), gettext(
691 "Warning - directory at inode `%lu' is corrupted\n"),
692 ino);
693 msg(msgbuf);
694 break;
696 loc += dp->d_reclen;
697 if (dp->d_ino == 0)
698 continue;
699 if (dp->d_name[0] == '.') {
700 if (dp->d_name[1] == '\0') {
701 if ((ino_t)(dp->d_ino) != ino) {
702 (void) snprintf(msgbuf, sizeof (msgbuf),
703 gettext(
704 "Warning - directory at inode `%lu' is corrupted:\n\
705 \t\".\" points to inode `%lu' - run fsck\n"),
706 ino, dp->d_ino);
707 msg(msgbuf);
709 continue;
711 if (dp->d_name[1] == '.' && dp->d_name[2] == '\0') {
712 if (!BIT(dp->d_ino, dirmap) &&
713 ((ip = getino(ino)) == NULL ||
714 (ip->di_mode & IFMT) != IFATTRDIR)) {
715 (void) snprintf(msgbuf, sizeof (msgbuf),
716 gettext(
717 "Warning - directory at inode `%lu' is corrupted:\n\
718 \t\"..\" points to non-directory inode `%lu' - run fsck\n"),
719 ino, dp->d_ino);
720 msg(msgbuf);
722 continue;
725 if (BIT(dp->d_ino, nodmap)) {
726 dadded++;
727 return;
729 if (BIT(dp->d_ino, dirmap))
730 nsubdir++;
734 #define CACHESIZE 32
736 struct dinode *
737 getino(ino)
738 ino_t ino;
740 static ino_t minino, maxino;
741 static struct dinode itab[MAXINOPB];
742 static struct dinode icache[CACHESIZE];
743 static ino_t icacheval[CACHESIZE], lasti = 0;
744 static int cacheoff = 0;
745 int i;
747 if (ino >= minino && ino < maxino) {
748 lasti = ino;
749 return (&itab[ino - minino]);
752 /* before we do major i/o, check for a secondary cache hit */
753 for (i = 0; i < CACHESIZE; i++)
754 if (icacheval[i] == ino)
755 return (icache + i);
757 /* we need to do major i/o. throw the last inode retrieved into */
758 /* the cache. note: this copies garbage the first time it is */
759 /* used, but no harm done. */
760 icacheval[cacheoff] = lasti;
761 bcopy(itab + (lasti - minino), icache + cacheoff, sizeof (itab[0]));
762 lasti = ino;
763 if (++cacheoff >= CACHESIZE)
764 cacheoff = 0;
766 #define INOPERDB (DEV_BSIZE / sizeof (struct dinode))
767 minino = ino &~ (INOPERDB - 1);
768 maxino = ((itog(sblock, ino) + 1) * (unsigned)(sblock->fs_ipg));
769 if (maxino > minino + MAXINOPB)
770 maxino = minino + MAXINOPB;
771 bread(
772 /* LINTED: can't make up for broken system macros here */
773 (fsbtodb(sblock, itod(sblock, ino)) + itoo(sblock, ino) / INOPERDB),
774 /* LINTED: (max - min) * size fits into a size_t */
775 (uchar_t *)itab, (size_t)((maxino - minino) * sizeof (*itab)));
776 return (&itab[ino - minino]);
779 #define BREADEMAX 32
781 #ifdef NO__LONGLONG__
782 #define DEV_LSEEK(fd, offset, whence) \
783 lseek((fd), (((off_t)(offset))*DEV_BSIZE), (whence))
784 #else
785 #define DEV_LSEEK(fd, offset, whence) \
786 llseek((fd), (((offset_t)((offset)))*DEV_BSIZE), (whence))
787 #endif
789 #define BREAD_FAIL(buf, size) { \
790 breaderrors += 1; \
791 bzero(buf, (size_t)size); \
796 void
797 bread(da, ba, cnt)
798 diskaddr_t da;
799 uchar_t *ba;
800 size_t cnt;
802 caddr_t maddr;
803 uchar_t *dest;
804 int saverr;
805 int n;
806 size_t len;
807 off64_t filoff;
808 off64_t mapoff;
809 off64_t displacement;
811 static size_t pagesize = 0;
812 static int breaderrors = 0;
814 /* mechanics for caching small bread requests. these are */
815 /* often small ACLs that are used over and over. */
816 static uchar_t bcache[DEV_BSIZE * CACHESIZE];
817 static diskaddr_t bcacheval[CACHESIZE];
818 static int cacheoff = 0;
819 int i;
821 if ((cnt >= DEV_BSIZE) && (mapfd != -1)) {
822 if (pagesize == 0)
823 pagesize = getpagesize();
825 * We depend on mmap(2)'s guarantee that mapping a
826 * partial page will cause the remainder of the page
827 * to be zero-filled.
829 filoff = ((off64_t)da) * DEV_BSIZE;
830 displacement = filoff & (pagesize - 1);
831 mapoff = filoff - displacement;
832 /* LINTED offset will fit into 32 bits */
833 len = (size_t)roundup(cnt + (filoff - mapoff), pagesize);
834 maddr = mmap64(NULL, len, PROT_READ, MAP_SHARED, mapfd, mapoff);
835 if (maddr != MAP_FAILED) {
836 (void) memcpy(ba, maddr + displacement, cnt);
837 (void) munmap(maddr, len);
838 return;
842 if (DEV_LSEEK(fi, da, L_SET) < 0) {
843 saverr = errno;
844 msg(gettext("bread: dev_seek error: %s\n"), strerror(saverr));
845 /* Don't know where we are, return the least-harmful data */
846 BREAD_FAIL(ba, cnt);
847 return;
850 if (read(fi, ba, (size_t)cnt) == (size_t)cnt)
851 return;
853 while (cnt != 0) {
855 if (da >= fsbtodb(sblock, sblock->fs_size)) {
856 msg(gettext(
857 "Warning - block %llu is beyond the end of `%s'\n"),
858 da, disk);
859 BREAD_FAIL(ba, cnt);
860 break;
863 if (DEV_LSEEK(fi, da, L_SET) < 0) {
864 msg(gettext("%s: %s error\n"), "bread", "DEV_LSEEK2");
865 BREAD_FAIL(ba, cnt);
866 break;
869 if (cnt < DEV_BSIZE) {
870 /* small read. check for cache hit. */
871 for (i = 0; i < CACHESIZE; i++)
872 if (bcacheval[i] == da) {
873 bcopy(bcache + (i * DEV_BSIZE),
874 ba, cnt);
875 return;
878 /* no cache hit; throw this one into the cache... */
879 len = cnt;
880 dest = bcache + (cacheoff * DEV_BSIZE);
881 bcacheval[cacheoff] = da;
882 if (++cacheoff >= CACHESIZE)
883 cacheoff = 0;
884 } else {
885 len = DEV_BSIZE;
886 dest = ba;
889 n = read(fi, dest, DEV_BSIZE);
890 if (n != DEV_BSIZE) {
891 n = MAX(n, 0);
892 bzero(dest+n, DEV_BSIZE-n);
893 breaderrors += 1;
894 msg(gettext(
895 "Warning - cannot read sector %llu of `%s'\n"),
896 da, disk);
898 if (dest != ba)
899 bcopy(dest, ba, len);
901 da++;
902 /* LINTED character pointers aren't signed */
903 ba += len;
904 cnt -= len;
907 if (breaderrors > BREADEMAX) {
908 msg(gettext(
909 "More than %d block read errors from dump device `%s'\n"),
910 BREADEMAX, disk);
911 dumpailing();
912 breaderrors = 0;