122050Sdist /*
261492Sbostic * Copyright (c) 1980, 1986, 1993
361492Sbostic * The Regents of the University of California. All rights reserved.
439976Smckusick *
542702Sbostic * %sccs.include.redist.c%
622050Sdist */
722050Sdist
816264Smckusick #ifndef lint
9*68992Sbostic static char sccsid[] = "@(#)pass2.c 8.9 (Berkeley) 04/28/95";
1039976Smckusick #endif /* not lint */
1116264Smckusick
1216264Smckusick #include <sys/param.h>
1353703Smckusick #include <sys/time.h>
1468908Smckusick
1551532Sbostic #include <ufs/ufs/dinode.h>
1651532Sbostic #include <ufs/ufs/dir.h>
1751532Sbostic #include <ufs/ffs/fs.h>
1868908Smckusick
1968908Smckusick #include <err.h>
2042041Sbostic #include <string.h>
2168908Smckusick
2216264Smckusick #include "fsck.h"
2316264Smckusick
2440026Smckusick #define MINDIRSIZE (sizeof (struct dirtemplate))
2516264Smckusick
2668908Smckusick static int blksort __P((const void *, const void *));
2768908Smckusick static int pass2check __P((struct inodesc *));
2840026Smckusick
2968908Smckusick void
pass2()3016264Smckusick pass2()
3116264Smckusick {
3239973Smckusick register struct dinode *dp;
3340026Smckusick register struct inoinfo **inpp, *inp;
3440026Smckusick struct inoinfo **inpend;
3540026Smckusick struct inodesc curino;
3640026Smckusick struct dinode dino;
3740026Smckusick char pathbuf[MAXPATHLEN + 1];
3816264Smckusick
3916264Smckusick switch (statemap[ROOTINO]) {
4016264Smckusick
4116264Smckusick case USTATE:
4217966Smckusick pfatal("ROOT INODE UNALLOCATED");
4317966Smckusick if (reply("ALLOCATE") == 0)
4468908Smckusick exit(EEXIT);
4536827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO)
4668908Smckusick errx(EEXIT, "CANNOT ALLOCATE ROOT INODE");
4717966Smckusick break;
4816264Smckusick
4917966Smckusick case DCLEAR:
5017966Smckusick pfatal("DUPS/BAD IN ROOT INODE");
5117966Smckusick if (reply("REALLOCATE")) {
5217966Smckusick freeino(ROOTINO);
5336827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO)
5468908Smckusick errx(EEXIT, "CANNOT ALLOCATE ROOT INODE");
5517966Smckusick break;
5617966Smckusick }
5717966Smckusick if (reply("CONTINUE") == 0)
5868908Smckusick exit(EEXIT);
5917966Smckusick break;
6017966Smckusick
6116264Smckusick case FSTATE:
6217937Smckusick case FCLEAR:
6316264Smckusick pfatal("ROOT INODE NOT DIRECTORY");
6417966Smckusick if (reply("REALLOCATE")) {
6517966Smckusick freeino(ROOTINO);
6636827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO)
6768908Smckusick errx(EEXIT, "CANNOT ALLOCATE ROOT INODE");
6817966Smckusick break;
6917966Smckusick }
7017943Smckusick if (reply("FIX") == 0)
7168908Smckusick exit(EEXIT);
7217943Smckusick dp = ginode(ROOTINO);
7368139Smckusick dp->di_mode &= ~IFMT;
7468139Smckusick dp->di_mode |= IFDIR;
7516264Smckusick inodirty();
7640026Smckusick break;
7716264Smckusick
7816264Smckusick case DSTATE:
7916264Smckusick break;
8016264Smckusick
8117937Smckusick default:
8268908Smckusick errx(EEXIT, "BAD STATE %d FOR ROOT INODE", statemap[ROOTINO]);
8316264Smckusick }
8440026Smckusick statemap[ROOTINO] = DFOUND;
8567772Smckusick if (newinofmt) {
8667772Smckusick statemap[WINO] = FSTATE;
8768139Smckusick typemap[WINO] = DT_WHT;
8867772Smckusick }
8940026Smckusick /*
9040026Smckusick * Sort the directory list into disk block order.
9140026Smckusick */
9244934Smckusick qsort((char *)inpsort, (size_t)inplast, sizeof *inpsort, blksort);
9340026Smckusick /*
9440026Smckusick * Check the integrity of each directory.
9540026Smckusick */
96*68992Sbostic memset(&curino, 0, sizeof(struct inodesc));
9740026Smckusick curino.id_type = DATA;
9840026Smckusick curino.id_func = pass2check;
9940026Smckusick dp = &dino;
10040026Smckusick inpend = &inpsort[inplast];
10140026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) {
10240026Smckusick inp = *inpp;
10341136Smckusick if (inp->i_isize == 0)
10441136Smckusick continue;
10540026Smckusick if (inp->i_isize < MINDIRSIZE) {
10640026Smckusick direrror(inp->i_number, "DIRECTORY TOO SHORT");
10745002Smckusick inp->i_isize = roundup(MINDIRSIZE, DIRBLKSIZ);
10840026Smckusick if (reply("FIX") == 1) {
10940026Smckusick dp = ginode(inp->i_number);
11045002Smckusick dp->di_size = inp->i_isize;
11140026Smckusick inodirty();
11240026Smckusick dp = &dino;
11340026Smckusick }
11445002Smckusick } else if ((inp->i_isize & (DIRBLKSIZ - 1)) != 0) {
11540026Smckusick getpathname(pathbuf, inp->i_number, inp->i_number);
11640026Smckusick pwarn("DIRECTORY %s: LENGTH %d NOT MULTIPLE OF %d",
11740026Smckusick pathbuf, inp->i_isize, DIRBLKSIZ);
11840026Smckusick if (preen)
11940026Smckusick printf(" (ADJUSTED)\n");
12040026Smckusick inp->i_isize = roundup(inp->i_isize, DIRBLKSIZ);
12140026Smckusick if (preen || reply("ADJUST") == 1) {
12240026Smckusick dp = ginode(inp->i_number);
12340026Smckusick dp->di_size = roundup(inp->i_isize, DIRBLKSIZ);
12440026Smckusick inodirty();
12540026Smckusick dp = &dino;
12640026Smckusick }
12740026Smckusick }
128*68992Sbostic memset(&dino, 0, sizeof(struct dinode));
12968139Smckusick dino.di_mode = IFDIR;
13040026Smckusick dp->di_size = inp->i_isize;
131*68992Sbostic memmove(&dp->di_db[0], &inp->i_blks[0], (size_t)inp->i_numblks);
13240026Smckusick curino.id_number = inp->i_number;
13340026Smckusick curino.id_parent = inp->i_parent;
13440026Smckusick (void)ckinode(dp, &curino);
13540026Smckusick }
13640026Smckusick /*
13740026Smckusick * Now that the parents of all directories have been found,
13840026Smckusick * make another pass to verify the value of `..'
13940026Smckusick */
14040026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) {
14140026Smckusick inp = *inpp;
14241136Smckusick if (inp->i_parent == 0 || inp->i_isize == 0)
14340026Smckusick continue;
14440026Smckusick if (statemap[inp->i_parent] == DFOUND &&
14540026Smckusick statemap[inp->i_number] == DSTATE)
14640026Smckusick statemap[inp->i_number] = DFOUND;
14740026Smckusick if (inp->i_dotdot == inp->i_parent ||
14840026Smckusick inp->i_dotdot == (ino_t)-1)
14940026Smckusick continue;
15040026Smckusick if (inp->i_dotdot == 0) {
15140026Smckusick inp->i_dotdot = inp->i_parent;
15240026Smckusick fileerror(inp->i_parent, inp->i_number, "MISSING '..'");
15340026Smckusick if (reply("FIX") == 0)
15440026Smckusick continue;
15544934Smckusick (void)makeentry(inp->i_number, inp->i_parent, "..");
15640026Smckusick lncntp[inp->i_parent]--;
15740026Smckusick continue;
15840026Smckusick }
15940026Smckusick fileerror(inp->i_parent, inp->i_number,
16054603Smckusick "BAD INODE NUMBER FOR '..'");
16140026Smckusick if (reply("FIX") == 0)
16240026Smckusick continue;
16340026Smckusick lncntp[inp->i_dotdot]++;
16440026Smckusick lncntp[inp->i_parent]--;
16540026Smckusick inp->i_dotdot = inp->i_parent;
16640026Smckusick (void)changeino(inp->i_number, "..", inp->i_parent);
16740026Smckusick }
16840026Smckusick /*
16940026Smckusick * Mark all the directories that can be found from the root.
17040026Smckusick */
17140026Smckusick propagate();
17216264Smckusick }
17316264Smckusick
17468908Smckusick static int
pass2check(idesc)17516264Smckusick pass2check(idesc)
17616264Smckusick struct inodesc *idesc;
17716264Smckusick {
17839973Smckusick register struct direct *dirp = idesc->id_dirp;
17940026Smckusick register struct inoinfo *inp;
18016264Smckusick int n, entrysize, ret = 0;
18139973Smckusick struct dinode *dp;
18241136Smckusick char *errmsg;
18339973Smckusick struct direct proto;
18440026Smckusick char namebuf[MAXPATHLEN + 1];
18540026Smckusick char pathbuf[MAXPATHLEN + 1];
18616264Smckusick
18754603Smckusick /*
18854603Smckusick * If converting, set directory entry type.
18954603Smckusick */
19054603Smckusick if (doinglevel2 && dirp->d_ino > 0 && dirp->d_ino < maxino) {
19154603Smckusick dirp->d_type = typemap[dirp->d_ino];
19254603Smckusick ret |= ALTERED;
19354603Smckusick }
19416264Smckusick /*
19516264Smckusick * check for "."
19616264Smckusick */
19716264Smckusick if (idesc->id_entryno != 0)
19816264Smckusick goto chk1;
19916264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") == 0) {
20016264Smckusick if (dirp->d_ino != idesc->id_number) {
20139973Smckusick direrror(idesc->id_number, "BAD INODE NUMBER FOR '.'");
20216264Smckusick dirp->d_ino = idesc->id_number;
20316264Smckusick if (reply("FIX") == 1)
20416264Smckusick ret |= ALTERED;
20516264Smckusick }
20654603Smckusick if (newinofmt && dirp->d_type != DT_DIR) {
20754603Smckusick direrror(idesc->id_number, "BAD TYPE VALUE FOR '.'");
20854603Smckusick dirp->d_type = DT_DIR;
20954603Smckusick if (reply("FIX") == 1)
21054603Smckusick ret |= ALTERED;
21154603Smckusick }
21216264Smckusick goto chk1;
21316264Smckusick }
21439973Smckusick direrror(idesc->id_number, "MISSING '.'");
21516264Smckusick proto.d_ino = idesc->id_number;
21654603Smckusick if (newinofmt)
21754603Smckusick proto.d_type = DT_DIR;
21855074Smckusick else
21955074Smckusick proto.d_type = 0;
22016264Smckusick proto.d_namlen = 1;
22116264Smckusick (void)strcpy(proto.d_name, ".");
22267864Smckusick # if BYTE_ORDER == LITTLE_ENDIAN
22367864Smckusick if (!newinofmt) {
22467864Smckusick u_char tmp;
22567864Smckusick
22667864Smckusick tmp = proto.d_type;
22767864Smckusick proto.d_type = proto.d_namlen;
22867864Smckusick proto.d_namlen = tmp;
22967864Smckusick }
23067864Smckusick # endif
23154603Smckusick entrysize = DIRSIZ(0, &proto);
23216264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") != 0) {
23316264Smckusick pfatal("CANNOT FIX, FIRST ENTRY IN DIRECTORY CONTAINS %s\n",
23416264Smckusick dirp->d_name);
23516264Smckusick } else if (dirp->d_reclen < entrysize) {
23616264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '.'\n");
23716264Smckusick } else if (dirp->d_reclen < 2 * entrysize) {
23816264Smckusick proto.d_reclen = dirp->d_reclen;
239*68992Sbostic memmove(dirp, &proto, (size_t)entrysize);
24016264Smckusick if (reply("FIX") == 1)
24116264Smckusick ret |= ALTERED;
24216264Smckusick } else {
24316264Smckusick n = dirp->d_reclen - entrysize;
24416264Smckusick proto.d_reclen = entrysize;
245*68992Sbostic memmove(dirp, &proto, (size_t)entrysize);
24616264Smckusick idesc->id_entryno++;
24716264Smckusick lncntp[dirp->d_ino]--;
24839973Smckusick dirp = (struct direct *)((char *)(dirp) + entrysize);
249*68992Sbostic memset(dirp, 0, (size_t)n);
25016264Smckusick dirp->d_reclen = n;
25116264Smckusick if (reply("FIX") == 1)
25216264Smckusick ret |= ALTERED;
25316264Smckusick }
25416264Smckusick chk1:
25516264Smckusick if (idesc->id_entryno > 1)
25616264Smckusick goto chk2;
25740026Smckusick inp = getinoinfo(idesc->id_number);
25840026Smckusick proto.d_ino = inp->i_parent;
25954603Smckusick if (newinofmt)
26054603Smckusick proto.d_type = DT_DIR;
26155074Smckusick else
26255074Smckusick proto.d_type = 0;
26316264Smckusick proto.d_namlen = 2;
26416264Smckusick (void)strcpy(proto.d_name, "..");
26567864Smckusick # if BYTE_ORDER == LITTLE_ENDIAN
26667864Smckusick if (!newinofmt) {
26767864Smckusick u_char tmp;
26867864Smckusick
26967864Smckusick tmp = proto.d_type;
27067864Smckusick proto.d_type = proto.d_namlen;
27167864Smckusick proto.d_namlen = tmp;
27267864Smckusick }
27367864Smckusick # endif
27454603Smckusick entrysize = DIRSIZ(0, &proto);
27516264Smckusick if (idesc->id_entryno == 0) {
27654603Smckusick n = DIRSIZ(0, dirp);
27716264Smckusick if (dirp->d_reclen < n + entrysize)
27816264Smckusick goto chk2;
27916264Smckusick proto.d_reclen = dirp->d_reclen - n;
28016264Smckusick dirp->d_reclen = n;
28116264Smckusick idesc->id_entryno++;
28216264Smckusick lncntp[dirp->d_ino]--;
28339973Smckusick dirp = (struct direct *)((char *)(dirp) + n);
284*68992Sbostic memset(dirp, 0, (size_t)proto.d_reclen);
28540026Smckusick dirp->d_reclen = proto.d_reclen;
28616264Smckusick }
28716264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") == 0) {
28840026Smckusick inp->i_dotdot = dirp->d_ino;
28954603Smckusick if (newinofmt && dirp->d_type != DT_DIR) {
29054603Smckusick direrror(idesc->id_number, "BAD TYPE VALUE FOR '..'");
29154603Smckusick dirp->d_type = DT_DIR;
29254603Smckusick if (reply("FIX") == 1)
29354603Smckusick ret |= ALTERED;
29454603Smckusick }
29516264Smckusick goto chk2;
29616264Smckusick }
29716264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") != 0) {
29840026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'");
29916264Smckusick pfatal("CANNOT FIX, SECOND ENTRY IN DIRECTORY CONTAINS %s\n",
30016264Smckusick dirp->d_name);
30140026Smckusick inp->i_dotdot = (ino_t)-1;
30216264Smckusick } else if (dirp->d_reclen < entrysize) {
30340026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'");
30416264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '..'\n");
30540026Smckusick inp->i_dotdot = (ino_t)-1;
30640026Smckusick } else if (inp->i_parent != 0) {
30740026Smckusick /*
30840026Smckusick * We know the parent, so fix now.
30940026Smckusick */
31040026Smckusick inp->i_dotdot = inp->i_parent;
31140026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'");
31216264Smckusick proto.d_reclen = dirp->d_reclen;
313*68992Sbostic memmove(dirp, &proto, (size_t)entrysize);
31416264Smckusick if (reply("FIX") == 1)
31516264Smckusick ret |= ALTERED;
31616264Smckusick }
31740026Smckusick idesc->id_entryno++;
31840026Smckusick if (dirp->d_ino != 0)
31940026Smckusick lncntp[dirp->d_ino]--;
32040026Smckusick return (ret|KEEPON);
32116264Smckusick chk2:
32216264Smckusick if (dirp->d_ino == 0)
32316264Smckusick return (ret|KEEPON);
32416264Smckusick if (dirp->d_namlen <= 2 &&
32516264Smckusick dirp->d_name[0] == '.' &&
32616264Smckusick idesc->id_entryno >= 2) {
32716264Smckusick if (dirp->d_namlen == 1) {
32839973Smckusick direrror(idesc->id_number, "EXTRA '.' ENTRY");
32916264Smckusick dirp->d_ino = 0;
33016264Smckusick if (reply("FIX") == 1)
33116264Smckusick ret |= ALTERED;
33216264Smckusick return (KEEPON | ret);
33316264Smckusick }
33416264Smckusick if (dirp->d_name[1] == '.') {
33539973Smckusick direrror(idesc->id_number, "EXTRA '..' ENTRY");
33616264Smckusick dirp->d_ino = 0;
33716264Smckusick if (reply("FIX") == 1)
33816264Smckusick ret |= ALTERED;
33916264Smckusick return (KEEPON | ret);
34016264Smckusick }
34116264Smckusick }
34216264Smckusick idesc->id_entryno++;
34316264Smckusick n = 0;
34445847Smckusick if (dirp->d_ino > maxino) {
34540026Smckusick fileerror(idesc->id_number, dirp->d_ino, "I OUT OF RANGE");
34616264Smckusick n = reply("REMOVE");
34767575Spendry } else if (newinofmt &&
34867670Smckusick ((dirp->d_ino == WINO && dirp->d_type != DT_WHT) ||
34967670Smckusick (dirp->d_ino != WINO && dirp->d_type == DT_WHT))) {
35067575Spendry fileerror(idesc->id_number, dirp->d_ino, "BAD WHITEOUT ENTRY");
35167575Spendry dirp->d_ino = WINO;
35267575Spendry dirp->d_type = DT_WHT;
35367575Spendry if (reply("FIX") == 1)
35467575Spendry ret |= ALTERED;
35516264Smckusick } else {
35616264Smckusick again:
35716264Smckusick switch (statemap[dirp->d_ino]) {
35816264Smckusick case USTATE:
35940569Smckusick if (idesc->id_entryno <= 2)
36040569Smckusick break;
36140026Smckusick fileerror(idesc->id_number, dirp->d_ino, "UNALLOCATED");
36216264Smckusick n = reply("REMOVE");
36316264Smckusick break;
36416264Smckusick
36517937Smckusick case DCLEAR:
36617937Smckusick case FCLEAR:
36740569Smckusick if (idesc->id_entryno <= 2)
36840569Smckusick break;
36957717Smckusick if (statemap[dirp->d_ino] == FCLEAR)
37057717Smckusick errmsg = "DUP/BAD";
37157717Smckusick else if (!preen)
37241136Smckusick errmsg = "ZERO LENGTH DIRECTORY";
37357717Smckusick else {
37457717Smckusick n = 1;
37557717Smckusick break;
37657717Smckusick }
37741136Smckusick fileerror(idesc->id_number, dirp->d_ino, errmsg);
37816264Smckusick if ((n = reply("REMOVE")) == 1)
37916264Smckusick break;
38017943Smckusick dp = ginode(dirp->d_ino);
38139973Smckusick statemap[dirp->d_ino] =
38268139Smckusick (dp->di_mode & IFMT) == IFDIR ? DSTATE : FSTATE;
38330599Smckusick lncntp[dirp->d_ino] = dp->di_nlink;
38416264Smckusick goto again;
38516264Smckusick
38640026Smckusick case DSTATE:
38740026Smckusick if (statemap[idesc->id_number] == DFOUND)
38840026Smckusick statemap[dirp->d_ino] = DFOUND;
38940026Smckusick /* fall through */
39040026Smckusick
39117937Smckusick case DFOUND:
39240026Smckusick inp = getinoinfo(dirp->d_ino);
39340026Smckusick if (inp->i_parent != 0 && idesc->id_entryno > 2) {
39440026Smckusick getpathname(pathbuf, idesc->id_number,
39540026Smckusick idesc->id_number);
39617991Smckusick getpathname(namebuf, dirp->d_ino, dirp->d_ino);
39740026Smckusick pwarn("%s %s %s\n", pathbuf,
39817991Smckusick "IS AN EXTRANEOUS HARD LINK TO DIRECTORY",
39917991Smckusick namebuf);
40017991Smckusick if (preen)
40117991Smckusick printf(" (IGNORED)\n");
40217991Smckusick else if ((n = reply("REMOVE")) == 1)
40317991Smckusick break;
40417991Smckusick }
40540026Smckusick if (idesc->id_entryno > 2)
40640026Smckusick inp->i_parent = idesc->id_number;
40717937Smckusick /* fall through */
40817937Smckusick
40916264Smckusick case FSTATE:
41054603Smckusick if (newinofmt && dirp->d_type != typemap[dirp->d_ino]) {
41154603Smckusick fileerror(idesc->id_number, dirp->d_ino,
41254603Smckusick "BAD TYPE VALUE");
41354603Smckusick dirp->d_type = typemap[dirp->d_ino];
41454603Smckusick if (reply("FIX") == 1)
41554603Smckusick ret |= ALTERED;
41654603Smckusick }
41716264Smckusick lncntp[dirp->d_ino]--;
41816264Smckusick break;
41916264Smckusick
42026480Smckusick default:
42168908Smckusick errx(EEXIT, "BAD STATE %d FOR INODE I=%d",
42226480Smckusick statemap[dirp->d_ino], dirp->d_ino);
42316264Smckusick }
42416264Smckusick }
42516264Smckusick if (n == 0)
42616264Smckusick return (ret|KEEPON);
42716264Smckusick dirp->d_ino = 0;
42816264Smckusick return (ret|KEEPON|ALTERED);
42916264Smckusick }
43040026Smckusick
43140026Smckusick /*
43240026Smckusick * Routine to sort disk blocks.
43340026Smckusick */
43468908Smckusick static int
blksort(arg1,arg2)43568908Smckusick blksort(arg1, arg2)
43668908Smckusick const void *arg1, *arg2;
43740026Smckusick {
43840026Smckusick
43968908Smckusick return ((*(struct inoinfo **)arg1)->i_blks[0] -
44068908Smckusick (*(struct inoinfo **)arg2)->i_blks[0]);
44140026Smckusick }
442