122050Sdist /* 239976Smckusick * Copyright (c) 1980, 1986 The Regents of the University of California. 339976Smckusick * All rights reserved. 439976Smckusick * 542702Sbostic * %sccs.include.redist.c% 622050Sdist */ 722050Sdist 816264Smckusick #ifndef lint 9*53703Smckusick static char sccsid[] = "@(#)pass2.c 5.19 (Berkeley) 05/27/92"; 1039976Smckusick #endif /* not lint */ 1116264Smckusick 1216264Smckusick #include <sys/param.h> 13*53703Smckusick #include <sys/time.h> 1451532Sbostic #include <ufs/ufs/dinode.h> 1540026Smckusick #define KERNEL 1651532Sbostic #include <ufs/ufs/dir.h> 1740026Smckusick #undef KERNEL 1851532Sbostic #include <ufs/ffs/fs.h> 1944934Smckusick #include <stdlib.h> 2042041Sbostic #include <string.h> 2116264Smckusick #include "fsck.h" 2216264Smckusick 2340026Smckusick #define MINDIRSIZE (sizeof (struct dirtemplate)) 2416264Smckusick 2540026Smckusick int pass2check(), blksort(); 2640026Smckusick 2716264Smckusick pass2() 2816264Smckusick { 2939973Smckusick register struct dinode *dp; 3040026Smckusick register struct inoinfo **inpp, *inp; 3140026Smckusick struct inoinfo **inpend; 3240026Smckusick struct inodesc curino; 3340026Smckusick struct dinode dino; 3440026Smckusick char pathbuf[MAXPATHLEN + 1]; 3516264Smckusick 3616264Smckusick switch (statemap[ROOTINO]) { 3716264Smckusick 3816264Smckusick case USTATE: 3917966Smckusick pfatal("ROOT INODE UNALLOCATED"); 4017966Smckusick if (reply("ALLOCATE") == 0) 4117966Smckusick errexit(""); 4236827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 4317966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 4417966Smckusick break; 4516264Smckusick 4617966Smckusick case DCLEAR: 4717966Smckusick pfatal("DUPS/BAD IN ROOT INODE"); 4817966Smckusick if (reply("REALLOCATE")) { 4917966Smckusick freeino(ROOTINO); 5036827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 5117966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 5217966Smckusick break; 5317966Smckusick } 5417966Smckusick if (reply("CONTINUE") == 0) 5517966Smckusick errexit(""); 5617966Smckusick break; 5717966Smckusick 5816264Smckusick case FSTATE: 5917937Smckusick case FCLEAR: 6016264Smckusick pfatal("ROOT INODE NOT DIRECTORY"); 6117966Smckusick if (reply("REALLOCATE")) { 6217966Smckusick freeino(ROOTINO); 6336827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 6417966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 6517966Smckusick break; 6617966Smckusick } 6717943Smckusick if (reply("FIX") == 0) 6816264Smckusick errexit(""); 6917943Smckusick dp = ginode(ROOTINO); 7016264Smckusick dp->di_mode &= ~IFMT; 7116264Smckusick dp->di_mode |= IFDIR; 7216264Smckusick inodirty(); 7340026Smckusick break; 7416264Smckusick 7516264Smckusick case DSTATE: 7616264Smckusick break; 7716264Smckusick 7817937Smckusick default: 7917937Smckusick errexit("BAD STATE %d FOR ROOT INODE", statemap[ROOTINO]); 8016264Smckusick } 8140026Smckusick statemap[ROOTINO] = DFOUND; 8240026Smckusick /* 8340026Smckusick * Sort the directory list into disk block order. 8440026Smckusick */ 8544934Smckusick qsort((char *)inpsort, (size_t)inplast, sizeof *inpsort, blksort); 8640026Smckusick /* 8740026Smckusick * Check the integrity of each directory. 8840026Smckusick */ 8940026Smckusick bzero((char *)&curino, sizeof(struct inodesc)); 9040026Smckusick curino.id_type = DATA; 9140026Smckusick curino.id_func = pass2check; 9245239Smckusick dino.di_mode = IFDIR; 9340026Smckusick dp = &dino; 9440026Smckusick inpend = &inpsort[inplast]; 9540026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) { 9640026Smckusick inp = *inpp; 9741136Smckusick if (inp->i_isize == 0) 9841136Smckusick continue; 9940026Smckusick if (inp->i_isize < MINDIRSIZE) { 10040026Smckusick direrror(inp->i_number, "DIRECTORY TOO SHORT"); 10145002Smckusick inp->i_isize = roundup(MINDIRSIZE, DIRBLKSIZ); 10240026Smckusick if (reply("FIX") == 1) { 10340026Smckusick dp = ginode(inp->i_number); 10445002Smckusick dp->di_size = inp->i_isize; 10540026Smckusick inodirty(); 10640026Smckusick dp = &dino; 10740026Smckusick } 10845002Smckusick } else if ((inp->i_isize & (DIRBLKSIZ - 1)) != 0) { 10940026Smckusick getpathname(pathbuf, inp->i_number, inp->i_number); 11040026Smckusick pwarn("DIRECTORY %s: LENGTH %d NOT MULTIPLE OF %d", 11140026Smckusick pathbuf, inp->i_isize, DIRBLKSIZ); 11240026Smckusick if (preen) 11340026Smckusick printf(" (ADJUSTED)\n"); 11440026Smckusick inp->i_isize = roundup(inp->i_isize, DIRBLKSIZ); 11540026Smckusick if (preen || reply("ADJUST") == 1) { 11640026Smckusick dp = ginode(inp->i_number); 11740026Smckusick dp->di_size = roundup(inp->i_isize, DIRBLKSIZ); 11840026Smckusick inodirty(); 11940026Smckusick dp = &dino; 12040026Smckusick } 12140026Smckusick } 12240026Smckusick dp->di_size = inp->i_isize; 12340026Smckusick bcopy((char *)&inp->i_blks[0], (char *)&dp->di_db[0], 12444934Smckusick (size_t)inp->i_numblks); 12540026Smckusick curino.id_number = inp->i_number; 12640026Smckusick curino.id_parent = inp->i_parent; 12740026Smckusick (void)ckinode(dp, &curino); 12840026Smckusick } 12940026Smckusick /* 13040026Smckusick * Now that the parents of all directories have been found, 13140026Smckusick * make another pass to verify the value of `..' 13240026Smckusick */ 13340026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) { 13440026Smckusick inp = *inpp; 13541136Smckusick if (inp->i_parent == 0 || inp->i_isize == 0) 13640026Smckusick continue; 13740026Smckusick if (statemap[inp->i_parent] == DFOUND && 13840026Smckusick statemap[inp->i_number] == DSTATE) 13940026Smckusick statemap[inp->i_number] = DFOUND; 14040026Smckusick if (inp->i_dotdot == inp->i_parent || 14140026Smckusick inp->i_dotdot == (ino_t)-1) 14240026Smckusick continue; 14340026Smckusick if (inp->i_dotdot == 0) { 14440026Smckusick inp->i_dotdot = inp->i_parent; 14540026Smckusick fileerror(inp->i_parent, inp->i_number, "MISSING '..'"); 14640026Smckusick if (reply("FIX") == 0) 14740026Smckusick continue; 14844934Smckusick (void)makeentry(inp->i_number, inp->i_parent, ".."); 14940026Smckusick lncntp[inp->i_parent]--; 15040026Smckusick continue; 15140026Smckusick } 15240026Smckusick fileerror(inp->i_parent, inp->i_number, 15340026Smckusick "BAD INODE NUMBER FOR '..'"); 15440026Smckusick if (reply("FIX") == 0) 15540026Smckusick continue; 15640026Smckusick lncntp[inp->i_dotdot]++; 15740026Smckusick lncntp[inp->i_parent]--; 15840026Smckusick inp->i_dotdot = inp->i_parent; 15940026Smckusick (void)changeino(inp->i_number, "..", inp->i_parent); 16040026Smckusick } 16140026Smckusick /* 16240026Smckusick * Mark all the directories that can be found from the root. 16340026Smckusick */ 16440026Smckusick propagate(); 16516264Smckusick } 16616264Smckusick 16716264Smckusick pass2check(idesc) 16816264Smckusick struct inodesc *idesc; 16916264Smckusick { 17039973Smckusick register struct direct *dirp = idesc->id_dirp; 17140026Smckusick register struct inoinfo *inp; 17216264Smckusick int n, entrysize, ret = 0; 17339973Smckusick struct dinode *dp; 17441136Smckusick char *errmsg; 17539973Smckusick struct direct proto; 17640026Smckusick char namebuf[MAXPATHLEN + 1]; 17740026Smckusick char pathbuf[MAXPATHLEN + 1]; 17816264Smckusick 17916264Smckusick /* 18016264Smckusick * check for "." 18116264Smckusick */ 18216264Smckusick if (idesc->id_entryno != 0) 18316264Smckusick goto chk1; 18416264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") == 0) { 18516264Smckusick if (dirp->d_ino != idesc->id_number) { 18639973Smckusick direrror(idesc->id_number, "BAD INODE NUMBER FOR '.'"); 18716264Smckusick dirp->d_ino = idesc->id_number; 18816264Smckusick if (reply("FIX") == 1) 18916264Smckusick ret |= ALTERED; 19016264Smckusick } 19116264Smckusick goto chk1; 19216264Smckusick } 19339973Smckusick direrror(idesc->id_number, "MISSING '.'"); 19416264Smckusick proto.d_ino = idesc->id_number; 19516264Smckusick proto.d_namlen = 1; 19616264Smckusick (void)strcpy(proto.d_name, "."); 19716264Smckusick entrysize = DIRSIZ(&proto); 19816264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") != 0) { 19916264Smckusick pfatal("CANNOT FIX, FIRST ENTRY IN DIRECTORY CONTAINS %s\n", 20016264Smckusick dirp->d_name); 20116264Smckusick } else if (dirp->d_reclen < entrysize) { 20216264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '.'\n"); 20316264Smckusick } else if (dirp->d_reclen < 2 * entrysize) { 20416264Smckusick proto.d_reclen = dirp->d_reclen; 20544934Smckusick bcopy((char *)&proto, (char *)dirp, (size_t)entrysize); 20616264Smckusick if (reply("FIX") == 1) 20716264Smckusick ret |= ALTERED; 20816264Smckusick } else { 20916264Smckusick n = dirp->d_reclen - entrysize; 21016264Smckusick proto.d_reclen = entrysize; 21144934Smckusick bcopy((char *)&proto, (char *)dirp, (size_t)entrysize); 21216264Smckusick idesc->id_entryno++; 21316264Smckusick lncntp[dirp->d_ino]--; 21439973Smckusick dirp = (struct direct *)((char *)(dirp) + entrysize); 21544934Smckusick bzero((char *)dirp, (size_t)n); 21616264Smckusick dirp->d_reclen = n; 21716264Smckusick if (reply("FIX") == 1) 21816264Smckusick ret |= ALTERED; 21916264Smckusick } 22016264Smckusick chk1: 22116264Smckusick if (idesc->id_entryno > 1) 22216264Smckusick goto chk2; 22340026Smckusick inp = getinoinfo(idesc->id_number); 22440026Smckusick proto.d_ino = inp->i_parent; 22516264Smckusick proto.d_namlen = 2; 22616264Smckusick (void)strcpy(proto.d_name, ".."); 22716264Smckusick entrysize = DIRSIZ(&proto); 22816264Smckusick if (idesc->id_entryno == 0) { 22916264Smckusick n = DIRSIZ(dirp); 23016264Smckusick if (dirp->d_reclen < n + entrysize) 23116264Smckusick goto chk2; 23216264Smckusick proto.d_reclen = dirp->d_reclen - n; 23316264Smckusick dirp->d_reclen = n; 23416264Smckusick idesc->id_entryno++; 23516264Smckusick lncntp[dirp->d_ino]--; 23639973Smckusick dirp = (struct direct *)((char *)(dirp) + n); 23744934Smckusick bzero((char *)dirp, (size_t)proto.d_reclen); 23840026Smckusick dirp->d_reclen = proto.d_reclen; 23916264Smckusick } 24016264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") == 0) { 24140026Smckusick inp->i_dotdot = dirp->d_ino; 24216264Smckusick goto chk2; 24316264Smckusick } 24416264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") != 0) { 24540026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 24616264Smckusick pfatal("CANNOT FIX, SECOND ENTRY IN DIRECTORY CONTAINS %s\n", 24716264Smckusick dirp->d_name); 24840026Smckusick inp->i_dotdot = (ino_t)-1; 24916264Smckusick } else if (dirp->d_reclen < entrysize) { 25040026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 25116264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '..'\n"); 25240026Smckusick inp->i_dotdot = (ino_t)-1; 25340026Smckusick } else if (inp->i_parent != 0) { 25440026Smckusick /* 25540026Smckusick * We know the parent, so fix now. 25640026Smckusick */ 25740026Smckusick inp->i_dotdot = inp->i_parent; 25840026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 25916264Smckusick proto.d_reclen = dirp->d_reclen; 26044934Smckusick bcopy((char *)&proto, (char *)dirp, (size_t)entrysize); 26116264Smckusick if (reply("FIX") == 1) 26216264Smckusick ret |= ALTERED; 26316264Smckusick } 26440026Smckusick idesc->id_entryno++; 26540026Smckusick if (dirp->d_ino != 0) 26640026Smckusick lncntp[dirp->d_ino]--; 26740026Smckusick return (ret|KEEPON); 26816264Smckusick chk2: 26916264Smckusick if (dirp->d_ino == 0) 27016264Smckusick return (ret|KEEPON); 27116264Smckusick if (dirp->d_namlen <= 2 && 27216264Smckusick dirp->d_name[0] == '.' && 27316264Smckusick idesc->id_entryno >= 2) { 27416264Smckusick if (dirp->d_namlen == 1) { 27539973Smckusick direrror(idesc->id_number, "EXTRA '.' ENTRY"); 27616264Smckusick dirp->d_ino = 0; 27716264Smckusick if (reply("FIX") == 1) 27816264Smckusick ret |= ALTERED; 27916264Smckusick return (KEEPON | ret); 28016264Smckusick } 28116264Smckusick if (dirp->d_name[1] == '.') { 28239973Smckusick direrror(idesc->id_number, "EXTRA '..' ENTRY"); 28316264Smckusick dirp->d_ino = 0; 28416264Smckusick if (reply("FIX") == 1) 28516264Smckusick ret |= ALTERED; 28616264Smckusick return (KEEPON | ret); 28716264Smckusick } 28816264Smckusick } 28916264Smckusick idesc->id_entryno++; 29016264Smckusick n = 0; 29145847Smckusick if (dirp->d_ino > maxino) { 29240026Smckusick fileerror(idesc->id_number, dirp->d_ino, "I OUT OF RANGE"); 29316264Smckusick n = reply("REMOVE"); 29416264Smckusick } else { 29516264Smckusick again: 29616264Smckusick switch (statemap[dirp->d_ino]) { 29716264Smckusick case USTATE: 29840569Smckusick if (idesc->id_entryno <= 2) 29940569Smckusick break; 30040026Smckusick fileerror(idesc->id_number, dirp->d_ino, "UNALLOCATED"); 30116264Smckusick n = reply("REMOVE"); 30216264Smckusick break; 30316264Smckusick 30417937Smckusick case DCLEAR: 30517937Smckusick case FCLEAR: 30640569Smckusick if (idesc->id_entryno <= 2) 30740569Smckusick break; 30841136Smckusick if (statemap[dirp->d_ino] == DCLEAR) 30941136Smckusick errmsg = "ZERO LENGTH DIRECTORY"; 31041136Smckusick else 31141136Smckusick errmsg = "DUP/BAD"; 31241136Smckusick fileerror(idesc->id_number, dirp->d_ino, errmsg); 31316264Smckusick if ((n = reply("REMOVE")) == 1) 31416264Smckusick break; 31517943Smckusick dp = ginode(dirp->d_ino); 31639973Smckusick statemap[dirp->d_ino] = 31739973Smckusick (dp->di_mode & IFMT) == IFDIR ? DSTATE : FSTATE; 31830599Smckusick lncntp[dirp->d_ino] = dp->di_nlink; 31916264Smckusick goto again; 32016264Smckusick 32140026Smckusick case DSTATE: 32240026Smckusick if (statemap[idesc->id_number] == DFOUND) 32340026Smckusick statemap[dirp->d_ino] = DFOUND; 32440026Smckusick /* fall through */ 32540026Smckusick 32617937Smckusick case DFOUND: 32740026Smckusick inp = getinoinfo(dirp->d_ino); 32840026Smckusick if (inp->i_parent != 0 && idesc->id_entryno > 2) { 32940026Smckusick getpathname(pathbuf, idesc->id_number, 33040026Smckusick idesc->id_number); 33117991Smckusick getpathname(namebuf, dirp->d_ino, dirp->d_ino); 33240026Smckusick pwarn("%s %s %s\n", pathbuf, 33317991Smckusick "IS AN EXTRANEOUS HARD LINK TO DIRECTORY", 33417991Smckusick namebuf); 33517991Smckusick if (preen) 33617991Smckusick printf(" (IGNORED)\n"); 33717991Smckusick else if ((n = reply("REMOVE")) == 1) 33817991Smckusick break; 33917991Smckusick } 34040026Smckusick if (idesc->id_entryno > 2) 34140026Smckusick inp->i_parent = idesc->id_number; 34217937Smckusick /* fall through */ 34317937Smckusick 34416264Smckusick case FSTATE: 34516264Smckusick lncntp[dirp->d_ino]--; 34616264Smckusick break; 34716264Smckusick 34826480Smckusick default: 34926480Smckusick errexit("BAD STATE %d FOR INODE I=%d", 35026480Smckusick statemap[dirp->d_ino], dirp->d_ino); 35116264Smckusick } 35216264Smckusick } 35316264Smckusick if (n == 0) 35416264Smckusick return (ret|KEEPON); 35516264Smckusick dirp->d_ino = 0; 35616264Smckusick return (ret|KEEPON|ALTERED); 35716264Smckusick } 35840026Smckusick 35940026Smckusick /* 36040026Smckusick * Routine to sort disk blocks. 36140026Smckusick */ 36240026Smckusick blksort(inpp1, inpp2) 36340026Smckusick struct inoinfo **inpp1, **inpp2; 36440026Smckusick { 36540026Smckusick 36640026Smckusick return ((*inpp1)->i_blks[0] - (*inpp2)->i_blks[0]); 36740026Smckusick } 368