122050Sdist /* 239976Smckusick * Copyright (c) 1980, 1986 The Regents of the University of California. 339976Smckusick * All rights reserved. 439976Smckusick * 5*42702Sbostic * %sccs.include.redist.c% 622050Sdist */ 722050Sdist 816264Smckusick #ifndef lint 9*42702Sbostic static char sccsid[] = "@(#)pass2.c 5.13 (Berkeley) 06/01/90"; 1039976Smckusick #endif /* not lint */ 1116264Smckusick 1216264Smckusick #include <sys/param.h> 1339383Smckusick #include <ufs/dinode.h> 1438337Smckusick #include <ufs/fs.h> 1540026Smckusick #define KERNEL 1638337Smckusick #include <ufs/dir.h> 1740026Smckusick #undef KERNEL 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 */ 8340026Smckusick qsort((char *)inpsort, (int)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"); 9840026Smckusick inp->i_isize = MINDIRSIZE; 9940026Smckusick if (reply("FIX") == 1) { 10040026Smckusick dp = ginode(inp->i_number); 10140026Smckusick dp->di_size = MINDIRSIZE; 10240026Smckusick inodirty(); 10340026Smckusick dp = &dino; 10440026Smckusick } 10540026Smckusick } 10640026Smckusick if ((inp->i_isize & (DIRBLKSIZ - 1)) != 0) { 10740026Smckusick getpathname(pathbuf, inp->i_number, inp->i_number); 10840026Smckusick pwarn("DIRECTORY %s: LENGTH %d NOT MULTIPLE OF %d", 10940026Smckusick pathbuf, inp->i_isize, DIRBLKSIZ); 11040026Smckusick if (preen) 11140026Smckusick printf(" (ADJUSTED)\n"); 11240026Smckusick inp->i_isize = roundup(inp->i_isize, DIRBLKSIZ); 11340026Smckusick if (preen || reply("ADJUST") == 1) { 11440026Smckusick dp = ginode(inp->i_number); 11540026Smckusick dp->di_size = roundup(inp->i_isize, DIRBLKSIZ); 11640026Smckusick inodirty(); 11740026Smckusick dp = &dino; 11840026Smckusick } 11940026Smckusick } 12040026Smckusick dp->di_size = inp->i_isize; 12140026Smckusick bcopy((char *)&inp->i_blks[0], (char *)&dp->di_db[0], 12240026Smckusick (int)inp->i_numblks); 12340026Smckusick curino.id_number = inp->i_number; 12440026Smckusick curino.id_parent = inp->i_parent; 12540026Smckusick (void)ckinode(dp, &curino); 12640026Smckusick } 12740026Smckusick /* 12840026Smckusick * Now that the parents of all directories have been found, 12940026Smckusick * make another pass to verify the value of `..' 13040026Smckusick */ 13140026Smckusick for (inpp = inpsort; inpp < inpend; inpp++) { 13240026Smckusick inp = *inpp; 13341136Smckusick if (inp->i_parent == 0 || inp->i_isize == 0) 13440026Smckusick continue; 13540026Smckusick if (statemap[inp->i_parent] == DFOUND && 13640026Smckusick statemap[inp->i_number] == DSTATE) 13740026Smckusick statemap[inp->i_number] = DFOUND; 13840026Smckusick if (inp->i_dotdot == inp->i_parent || 13940026Smckusick inp->i_dotdot == (ino_t)-1) 14040026Smckusick continue; 14140026Smckusick if (inp->i_dotdot == 0) { 14240026Smckusick inp->i_dotdot = inp->i_parent; 14340026Smckusick fileerror(inp->i_parent, inp->i_number, "MISSING '..'"); 14440026Smckusick if (reply("FIX") == 0) 14540026Smckusick continue; 14640026Smckusick makeentry(inp->i_number, inp->i_parent, ".."); 14740026Smckusick lncntp[inp->i_parent]--; 14840026Smckusick continue; 14940026Smckusick } 15040026Smckusick fileerror(inp->i_parent, inp->i_number, 15140026Smckusick "BAD INODE NUMBER FOR '..'"); 15240026Smckusick if (reply("FIX") == 0) 15340026Smckusick continue; 15440026Smckusick lncntp[inp->i_dotdot]++; 15540026Smckusick lncntp[inp->i_parent]--; 15640026Smckusick inp->i_dotdot = inp->i_parent; 15740026Smckusick (void)changeino(inp->i_number, "..", inp->i_parent); 15840026Smckusick } 15940026Smckusick /* 16040026Smckusick * Mark all the directories that can be found from the root. 16140026Smckusick */ 16240026Smckusick propagate(); 16316264Smckusick } 16416264Smckusick 16516264Smckusick pass2check(idesc) 16616264Smckusick struct inodesc *idesc; 16716264Smckusick { 16839973Smckusick register struct direct *dirp = idesc->id_dirp; 16940026Smckusick register struct inoinfo *inp; 17016264Smckusick int n, entrysize, ret = 0; 17139973Smckusick struct dinode *dp; 17241136Smckusick char *errmsg; 17339973Smckusick struct direct proto; 17440026Smckusick char namebuf[MAXPATHLEN + 1]; 17540026Smckusick char pathbuf[MAXPATHLEN + 1]; 17616264Smckusick 17716264Smckusick /* 17816264Smckusick * check for "." 17916264Smckusick */ 18016264Smckusick if (idesc->id_entryno != 0) 18116264Smckusick goto chk1; 18216264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") == 0) { 18316264Smckusick if (dirp->d_ino != idesc->id_number) { 18439973Smckusick direrror(idesc->id_number, "BAD INODE NUMBER FOR '.'"); 18516264Smckusick dirp->d_ino = idesc->id_number; 18616264Smckusick if (reply("FIX") == 1) 18716264Smckusick ret |= ALTERED; 18816264Smckusick } 18916264Smckusick goto chk1; 19016264Smckusick } 19139973Smckusick direrror(idesc->id_number, "MISSING '.'"); 19216264Smckusick proto.d_ino = idesc->id_number; 19316264Smckusick proto.d_namlen = 1; 19416264Smckusick (void)strcpy(proto.d_name, "."); 19516264Smckusick entrysize = DIRSIZ(&proto); 19616264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") != 0) { 19716264Smckusick pfatal("CANNOT FIX, FIRST ENTRY IN DIRECTORY CONTAINS %s\n", 19816264Smckusick dirp->d_name); 19916264Smckusick } else if (dirp->d_reclen < entrysize) { 20016264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '.'\n"); 20116264Smckusick } else if (dirp->d_reclen < 2 * entrysize) { 20216264Smckusick proto.d_reclen = dirp->d_reclen; 20316264Smckusick bcopy((char *)&proto, (char *)dirp, entrysize); 20416264Smckusick if (reply("FIX") == 1) 20516264Smckusick ret |= ALTERED; 20616264Smckusick } else { 20716264Smckusick n = dirp->d_reclen - entrysize; 20816264Smckusick proto.d_reclen = entrysize; 20916264Smckusick bcopy((char *)&proto, (char *)dirp, entrysize); 21016264Smckusick idesc->id_entryno++; 21116264Smckusick lncntp[dirp->d_ino]--; 21239973Smckusick dirp = (struct direct *)((char *)(dirp) + entrysize); 21316264Smckusick bzero((char *)dirp, n); 21416264Smckusick dirp->d_reclen = n; 21516264Smckusick if (reply("FIX") == 1) 21616264Smckusick ret |= ALTERED; 21716264Smckusick } 21816264Smckusick chk1: 21916264Smckusick if (idesc->id_entryno > 1) 22016264Smckusick goto chk2; 22140026Smckusick inp = getinoinfo(idesc->id_number); 22240026Smckusick proto.d_ino = inp->i_parent; 22316264Smckusick proto.d_namlen = 2; 22416264Smckusick (void)strcpy(proto.d_name, ".."); 22516264Smckusick entrysize = DIRSIZ(&proto); 22616264Smckusick if (idesc->id_entryno == 0) { 22716264Smckusick n = DIRSIZ(dirp); 22816264Smckusick if (dirp->d_reclen < n + entrysize) 22916264Smckusick goto chk2; 23016264Smckusick proto.d_reclen = dirp->d_reclen - n; 23116264Smckusick dirp->d_reclen = n; 23216264Smckusick idesc->id_entryno++; 23316264Smckusick lncntp[dirp->d_ino]--; 23439973Smckusick dirp = (struct direct *)((char *)(dirp) + n); 23540026Smckusick bzero((char *)dirp, (int)proto.d_reclen); 23640026Smckusick dirp->d_reclen = proto.d_reclen; 23716264Smckusick } 23816264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") == 0) { 23940026Smckusick inp->i_dotdot = dirp->d_ino; 24016264Smckusick goto chk2; 24116264Smckusick } 24216264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") != 0) { 24340026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 24416264Smckusick pfatal("CANNOT FIX, SECOND ENTRY IN DIRECTORY CONTAINS %s\n", 24516264Smckusick dirp->d_name); 24640026Smckusick inp->i_dotdot = (ino_t)-1; 24716264Smckusick } else if (dirp->d_reclen < entrysize) { 24840026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 24916264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '..'\n"); 25040026Smckusick inp->i_dotdot = (ino_t)-1; 25140026Smckusick } else if (inp->i_parent != 0) { 25240026Smckusick /* 25340026Smckusick * We know the parent, so fix now. 25440026Smckusick */ 25540026Smckusick inp->i_dotdot = inp->i_parent; 25640026Smckusick fileerror(inp->i_parent, idesc->id_number, "MISSING '..'"); 25716264Smckusick proto.d_reclen = dirp->d_reclen; 25816264Smckusick bcopy((char *)&proto, (char *)dirp, entrysize); 25916264Smckusick if (reply("FIX") == 1) 26016264Smckusick ret |= ALTERED; 26116264Smckusick } 26240026Smckusick idesc->id_entryno++; 26340026Smckusick if (dirp->d_ino != 0) 26440026Smckusick lncntp[dirp->d_ino]--; 26540026Smckusick return (ret|KEEPON); 26616264Smckusick chk2: 26716264Smckusick if (dirp->d_ino == 0) 26816264Smckusick return (ret|KEEPON); 26916264Smckusick if (dirp->d_namlen <= 2 && 27016264Smckusick dirp->d_name[0] == '.' && 27116264Smckusick idesc->id_entryno >= 2) { 27216264Smckusick if (dirp->d_namlen == 1) { 27339973Smckusick direrror(idesc->id_number, "EXTRA '.' ENTRY"); 27416264Smckusick dirp->d_ino = 0; 27516264Smckusick if (reply("FIX") == 1) 27616264Smckusick ret |= ALTERED; 27716264Smckusick return (KEEPON | ret); 27816264Smckusick } 27916264Smckusick if (dirp->d_name[1] == '.') { 28039973Smckusick direrror(idesc->id_number, "EXTRA '..' ENTRY"); 28116264Smckusick dirp->d_ino = 0; 28216264Smckusick if (reply("FIX") == 1) 28316264Smckusick ret |= ALTERED; 28416264Smckusick return (KEEPON | ret); 28516264Smckusick } 28616264Smckusick } 28716264Smckusick idesc->id_entryno++; 28816264Smckusick n = 0; 28939973Smckusick if (dirp->d_ino > maxino || dirp->d_ino <= 0) { 29040026Smckusick fileerror(idesc->id_number, dirp->d_ino, "I OUT OF RANGE"); 29116264Smckusick n = reply("REMOVE"); 29216264Smckusick } else { 29316264Smckusick again: 29416264Smckusick switch (statemap[dirp->d_ino]) { 29516264Smckusick case USTATE: 29640569Smckusick if (idesc->id_entryno <= 2) 29740569Smckusick break; 29840026Smckusick fileerror(idesc->id_number, dirp->d_ino, "UNALLOCATED"); 29916264Smckusick n = reply("REMOVE"); 30016264Smckusick break; 30116264Smckusick 30217937Smckusick case DCLEAR: 30317937Smckusick case FCLEAR: 30440569Smckusick if (idesc->id_entryno <= 2) 30540569Smckusick break; 30641136Smckusick if (statemap[dirp->d_ino] == DCLEAR) 30741136Smckusick errmsg = "ZERO LENGTH DIRECTORY"; 30841136Smckusick else 30941136Smckusick errmsg = "DUP/BAD"; 31041136Smckusick fileerror(idesc->id_number, dirp->d_ino, errmsg); 31116264Smckusick if ((n = reply("REMOVE")) == 1) 31216264Smckusick break; 31317943Smckusick dp = ginode(dirp->d_ino); 31439973Smckusick statemap[dirp->d_ino] = 31539973Smckusick (dp->di_mode & IFMT) == IFDIR ? DSTATE : FSTATE; 31630599Smckusick lncntp[dirp->d_ino] = dp->di_nlink; 31716264Smckusick goto again; 31816264Smckusick 31940026Smckusick case DSTATE: 32040026Smckusick if (statemap[idesc->id_number] == DFOUND) 32140026Smckusick statemap[dirp->d_ino] = DFOUND; 32240026Smckusick /* fall through */ 32340026Smckusick 32417937Smckusick case DFOUND: 32540026Smckusick inp = getinoinfo(dirp->d_ino); 32640026Smckusick if (inp->i_parent != 0 && idesc->id_entryno > 2) { 32740026Smckusick getpathname(pathbuf, idesc->id_number, 32840026Smckusick idesc->id_number); 32917991Smckusick getpathname(namebuf, dirp->d_ino, dirp->d_ino); 33040026Smckusick pwarn("%s %s %s\n", pathbuf, 33117991Smckusick "IS AN EXTRANEOUS HARD LINK TO DIRECTORY", 33217991Smckusick namebuf); 33317991Smckusick if (preen) 33417991Smckusick printf(" (IGNORED)\n"); 33517991Smckusick else if ((n = reply("REMOVE")) == 1) 33617991Smckusick break; 33717991Smckusick } 33840026Smckusick if (idesc->id_entryno > 2) 33940026Smckusick inp->i_parent = idesc->id_number; 34017937Smckusick /* fall through */ 34117937Smckusick 34216264Smckusick case FSTATE: 34316264Smckusick lncntp[dirp->d_ino]--; 34416264Smckusick break; 34516264Smckusick 34626480Smckusick default: 34726480Smckusick errexit("BAD STATE %d FOR INODE I=%d", 34826480Smckusick statemap[dirp->d_ino], dirp->d_ino); 34916264Smckusick } 35016264Smckusick } 35116264Smckusick if (n == 0) 35216264Smckusick return (ret|KEEPON); 35316264Smckusick dirp->d_ino = 0; 35416264Smckusick return (ret|KEEPON|ALTERED); 35516264Smckusick } 35640026Smckusick 35740026Smckusick /* 35840026Smckusick * Routine to sort disk blocks. 35940026Smckusick */ 36040026Smckusick blksort(inpp1, inpp2) 36140026Smckusick struct inoinfo **inpp1, **inpp2; 36240026Smckusick { 36340026Smckusick 36440026Smckusick return ((*inpp1)->i_blks[0] - (*inpp2)->i_blks[0]); 36540026Smckusick } 366