122050Sdist /* 222050Sdist * Copyright (c) 1980 Regents of the University of California. 322050Sdist * All rights reserved. The Berkeley software License Agreement 422050Sdist * specifies the terms and conditions for redistribution. 522050Sdist */ 622050Sdist 716264Smckusick #ifndef lint 8*38337Smckusick static char sccsid[] = "@(#)pass2.c 5.5 (Berkeley) 06/26/89"; 922050Sdist #endif not lint 1016264Smckusick 1116264Smckusick #include <sys/param.h> 12*38337Smckusick #include <sys/time.h> 13*38337Smckusick #include <sys/vnode.h> 14*38337Smckusick #include <ufs/inode.h> 15*38337Smckusick #include <ufs/fs.h> 16*38337Smckusick #include <ufs/dir.h> 1716264Smckusick #include <strings.h> 1816264Smckusick #include "fsck.h" 1916264Smckusick 2016264Smckusick int pass2check(); 2116264Smckusick 2216264Smckusick pass2() 2316264Smckusick { 2416264Smckusick register DINODE *dp; 2516264Smckusick struct inodesc rootdesc; 2616264Smckusick 2716264Smckusick bzero((char *)&rootdesc, sizeof(struct inodesc)); 2816264Smckusick rootdesc.id_type = ADDR; 2916264Smckusick rootdesc.id_func = pass2check; 3016264Smckusick rootdesc.id_number = ROOTINO; 3116264Smckusick pathp = pathname; 3216264Smckusick switch (statemap[ROOTINO]) { 3316264Smckusick 3416264Smckusick case USTATE: 3517966Smckusick pfatal("ROOT INODE UNALLOCATED"); 3617966Smckusick if (reply("ALLOCATE") == 0) 3717966Smckusick errexit(""); 3836827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 3917966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 4017966Smckusick descend(&rootdesc, ROOTINO); 4117966Smckusick break; 4216264Smckusick 4317966Smckusick case DCLEAR: 4417966Smckusick pfatal("DUPS/BAD IN ROOT INODE"); 4517966Smckusick if (reply("REALLOCATE")) { 4617966Smckusick freeino(ROOTINO); 4736827Smckusick if (allocdir(ROOTINO, ROOTINO, 0755) != ROOTINO) 4817966Smckusick errexit("CANNOT ALLOCATE ROOT INODE\n"); 4917966Smckusick descend(&rootdesc, ROOTINO); 5017966Smckusick break; 5117966Smckusick } 5217966Smckusick if (reply("CONTINUE") == 0) 5317966Smckusick errexit(""); 5417966Smckusick statemap[ROOTINO] = DSTATE; 5517966Smckusick descend(&rootdesc, ROOTINO); 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 descend(&rootdesc, ROOTINO); 6617966Smckusick break; 6717966Smckusick } 6817943Smckusick if (reply("FIX") == 0) 6916264Smckusick errexit(""); 7017943Smckusick dp = ginode(ROOTINO); 7116264Smckusick dp->di_mode &= ~IFMT; 7216264Smckusick dp->di_mode |= IFDIR; 7316264Smckusick inodirty(); 7416264Smckusick statemap[ROOTINO] = DSTATE; 7516264Smckusick /* fall into ... */ 7616264Smckusick 7716264Smckusick case DSTATE: 7816264Smckusick descend(&rootdesc, ROOTINO); 7916264Smckusick break; 8016264Smckusick 8117937Smckusick default: 8217937Smckusick errexit("BAD STATE %d FOR ROOT INODE", statemap[ROOTINO]); 8316264Smckusick } 8416264Smckusick } 8516264Smckusick 8616264Smckusick pass2check(idesc) 8716264Smckusick struct inodesc *idesc; 8816264Smckusick { 8916264Smckusick register DIRECT *dirp = idesc->id_dirp; 9016264Smckusick char *curpathloc; 9116264Smckusick int n, entrysize, ret = 0; 9216264Smckusick DINODE *dp; 9316264Smckusick DIRECT proto; 9417991Smckusick char namebuf[BUFSIZ]; 9516264Smckusick 9616264Smckusick /* 9716264Smckusick * check for "." 9816264Smckusick */ 9916264Smckusick if (idesc->id_entryno != 0) 10016264Smckusick goto chk1; 10116264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") == 0) { 10216264Smckusick if (dirp->d_ino != idesc->id_number) { 10316264Smckusick direrr(idesc->id_number, "BAD INODE NUMBER FOR '.'"); 10416264Smckusick dirp->d_ino = idesc->id_number; 10516264Smckusick if (reply("FIX") == 1) 10616264Smckusick ret |= ALTERED; 10716264Smckusick } 10816264Smckusick goto chk1; 10916264Smckusick } 11016264Smckusick direrr(idesc->id_number, "MISSING '.'"); 11116264Smckusick proto.d_ino = idesc->id_number; 11216264Smckusick proto.d_namlen = 1; 11316264Smckusick (void)strcpy(proto.d_name, "."); 11416264Smckusick entrysize = DIRSIZ(&proto); 11516264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") != 0) { 11616264Smckusick pfatal("CANNOT FIX, FIRST ENTRY IN DIRECTORY CONTAINS %s\n", 11716264Smckusick dirp->d_name); 11816264Smckusick } else if (dirp->d_reclen < entrysize) { 11916264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '.'\n"); 12016264Smckusick } else if (dirp->d_reclen < 2 * entrysize) { 12116264Smckusick proto.d_reclen = dirp->d_reclen; 12216264Smckusick bcopy((char *)&proto, (char *)dirp, entrysize); 12316264Smckusick if (reply("FIX") == 1) 12416264Smckusick ret |= ALTERED; 12516264Smckusick } else { 12616264Smckusick n = dirp->d_reclen - entrysize; 12716264Smckusick proto.d_reclen = entrysize; 12816264Smckusick bcopy((char *)&proto, (char *)dirp, entrysize); 12916264Smckusick idesc->id_entryno++; 13016264Smckusick lncntp[dirp->d_ino]--; 13116264Smckusick dirp = (DIRECT *)((char *)(dirp) + entrysize); 13216264Smckusick bzero((char *)dirp, n); 13316264Smckusick dirp->d_reclen = n; 13416264Smckusick if (reply("FIX") == 1) 13516264Smckusick ret |= ALTERED; 13616264Smckusick } 13716264Smckusick chk1: 13816264Smckusick if (idesc->id_entryno > 1) 13916264Smckusick goto chk2; 14016264Smckusick proto.d_ino = idesc->id_parent; 14116264Smckusick proto.d_namlen = 2; 14216264Smckusick (void)strcpy(proto.d_name, ".."); 14316264Smckusick entrysize = DIRSIZ(&proto); 14416264Smckusick if (idesc->id_entryno == 0) { 14516264Smckusick n = DIRSIZ(dirp); 14616264Smckusick if (dirp->d_reclen < n + entrysize) 14716264Smckusick goto chk2; 14816264Smckusick proto.d_reclen = dirp->d_reclen - n; 14916264Smckusick dirp->d_reclen = n; 15016264Smckusick idesc->id_entryno++; 15116264Smckusick lncntp[dirp->d_ino]--; 15216264Smckusick dirp = (DIRECT *)((char *)(dirp) + n); 15316264Smckusick bzero((char *)dirp, n); 15416264Smckusick dirp->d_reclen = n; 15516264Smckusick } 15616264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, "..") == 0) { 15716264Smckusick if (dirp->d_ino != idesc->id_parent) { 15816264Smckusick direrr(idesc->id_number, "BAD INODE NUMBER FOR '..'"); 15916264Smckusick dirp->d_ino = idesc->id_parent; 16016264Smckusick if (reply("FIX") == 1) 16116264Smckusick ret |= ALTERED; 16216264Smckusick } 16316264Smckusick goto chk2; 16416264Smckusick } 16516264Smckusick direrr(idesc->id_number, "MISSING '..'"); 16616264Smckusick if (dirp->d_ino != 0 && strcmp(dirp->d_name, ".") != 0) { 16716264Smckusick pfatal("CANNOT FIX, SECOND ENTRY IN DIRECTORY CONTAINS %s\n", 16816264Smckusick dirp->d_name); 16916264Smckusick } else if (dirp->d_reclen < entrysize) { 17016264Smckusick pfatal("CANNOT FIX, INSUFFICIENT SPACE TO ADD '..'\n"); 17116264Smckusick } else { 17216264Smckusick proto.d_reclen = dirp->d_reclen; 17316264Smckusick bcopy((char *)&proto, (char *)dirp, entrysize); 17416264Smckusick if (reply("FIX") == 1) 17516264Smckusick ret |= ALTERED; 17616264Smckusick } 17716264Smckusick chk2: 17816264Smckusick if (dirp->d_ino == 0) 17916264Smckusick return (ret|KEEPON); 18016264Smckusick if (dirp->d_namlen <= 2 && 18116264Smckusick dirp->d_name[0] == '.' && 18216264Smckusick idesc->id_entryno >= 2) { 18316264Smckusick if (dirp->d_namlen == 1) { 18416264Smckusick direrr(idesc->id_number, "EXTRA '.' ENTRY"); 18516264Smckusick dirp->d_ino = 0; 18616264Smckusick if (reply("FIX") == 1) 18716264Smckusick ret |= ALTERED; 18816264Smckusick return (KEEPON | ret); 18916264Smckusick } 19016264Smckusick if (dirp->d_name[1] == '.') { 19116264Smckusick direrr(idesc->id_number, "EXTRA '..' ENTRY"); 19216264Smckusick dirp->d_ino = 0; 19316264Smckusick if (reply("FIX") == 1) 19416264Smckusick ret |= ALTERED; 19516264Smckusick return (KEEPON | ret); 19616264Smckusick } 19716264Smckusick } 19816264Smckusick curpathloc = pathp; 19916264Smckusick *pathp++ = '/'; 20016264Smckusick if (pathp + dirp->d_namlen >= endpathname) { 20116264Smckusick *pathp = '\0'; 20216264Smckusick errexit("NAME TOO LONG %s%s\n", pathname, dirp->d_name); 20316264Smckusick } 20416264Smckusick bcopy(dirp->d_name, pathp, dirp->d_namlen + 1); 20516264Smckusick pathp += dirp->d_namlen; 20616264Smckusick idesc->id_entryno++; 20716264Smckusick n = 0; 20816264Smckusick if (dirp->d_ino > imax || dirp->d_ino <= 0) { 20916264Smckusick direrr(dirp->d_ino, "I OUT OF RANGE"); 21016264Smckusick n = reply("REMOVE"); 21116264Smckusick } else { 21216264Smckusick again: 21316264Smckusick switch (statemap[dirp->d_ino]) { 21416264Smckusick case USTATE: 21516264Smckusick direrr(dirp->d_ino, "UNALLOCATED"); 21616264Smckusick n = reply("REMOVE"); 21716264Smckusick break; 21816264Smckusick 21917937Smckusick case DCLEAR: 22017937Smckusick case FCLEAR: 22116264Smckusick direrr(dirp->d_ino, "DUP/BAD"); 22216264Smckusick if ((n = reply("REMOVE")) == 1) 22316264Smckusick break; 22417943Smckusick dp = ginode(dirp->d_ino); 22517931Smckusick statemap[dirp->d_ino] = DIRCT(dp) ? DSTATE : FSTATE; 22630599Smckusick lncntp[dirp->d_ino] = dp->di_nlink; 22716264Smckusick goto again; 22816264Smckusick 22917937Smckusick case DFOUND: 23017991Smckusick if (idesc->id_entryno > 2) { 23117991Smckusick getpathname(namebuf, dirp->d_ino, dirp->d_ino); 23217991Smckusick pwarn("%s %s %s\n", pathname, 23317991Smckusick "IS AN EXTRANEOUS HARD LINK TO DIRECTORY", 23417991Smckusick namebuf); 23517991Smckusick if (preen) 23617991Smckusick printf(" (IGNORED)\n"); 23717991Smckusick else if ((n = reply("REMOVE")) == 1) 23817991Smckusick break; 23917991Smckusick } 24017937Smckusick /* fall through */ 24117937Smckusick 24216264Smckusick case FSTATE: 24316264Smckusick lncntp[dirp->d_ino]--; 24416264Smckusick break; 24516264Smckusick 24616264Smckusick case DSTATE: 24716264Smckusick descend(idesc, dirp->d_ino); 24817937Smckusick if (statemap[dirp->d_ino] == DFOUND) { 24916264Smckusick lncntp[dirp->d_ino]--; 25017937Smckusick } else if (statemap[dirp->d_ino] == DCLEAR) { 25116264Smckusick dirp->d_ino = 0; 25216264Smckusick ret |= ALTERED; 25317937Smckusick } else 25417937Smckusick errexit("BAD RETURN STATE %d FROM DESCEND", 25517937Smckusick statemap[dirp->d_ino]); 25616264Smckusick break; 25726480Smckusick 25826480Smckusick default: 25926480Smckusick errexit("BAD STATE %d FOR INODE I=%d", 26026480Smckusick statemap[dirp->d_ino], dirp->d_ino); 26116264Smckusick } 26216264Smckusick } 26316264Smckusick pathp = curpathloc; 26416264Smckusick *pathp = '\0'; 26516264Smckusick if (n == 0) 26616264Smckusick return (ret|KEEPON); 26716264Smckusick dirp->d_ino = 0; 26816264Smckusick return (ret|KEEPON|ALTERED); 26916264Smckusick } 270