xref: /csrg-svn/sbin/fsck/preen.c (revision 68992)
141098Smckusick /*
261492Sbostic  * Copyright (c) 1990, 1993
361492Sbostic  *	The Regents of the University of California.  All rights reserved.
441098Smckusick  *
542702Sbostic  * %sccs.include.redist.c%
641098Smckusick  */
741098Smckusick 
841098Smckusick #ifndef lint
9*68992Sbostic static char sccsid[] = "@(#)preen.c	8.5 (Berkeley) 04/28/95";
1041098Smckusick #endif /* not lint */
1141098Smckusick 
1241098Smckusick #include <sys/param.h>
1341098Smckusick #include <sys/stat.h>
1441098Smckusick #include <sys/wait.h>
1568908Smckusick 
1668908Smckusick #include <ufs/ufs/dinode.h>
1768908Smckusick 
1868908Smckusick #include <ctype.h>
1941098Smckusick #include <fstab.h>
2042041Sbostic #include <string.h>
2141098Smckusick 
2268908Smckusick #include "fsck.h"
2341098Smckusick 
2441098Smckusick struct part {
2541098Smckusick 	struct	part *next;		/* forward link of partitions on disk */
2641098Smckusick 	char	*name;			/* device name */
2741098Smckusick 	char	*fsname;		/* mounted filesystem name */
2841098Smckusick 	long	auxdata;		/* auxillary data for application */
2941098Smckusick } *badlist, **badnext = &badlist;
3041098Smckusick 
3141098Smckusick struct disk {
3241098Smckusick 	char	*name;			/* disk base name */
3341098Smckusick 	struct	disk *next;		/* forward link for list of disks */
3441098Smckusick 	struct	part *part;		/* head of list of partitions on disk */
3541098Smckusick 	int	pid;			/* If != 0, pid of proc working on */
3641098Smckusick } *disks;
3741098Smckusick 
3841416Smckusick int	nrun, ndisks;
3941416Smckusick char	hotroot;
4041098Smckusick 
4168908Smckusick static void addpart __P((char *name, char *fsname, long auxdata));
4268908Smckusick static struct disk *finddisk __P((char *name));
4368908Smckusick static char *rawname __P((char *name));
4468908Smckusick static int startdisk __P((struct disk *dk,
4568908Smckusick 		int (*checkit)(char *, char *, long, int)));
4668908Smckusick static char *unrawname __P((char *name));
4768908Smckusick 
4868908Smckusick int
checkfstab(preen,maxrun,docheck,chkit)4941098Smckusick checkfstab(preen, maxrun, docheck, chkit)
5068908Smckusick 	int preen;
5168908Smckusick 	int maxrun;
5268908Smckusick 	int (*docheck)(struct fstab *);
5368908Smckusick 	int (*chkit)(char *, char *, long, int);
5441098Smckusick {
5541098Smckusick 	register struct fstab *fsp;
5641098Smckusick 	register struct disk *dk, *nextdisk;
5741098Smckusick 	register struct part *pt;
5841098Smckusick 	int ret, pid, retcode, passno, sumstatus, status;
5941098Smckusick 	long auxdata;
6041098Smckusick 	char *name;
6141098Smckusick 
6241098Smckusick 	sumstatus = 0;
6341098Smckusick 	for (passno = 1; passno <= 2; passno++) {
6441098Smckusick 		if (setfsent() == 0) {
6541098Smckusick 			fprintf(stderr, "Can't open checklist file: %s\n",
6641098Smckusick 			    _PATH_FSTAB);
6741098Smckusick 			return (8);
6841098Smckusick 		}
6941098Smckusick 		while ((fsp = getfsent()) != 0) {
7041098Smckusick 			if ((auxdata = (*docheck)(fsp)) == 0)
7141098Smckusick 				continue;
7268908Smckusick 			if (preen == 0 ||
7368908Smckusick 			    (passno == 1 && fsp->fs_passno == 1)) {
7468908Smckusick 				if ((name = blockcheck(fsp->fs_spec)) != 0) {
7568908Smckusick 					if ((sumstatus = (*chkit)(name,
7668908Smckusick 					    fsp->fs_file, auxdata, 0)) != 0)
7741098Smckusick 						return (sumstatus);
7841098Smckusick 				} else if (preen)
7941098Smckusick 					return (8);
8041098Smckusick 			} else if (passno == 2 && fsp->fs_passno > 1) {
8141098Smckusick 				if ((name = blockcheck(fsp->fs_spec)) == NULL) {
8241098Smckusick 					fprintf(stderr, "BAD DISK NAME %s\n",
8341098Smckusick 						fsp->fs_spec);
8441098Smckusick 					sumstatus |= 8;
8541098Smckusick 					continue;
8641098Smckusick 				}
8741098Smckusick 				addpart(name, fsp->fs_file, auxdata);
8841098Smckusick 			}
8941098Smckusick 		}
9041098Smckusick 		if (preen == 0)
9141098Smckusick 			return (0);
9241098Smckusick 	}
9341098Smckusick 	if (preen) {
9441098Smckusick 		if (maxrun == 0)
9541098Smckusick 			maxrun = ndisks;
9641098Smckusick 		if (maxrun > ndisks)
9741098Smckusick 			maxrun = ndisks;
9841098Smckusick 		nextdisk = disks;
9941098Smckusick 		for (passno = 0; passno < maxrun; ++passno) {
10068908Smckusick 			while ((ret = startdisk(nextdisk, chkit)) && nrun > 0)
10141098Smckusick 				sleep(10);
10241098Smckusick 			if (ret)
10341098Smckusick 				return (ret);
10441098Smckusick 			nextdisk = nextdisk->next;
10541098Smckusick 		}
10641098Smckusick 		while ((pid = wait(&status)) != -1) {
10741098Smckusick 			for (dk = disks; dk; dk = dk->next)
10841098Smckusick 				if (dk->pid == pid)
10941098Smckusick 					break;
11041098Smckusick 			if (dk == 0) {
11141098Smckusick 				printf("Unknown pid %d\n", pid);
11241098Smckusick 				continue;
11341098Smckusick 			}
11441098Smckusick 			if (WIFEXITED(status))
11541098Smckusick 				retcode = WEXITSTATUS(status);
11641098Smckusick 			else
11741098Smckusick 				retcode = 0;
11841098Smckusick 			if (WIFSIGNALED(status)) {
11941098Smckusick 				printf("%s (%s): EXITED WITH SIGNAL %d\n",
12041098Smckusick 					dk->part->name, dk->part->fsname,
12141098Smckusick 					WTERMSIG(status));
12241098Smckusick 				retcode = 8;
12341098Smckusick 			}
12441098Smckusick 			if (retcode != 0) {
12541098Smckusick 				sumstatus |= retcode;
12641098Smckusick 				*badnext = dk->part;
12741098Smckusick 				badnext = &dk->part->next;
12841098Smckusick 				dk->part = dk->part->next;
12941098Smckusick 				*badnext = NULL;
13041098Smckusick 			} else
13141098Smckusick 				dk->part = dk->part->next;
13241098Smckusick 			dk->pid = 0;
13341098Smckusick 			nrun--;
13441098Smckusick 			if (dk->part == NULL)
13541098Smckusick 				ndisks--;
13641098Smckusick 
13741098Smckusick 			if (nextdisk == NULL) {
13841098Smckusick 				if (dk->part) {
13968908Smckusick 					while ((ret = startdisk(dk, chkit)) &&
14041098Smckusick 					    nrun > 0)
14141098Smckusick 						sleep(10);
14241098Smckusick 					if (ret)
14341098Smckusick 						return (ret);
14441098Smckusick 				}
14541098Smckusick 			} else if (nrun < maxrun && nrun < ndisks) {
14641098Smckusick 				for ( ;; ) {
14741098Smckusick 					if ((nextdisk = nextdisk->next) == NULL)
14841098Smckusick 						nextdisk = disks;
14941098Smckusick 					if (nextdisk->part != NULL &&
15041098Smckusick 					    nextdisk->pid == 0)
15141098Smckusick 						break;
15241098Smckusick 				}
15368908Smckusick 				while ((ret = startdisk(nextdisk, chkit)) &&
15441098Smckusick 				    nrun > 0)
15541098Smckusick 					sleep(10);
15641098Smckusick 				if (ret)
15741098Smckusick 					return (ret);
15841098Smckusick 			}
15941098Smckusick 		}
16041098Smckusick 	}
16141098Smckusick 	if (sumstatus) {
16241098Smckusick 		if (badlist == 0)
16341098Smckusick 			return (sumstatus);
16441098Smckusick 		fprintf(stderr, "THE FOLLOWING FILE SYSTEM%s HAD AN %s\n\t",
16541098Smckusick 			badlist->next ? "S" : "", "UNEXPECTED INCONSISTENCY:");
16641098Smckusick 		for (pt = badlist; pt; pt = pt->next)
16741098Smckusick 			fprintf(stderr, "%s (%s)%s", pt->name, pt->fsname,
16841098Smckusick 			    pt->next ? ", " : "\n");
16941098Smckusick 		return (sumstatus);
17041098Smckusick 	}
17141098Smckusick 	(void)endfsent();
17241098Smckusick 	return (0);
17341098Smckusick }
17441098Smckusick 
17568908Smckusick static struct disk *
finddisk(name)17641098Smckusick finddisk(name)
17741098Smckusick 	char *name;
17841098Smckusick {
17941098Smckusick 	register struct disk *dk, **dkp;
18041098Smckusick 	register char *p;
18144934Smckusick 	size_t len;
18241098Smckusick 
18368908Smckusick 	for (len = strlen(name), p = name + len - 1; p >= name; --p)
18441098Smckusick 		if (isdigit(*p)) {
18541098Smckusick 			len = p - name + 1;
18641098Smckusick 			break;
18741098Smckusick 		}
18841098Smckusick 
18941098Smckusick 	for (dk = disks, dkp = &disks; dk; dkp = &dk->next, dk = dk->next) {
19041098Smckusick 		if (strncmp(dk->name, name, len) == 0 &&
19141098Smckusick 		    dk->name[len] == 0)
19241098Smckusick 			return (dk);
19341098Smckusick 	}
19441098Smckusick 	if ((*dkp = (struct disk *)malloc(sizeof(struct disk))) == NULL) {
19541098Smckusick 		fprintf(stderr, "out of memory");
19641098Smckusick 		exit (8);
19741098Smckusick 	}
19841098Smckusick 	dk = *dkp;
19944934Smckusick 	if ((dk->name = malloc(len + 1)) == NULL) {
20041098Smckusick 		fprintf(stderr, "out of memory");
20141098Smckusick 		exit (8);
20241098Smckusick 	}
20344934Smckusick 	(void)strncpy(dk->name, name, len);
20441098Smckusick 	dk->name[len] = '\0';
20541098Smckusick 	dk->part = NULL;
20641098Smckusick 	dk->next = NULL;
20741098Smckusick 	dk->pid = 0;
20841098Smckusick 	ndisks++;
20941098Smckusick 	return (dk);
21041098Smckusick }
21141098Smckusick 
21268908Smckusick static void
addpart(name,fsname,auxdata)21341098Smckusick addpart(name, fsname, auxdata)
21441098Smckusick 	char *name, *fsname;
21541098Smckusick 	long auxdata;
21641098Smckusick {
21741098Smckusick 	struct disk *dk = finddisk(name);
21841098Smckusick 	register struct part *pt, **ppt = &dk->part;
21941098Smckusick 
22041098Smckusick 	for (pt = dk->part; pt; ppt = &pt->next, pt = pt->next)
22141098Smckusick 		if (strcmp(pt->name, name) == 0) {
22241098Smckusick 			printf("%s in fstab more than once!\n", name);
22341098Smckusick 			return;
22441098Smckusick 		}
22541098Smckusick 	if ((*ppt = (struct part *)malloc(sizeof(struct part))) == NULL) {
22641098Smckusick 		fprintf(stderr, "out of memory");
22741098Smckusick 		exit (8);
22841098Smckusick 	}
22941098Smckusick 	pt = *ppt;
23044934Smckusick 	if ((pt->name = malloc(strlen(name) + 1)) == NULL) {
23141098Smckusick 		fprintf(stderr, "out of memory");
23241098Smckusick 		exit (8);
23341098Smckusick 	}
23444934Smckusick 	(void)strcpy(pt->name, name);
23544934Smckusick 	if ((pt->fsname = malloc(strlen(fsname) + 1)) == NULL) {
23641098Smckusick 		fprintf(stderr, "out of memory");
23741098Smckusick 		exit (8);
23841098Smckusick 	}
23944934Smckusick 	(void)strcpy(pt->fsname, fsname);
24041098Smckusick 	pt->next = NULL;
24141098Smckusick 	pt->auxdata = auxdata;
24241098Smckusick }
24341098Smckusick 
24468908Smckusick static int
startdisk(dk,checkit)24541098Smckusick startdisk(dk, checkit)
24641098Smckusick 	register struct disk *dk;
24768908Smckusick 	int (*checkit)(char *, char *, long, int);
24841098Smckusick {
24941098Smckusick 	register struct part *pt = dk->part;
25041098Smckusick 
25141098Smckusick 	dk->pid = fork();
25241098Smckusick 	if (dk->pid < 0) {
25341098Smckusick 		perror("fork");
25441098Smckusick 		return (8);
25541098Smckusick 	}
25641098Smckusick 	if (dk->pid == 0)
25745027Skarels 		exit((*checkit)(pt->name, pt->fsname, pt->auxdata, 1));
25841098Smckusick 	nrun++;
25941098Smckusick 	return (0);
26041098Smckusick }
26141098Smckusick 
26241098Smckusick char *
blockcheck(origname)26368058Smckusick blockcheck(origname)
26468058Smckusick 	char *origname;
26541098Smckusick {
26641098Smckusick 	struct stat stslash, stblock, stchar;
26768058Smckusick 	char *newname, *raw;
26841098Smckusick 	int retried = 0;
26941098Smckusick 
27041098Smckusick 	hotroot = 0;
27141098Smckusick 	if (stat("/", &stslash) < 0) {
27241098Smckusick 		perror("/");
27341098Smckusick 		printf("Can't stat root\n");
27468058Smckusick 		return (origname);
27541098Smckusick 	}
27668058Smckusick 	newname = origname;
27741098Smckusick retry:
27868058Smckusick 	if (stat(newname, &stblock) < 0) {
27968058Smckusick 		perror(newname);
28068058Smckusick 		printf("Can't stat %s\n", newname);
28168058Smckusick 		return (origname);
28241098Smckusick 	}
28341098Smckusick 	if ((stblock.st_mode & S_IFMT) == S_IFBLK) {
28441098Smckusick 		if (stslash.st_dev == stblock.st_rdev)
28541098Smckusick 			hotroot++;
28668058Smckusick 		raw = rawname(newname);
28741098Smckusick 		if (stat(raw, &stchar) < 0) {
28841098Smckusick 			perror(raw);
28941098Smckusick 			printf("Can't stat %s\n", raw);
29068058Smckusick 			return (origname);
29141098Smckusick 		}
29241098Smckusick 		if ((stchar.st_mode & S_IFMT) == S_IFCHR) {
29341098Smckusick 			return (raw);
29441098Smckusick 		} else {
29541098Smckusick 			printf("%s is not a character device\n", raw);
29668058Smckusick 			return (origname);
29741098Smckusick 		}
29841098Smckusick 	} else if ((stblock.st_mode & S_IFMT) == S_IFCHR && !retried) {
29968058Smckusick 		newname = unrawname(newname);
30041098Smckusick 		retried++;
30141098Smckusick 		goto retry;
30241098Smckusick 	}
30368018Smckusick 	/*
30468018Smckusick 	 * Not a block or character device, just return name and
30568018Smckusick 	 * let the user decide whether to use it.
30668018Smckusick 	 */
30768058Smckusick 	return (origname);
30841098Smckusick }
30941098Smckusick 
31068908Smckusick static char *
unrawname(name)31141098Smckusick unrawname(name)
31241098Smckusick 	char *name;
31341098Smckusick {
31441098Smckusick 	char *dp;
31541098Smckusick 	struct stat stb;
31641098Smckusick 
317*68992Sbostic 	if ((dp = strrchr(name, '/')) == 0)
31841098Smckusick 		return (name);
31941098Smckusick 	if (stat(name, &stb) < 0)
32041098Smckusick 		return (name);
32141098Smckusick 	if ((stb.st_mode & S_IFMT) != S_IFCHR)
32241098Smckusick 		return (name);
32347578Smckusick 	if (dp[1] != 'r')
32441098Smckusick 		return (name);
32547578Smckusick 	(void)strcpy(&dp[1], &dp[2]);
32641098Smckusick 	return (name);
32741098Smckusick }
32841098Smckusick 
32968908Smckusick static char *
rawname(name)33041098Smckusick rawname(name)
33141098Smckusick 	char *name;
33241098Smckusick {
33341098Smckusick 	static char rawbuf[32];
33441098Smckusick 	char *dp;
33541098Smckusick 
336*68992Sbostic 	if ((dp = strrchr(name, '/')) == 0)
33741098Smckusick 		return (0);
33841098Smckusick 	*dp = 0;
33941098Smckusick 	(void)strcpy(rawbuf, name);
34041098Smckusick 	*dp = '/';
34141098Smckusick 	(void)strcat(rawbuf, "/r");
34247578Smckusick 	(void)strcat(rawbuf, &dp[1]);
34341098Smckusick 	return (rawbuf);
34441098Smckusick }
345