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*67670Smckusick static char sccsid[] = "@(#)pass2.c 8.4 (Berkeley) 08/14/94"; 1039976Smckusick #endif /* not lint */ 1116264Smckusick 1216264Smckusick #include <sys/param.h> 1353703Smckusick #include <sys/time.h> 1451532Sbostic #include <ufs/ufs/dinode.h> 1551532Sbostic #include <ufs/ufs/dir.h> 1651532Sbostic #include <ufs/ffs/fs.h> 1744934Smckusick #include <stdlib.h> 1842041Sbostic #include <string.h> 1916264Smckusick #include "fsck.h" 2016264Smckusick 2140026Smckusick #define MINDIRSIZE (sizeof (struct dirtemplate)) 2216264Smckusick 2340026Smckusick int pass2check(), blksort(); 2440026Smckusick 2516264Smckusick pass2() 2616264Smckusick { 2739973Smckusick register struct dinode *dp; 2840026Smckusick register struct inoinfo **inpp, *inp; 2940026Smckusick struct inoinfo **inpend; 3040026Smckusick struct inodesc curino; 3140026Smckusick struct dinode dino; 3240026Smckusick char pathbuf[MAXPATHLEN + 1]; 3316264Smckusick 3416264Smckusick switch (statemap[ROOTINO]) { 3516264Smckusick 3616264Smckusick case USTATE: 3717966Smckusick pfatal("ROOT INODE UNALLOCATED"); 3817966Smckusick if (reply("ALLOCATE") == 0) 3917966Smckusick errexit(""); 4036827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 4117966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 4217966Smckusick break; 4316264Smckusick 4417966Smckusick case DCLEAR: 4517966Smckusick pfatal("DUPS/BAD IN ROOT INODE"); 4617966Smckusick if (reply("REALLOCATE")) { 4717966Smckusick freeino(ROOTINO); 4836827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 4917966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 5017966Smckusick break; 5117966Smckusick } 5217966Smckusick if (reply("CONTINUE") == 0) 5317966Smckusick errexit(""); 5417966Smckusick break; 5517966Smckusick 5616264Smckusick case FSTATE: 5717937Smckusick case FCLEAR: 5816264Smckusick pfatal("ROOT INODE NOT DIRECTORY"); 5917966Smckusick if (reply("REALLOCATE")) { 6017966Smckusick freeino(ROOTINO); 6136827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 6217966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 6317966Smckusick break; 6417966Smckusick } 6517943Smckusick if (reply("FIX") == 0) 6616264Smckusick errexit(""); 6717943Smckusick dp = ginode(ROOTINO); 6816264Smckusick dp->di_mode &= ~IFMT; 6916264Smckusick dp->di_mode |= IFDIR; 7016264Smckusick inodirty(); 7140026Smckusick break; 7216264Smckusick 7316264Smckusick case DSTATE: 7416264Smckusick break; 7516264Smckusick 7617937Smckusick default: 7717937Smckusick errexit("BAD STATE %d FOR ROOT INODE", statemap[ROOTINO]); 7816264Smckusick } 7940026Smckusick statemap[ROOTINO] = DFOUND; 8040026Smckusick /* 8140026Smckusick * Sort the directory list into disk block order. 8240026Smckusick */ 8344934Smckusick qsort((char *)inpsort, (size_t)inplast, sizeof *inpsort, blksort); 8440026Smckusick /* 8540026Smckusick * Check the integrity of each directory. 8640026Smckusick */ 8740026Smckusick bzero((char *)&curino, sizeof(struct inodesc)); 8840026Smckusick curino.id_type = DATA; 8940026Smckusick curino.id_func = pass2check; 9040026Smckusick dp = &dino; 9140026Smckusick inpend = &inpsort[inplast]; 9240026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) { 9340026Smckusick inp = *inpp; 9441136Smckusick if (inp->i_isize == 0) 9541136Smckusick continue; 9640026Smckusick if (inp->i_isize < MINDIRSIZE) { 9740026Smckusick direrror(inp->i_number, "DIRECTORY TOO SHORT"); 9845002Smckusick inp->i_isize = roundup(MINDIRSIZE, DIRBLKSIZ); 9940026Smckusick if (reply("FIX") == 1) { 10040026Smckusick dp = ginode(inp->i_number); 10145002Smckusick dp->di_size = inp->i_isize; 10240026Smckusick inodirty(); 10340026Smckusick dp = &dino; 10440026Smckusick } 10545002Smckusick } else if ((inp->i_isize & (DIRBLKSIZ - 1)) != 0) { 10640026Smckusick getpathname(pathbuf, inp->i_number, inp->i_number); 10740026Smckusick pwarn("DIRECTORY %s: LENGTH %d NOT MULTIPLE OF %d", 10840026Smckusick pathbuf, inp->i_isize, DIRBLKSIZ); 10940026Smckusick if (preen) 11040026Smckusick printf(" (ADJUSTED)\n"); 11140026Smckusick inp->i_isize = roundup(inp->i_isize, DIRBLKSIZ); 11240026Smckusick if (preen || reply("ADJUST") == 1) { 11340026Smckusick dp = ginode(inp->i_number); 11440026Smckusick dp->di_size = roundup(inp->i_isize, DIRBLKSIZ); 11540026Smckusick inodirty(); 11640026Smckusick dp = &dino; 11740026Smckusick } 11840026Smckusick } 11966276Smkm bzero((char *)&dino, sizeof(struct dinode)); 12066276Smkm dino.di_mode = IFDIR; 12140026Smckusick dp->di_size = inp->i_isize; 12240026Smckusick bcopy((char *)&inp->i_blks[0], (char *)&dp->di_db[0], 12344934Smckusick (size_t)inp->i_numblks); 12440026Smckusick curino.id_number = inp->i_number; 12540026Smckusick curino.id_parent = inp->i_parent; 12640026Smckusick (void)ckinode(dp, &curino); 12740026Smckusick } 12840026Smckusick /* 12940026Smckusick * Now that the parents of all directories have been found, 13040026Smckusick * make another pass to verify the value of `..' 13140026Smckusick */ 13240026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) { 13340026Smckusick inp = *inpp; 13441136Smckusick if (inp->i_parent == 0 || inp->i_isize == 0) 13540026Smckusick continue; 13640026Smckusick if (statemap[inp->i_parent] == DFOUND && 13740026Smckusick statemap[inp->i_number] == DSTATE) 13840026Smckusick statemap[inp->i_number] = DFOUND; 13940026Smckusick if (inp->i_dotdot == inp->i_parent || 14040026Smckusick inp->i_dotdot == (ino_t)-1) 14140026Smckusick continue; 14240026Smckusick if (inp->i_dotdot == 0) { 14340026Smckusick inp->i_dotdot = inp->i_parent; 14440026Smckusick fileerror(inp->i_parent, inp->i_number, "MISSING '..'"); 14540026Smckusick if (reply("FIX") == 0) 14640026Smckusick continue; 14744934Smckusick (void)makeentry(inp->i_number, inp->i_parent, ".."); 14840026Smckusick lncntp[inp->i_parent]--; 14940026Smckusick continue; 15040026Smckusick } 15140026Smckusick fileerror(inp->i_parent, inp->i_number, 15254603Smckusick "BAD INODE NUMBER FOR '..'"); 15340026Smckusick if (reply("FIX") == 0) 15440026Smckusick continue; 15540026Smckusick lncntp[inp->i_dotdot]++; 15640026Smckusick lncntp[inp->i_parent]--; 15740026Smckusick inp->i_dotdot = inp->i_parent; 15840026Smckusick (void)changeino(inp->i_number, "..", inp->i_parent); 15940026Smckusick } 16040026Smckusick /* 16140026Smckusick * Mark all the directories that can be found from the root. 16240026Smckusick */ 16340026Smckusick propagate(); 16416264Smckusick } 16516264Smckusick 16616264Smckusick pass2check(idesc) 16716264Smckusick struct inodesc *idesc; 16816264Smckusick { 16939973Smckusick register struct direct *dirp = idesc->id_dirp; 17040026Smckusick register struct inoinfo *inp; 17116264Smckusick int n, entrysize, ret = 0; 17239973Smckusick struct dinode *dp; 17341136Smckusick char *errmsg; 17439973Smckusick struct direct proto; 17540026Smckusick char namebuf[MAXPATHLEN + 1]; 17640026Smckusick char pathbuf[MAXPATHLEN + 1]; 17716264Smckusick 17854603Smckusick /* 17954603Smckusick * If converting, set directory entry type. 18054603Smckusick */ 18154603Smckusick if (doinglevel2 && dirp->d_ino > 0 && dirp->d_ino < maxino) { 18254603Smckusick dirp->d_type = typemap[dirp->d_ino]; 18354603Smckusick ret |= ALTERED; 18454603Smckusick } 18516264Smckusick /* 18616264Smckusick * check for "." 18716264Smckusick */ 18816264Smckusick if (idesc->id_entryno != 0) 18916264Smckusick goto chk1; 19016264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") == 0) { 19116264Smckusick if (dirp->d_ino != idesc->id_number) { 19239973Smckusick direrror(idesc->id_number, "BAD INODE NUMBER FOR '.'"); 19316264Smckusick dirp->d_ino = idesc->id_number; 19416264Smckusick if (reply("FIX") == 1) 19516264Smckusick ret |= ALTERED; 19616264Smckusick } 19754603Smckusick if (newinofmt && dirp->d_type != DT_DIR) { 19854603Smckusick direrror(idesc->id_number, "BAD TYPE VALUE FOR '.'"); 19954603Smckusick dirp->d_type = DT_DIR; 20054603Smckusick if (reply("FIX") == 1) 20154603Smckusick ret |= ALTERED; 20254603Smckusick } 20316264Smckusick goto chk1; 20416264Smckusick } 20539973Smckusick direrror(idesc->id_number, "MISSING '.'"); 20616264Smckusick proto.d_ino = idesc->id_number; 20754603Smckusick if (newinofmt) 20854603Smckusick proto.d_type = DT_DIR; 20955074Smckusick else 21055074Smckusick proto.d_type = 0; 21116264Smckusick proto.d_namlen = 1; 21216264Smckusick (void)strcpy(proto.d_name, "."); 21354603Smckusick entrysize = DIRSIZ(0, &proto); 21416264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") != 0) { 21516264Smckusick pfatal("CANNOT FIX, FIRST ENTRY IN DIRECTORY CONTAINS %s\n", 21616264Smckusick dirp->d_name); 21716264Smckusick } else if (dirp->d_reclen < entrysize) { 21816264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '.'\n"); 21916264Smckusick } else if (dirp->d_reclen < 2 * entrysize) { 22016264Smckusick proto.d_reclen = dirp->d_reclen; 22144934Smckusick bcopy((char *)&proto, (char *)dirp, (size_t)entrysize); 22216264Smckusick if (reply("FIX") == 1) 22316264Smckusick ret |= ALTERED; 22416264Smckusick } else { 22516264Smckusick n = dirp->d_reclen - entrysize; 22616264Smckusick proto.d_reclen = entrysize; 22744934Smckusick bcopy((char *)&proto, (char *)dirp, (size_t)entrysize); 22816264Smckusick idesc->id_entryno++; 22916264Smckusick lncntp[dirp->d_ino]--; 23039973Smckusick dirp = (struct direct *)((char *)(dirp) + entrysize); 23144934Smckusick bzero((char *)dirp, (size_t)n); 23216264Smckusick dirp->d_reclen = n; 23316264Smckusick if (reply("FIX") == 1) 23416264Smckusick ret |= ALTERED; 23516264Smckusick } 23616264Smckusick chk1: 23716264Smckusick if (idesc->id_entryno > 1) 23816264Smckusick goto chk2; 23940026Smckusick inp = getinoinfo(idesc->id_number); 24040026Smckusick proto.d_ino = inp->i_parent; 24154603Smckusick if (newinofmt) 24254603Smckusick proto.d_type = DT_DIR; 24355074Smckusick else 24455074Smckusick proto.d_type = 0; 24516264Smckusick proto.d_namlen = 2; 24616264Smckusick (void)strcpy(proto.d_name, ".."); 24754603Smckusick entrysize = DIRSIZ(0, &proto); 24816264Smckusick if (idesc->id_entryno == 0) { 24954603Smckusick n = DIRSIZ(0, dirp); 25016264Smckusick if (dirp->d_reclen < n + entrysize) 25116264Smckusick goto chk2; 25216264Smckusick proto.d_reclen = dirp->d_reclen - n; 25316264Smckusick dirp->d_reclen = n; 25416264Smckusick idesc->id_entryno++; 25516264Smckusick lncntp[dirp->d_ino]--; 25639973Smckusick dirp = (struct direct *)((char *)(dirp) + n); 25744934Smckusick bzero((char *)dirp, (size_t)proto.d_reclen); 25840026Smckusick dirp->d_reclen = proto.d_reclen; 25916264Smckusick } 26016264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") == 0) { 26140026Smckusick inp->i_dotdot = dirp->d_ino; 26254603Smckusick if (newinofmt && dirp->d_type != DT_DIR) { 26354603Smckusick direrror(idesc->id_number, "BAD TYPE VALUE FOR '..'"); 26454603Smckusick dirp->d_type = DT_DIR; 26554603Smckusick if (reply("FIX") == 1) 26654603Smckusick ret |= ALTERED; 26754603Smckusick } 26816264Smckusick goto chk2; 26916264Smckusick } 27016264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") != 0) { 27140026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 27216264Smckusick pfatal("CANNOT FIX, SECOND ENTRY IN DIRECTORY CONTAINS %s\n", 27316264Smckusick dirp->d_name); 27440026Smckusick inp->i_dotdot = (ino_t)-1; 27516264Smckusick } else if (dirp->d_reclen < entrysize) { 27640026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 27716264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '..'\n"); 27840026Smckusick inp->i_dotdot = (ino_t)-1; 27940026Smckusick } else if (inp->i_parent != 0) { 28040026Smckusick /* 28140026Smckusick * We know the parent, so fix now. 28240026Smckusick */ 28340026Smckusick inp->i_dotdot = inp->i_parent; 28440026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 28516264Smckusick proto.d_reclen = dirp->d_reclen; 28644934Smckusick bcopy((char *)&proto, (char *)dirp, (size_t)entrysize); 28716264Smckusick if (reply("FIX") == 1) 28816264Smckusick ret |= ALTERED; 28916264Smckusick } 29040026Smckusick idesc->id_entryno++; 29140026Smckusick if (dirp->d_ino != 0) 29240026Smckusick lncntp[dirp->d_ino]--; 29340026Smckusick return (ret|KEEPON); 29416264Smckusick chk2: 29516264Smckusick if (dirp->d_ino == 0) 29616264Smckusick return (ret|KEEPON); 29716264Smckusick if (dirp->d_namlen <= 2 && 29816264Smckusick dirp->d_name[0] == '.' && 29916264Smckusick idesc->id_entryno >= 2) { 30016264Smckusick if (dirp->d_namlen == 1) { 30139973Smckusick direrror(idesc->id_number, "EXTRA '.' ENTRY"); 30216264Smckusick dirp->d_ino = 0; 30316264Smckusick if (reply("FIX") == 1) 30416264Smckusick ret |= ALTERED; 30516264Smckusick return (KEEPON | ret); 30616264Smckusick } 30716264Smckusick if (dirp->d_name[1] == '.') { 30839973Smckusick direrror(idesc->id_number, "EXTRA '..' ENTRY"); 30916264Smckusick dirp->d_ino = 0; 31016264Smckusick if (reply("FIX") == 1) 31116264Smckusick ret |= ALTERED; 31216264Smckusick return (KEEPON | ret); 31316264Smckusick } 31416264Smckusick } 31516264Smckusick idesc->id_entryno++; 31616264Smckusick n = 0; 31745847Smckusick if (dirp->d_ino > maxino) { 31840026Smckusick fileerror(idesc->id_number, dirp->d_ino, "I OUT OF RANGE"); 31916264Smckusick n = reply("REMOVE"); 32067575Spendry } else if (newinofmt && 321*67670Smckusick ((dirp->d_ino == WINO && dirp->d_type != DT_WHT) || 322*67670Smckusick (dirp->d_ino != WINO && dirp->d_type == DT_WHT))) { 32367575Spendry fileerror(idesc->id_number, dirp->d_ino, "BAD WHITEOUT ENTRY"); 32467575Spendry dirp->d_ino = WINO; 32567575Spendry dirp->d_type = DT_WHT; 32667575Spendry if (reply("FIX") == 1) 32767575Spendry ret |= ALTERED; 32816264Smckusick } else { 32916264Smckusick again: 33016264Smckusick switch (statemap[dirp->d_ino]) { 33116264Smckusick case USTATE: 33240569Smckusick if (idesc->id_entryno <= 2) 33340569Smckusick break; 33440026Smckusick fileerror(idesc->id_number, dirp->d_ino, "UNALLOCATED"); 33516264Smckusick n = reply("REMOVE"); 33616264Smckusick break; 33716264Smckusick 33817937Smckusick case DCLEAR: 33917937Smckusick case FCLEAR: 34040569Smckusick if (idesc->id_entryno <= 2) 34140569Smckusick break; 34257717Smckusick if (statemap[dirp->d_ino] == FCLEAR) 34357717Smckusick errmsg = "DUP/BAD"; 34457717Smckusick else if (!preen) 34541136Smckusick errmsg = "ZERO LENGTH DIRECTORY"; 34657717Smckusick else { 34757717Smckusick n = 1; 34857717Smckusick break; 34957717Smckusick } 35041136Smckusick fileerror(idesc->id_number, dirp->d_ino, errmsg); 35116264Smckusick if ((n = reply("REMOVE")) == 1) 35216264Smckusick break; 35317943Smckusick dp = ginode(dirp->d_ino); 35439973Smckusick statemap[dirp->d_ino] = 35539973Smckusick (dp->di_mode & IFMT) == IFDIR ? DSTATE : FSTATE; 35630599Smckusick lncntp[dirp->d_ino] = dp->di_nlink; 35716264Smckusick goto again; 35816264Smckusick 35940026Smckusick case DSTATE: 36040026Smckusick if (statemap[idesc->id_number] == DFOUND) 36140026Smckusick statemap[dirp->d_ino] = DFOUND; 36240026Smckusick /* fall through */ 36340026Smckusick 36417937Smckusick case DFOUND: 36540026Smckusick inp = getinoinfo(dirp->d_ino); 36640026Smckusick if (inp->i_parent != 0 && idesc->id_entryno > 2) { 36740026Smckusick getpathname(pathbuf, idesc->id_number, 36840026Smckusick idesc->id_number); 36917991Smckusick getpathname(namebuf, dirp->d_ino, dirp->d_ino); 37040026Smckusick pwarn("%s %s %s\n", pathbuf, 37117991Smckusick "IS AN EXTRANEOUS HARD LINK TO DIRECTORY", 37217991Smckusick namebuf); 37317991Smckusick if (preen) 37417991Smckusick printf(" (IGNORED)\n"); 37517991Smckusick else if ((n = reply("REMOVE")) == 1) 37617991Smckusick break; 37717991Smckusick } 37840026Smckusick if (idesc->id_entryno > 2) 37940026Smckusick inp->i_parent = idesc->id_number; 38017937Smckusick /* fall through */ 38117937Smckusick 38216264Smckusick case FSTATE: 38354603Smckusick if (newinofmt && dirp->d_type != typemap[dirp->d_ino]) { 38454603Smckusick fileerror(idesc->id_number, dirp->d_ino, 38554603Smckusick "BAD TYPE VALUE"); 38654603Smckusick dirp->d_type = typemap[dirp->d_ino]; 38754603Smckusick if (reply("FIX") == 1) 38854603Smckusick ret |= ALTERED; 38954603Smckusick } 39016264Smckusick lncntp[dirp->d_ino]--; 39116264Smckusick break; 39216264Smckusick 39326480Smckusick default: 39426480Smckusick errexit("BAD STATE %d FOR INODE I=%d", 39526480Smckusick statemap[dirp->d_ino], dirp->d_ino); 39616264Smckusick } 39716264Smckusick } 39816264Smckusick if (n == 0) 39916264Smckusick return (ret|KEEPON); 40016264Smckusick dirp->d_ino = 0; 40116264Smckusick return (ret|KEEPON|ALTERED); 40216264Smckusick } 40340026Smckusick 40440026Smckusick /* 40540026Smckusick * Routine to sort disk blocks. 40640026Smckusick */ 40740026Smckusick blksort(inpp1, inpp2) 40840026Smckusick struct inoinfo **inpp1, **inpp2; 40940026Smckusick { 41040026Smckusick 41140026Smckusick return ((*inpp1)->i_blks[0] - (*inpp2)->i_blks[0]); 41240026Smckusick } 413