/* * dba dbm analysis/recovery */ #include #ifdef LiamSysV # include #else # include #endif #include "sdbm.h" static char *progname; int main(argc, argv) char **argv; { int n; char *p; char *name; int pagf; progname = argv[0]; if (p = argv[1]) { name = (char *) malloc((n = strlen(p)) + 5); strcpy(name, p); strcpy(name + n, ".pag"); if ((pagf = open(name, O_RDONLY)) < 0) oops("cannot open %s.", name); sdump(pagf); } else oops("usage: %s dbname", progname); return 0; } sdump(pagf) int pagf; { register b; register n = 0; register t = 0; register o = 0; register e; char pag[PBLKSIZ]; while ((b = read(pagf, pag, PBLKSIZ)) > 0) { printf("#%d: ", n); if (!chkpage(pag)) printf("bad\n"); else { printf("ok. "); if (!(e = pagestat(pag))) o++; else t += e; } n++; } if (b == 0) printf("%d pages (%d holes): %d entries\n", n, o, t); else oops("read failed: block %d", n); } pagestat(pag) char *pag; { register n; register free; register short *ino = (short *) pag; if (!(n = ino[0])) printf("no entries.\n"); else { free = ino[n] - (n + 1) * sizeof(short); printf("%3d entries %2d%% used free %d.\n", n / 2, ((PBLKSIZ - free) * 100) / PBLKSIZ, free); } return (n / 2); } dispage(pag) char *pag; { register i, n; register off; register short *ino = (short *) pag; off = PBLKSIZ; for (i = 1; i < ino[0]; i += 2) { printf("\t[%d]: ", ino[i]); for (n = ino[i]; n < off; n++) putchar(pag[n]); putchar(' '); off = ino[i]; printf("[%d]: ", ino[i + 1]); for (n = ino[i + 1]; n < off; n++) putchar(pag[n]); off = ino[i + 1]; putchar('\n'); } } chkpage(pag) char *pag; { register unsigned n; register off; register short *ino = (short *) pag; if ((n = ino[0]) > PBLKSIZ / sizeof(short)) return (0); if (!n) return (1); off = PBLKSIZ; for (ino++; n; ino += 2) { if (ino[0] > off || ino[1] > off || ino[1] > ino[0]) return (0); off = ino[1]; n -= 2; } return (1); } oops(s1, s2) register char *s1; register char *s2; { extern int errno, sys_nerr; extern char *sys_errlist[]; extern char *progname; if (progname) printf("%s: ", progname); printf(s1, s2); if (errno > 0 && errno < sys_nerr) printf(" (%s)", sys_errlist[errno]); printf("\n"); exit(1); }