xref: /titanic_44/usr/src/cmd/fs.d/cachefs/fsck/fsck.c (revision 549ec3fff108310966327d1dc9004551b63210b7)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2004 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 /*	Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */
28 /*	    All Rights Reserved */
29 
30 /*
31  * Portions of this source code were derived from Berkeley 4.3 BSD
32  * under license from the Regents of the University of California.
33  */
34 
35 #pragma ident	"%Z%%M%	%I%	%E% SMI"
36 
37 /*
38  *
39  *			fsck.c
40  *
41  * Cachefs fsck program.
42  */
43 
44 #include <locale.h>
45 #include <stdio.h>
46 #include <stdlib.h>
47 #include <string.h>
48 #include <assert.h>
49 #include <stdarg.h>
50 #include <unistd.h>
51 #include <limits.h>
52 #include <errno.h>
53 #include <wait.h>
54 #include <ctype.h>
55 #include <fcntl.h>
56 #include <ftw.h>
57 #include <dirent.h>
58 #include <search.h>
59 #include <sys/types.h>
60 #include <sys/uio.h>
61 #include <sys/param.h>
62 #include <sys/stat.h>
63 #include <sys/fcntl.h>
64 #include <sys/mount.h>
65 #include <sys/mntent.h>
66 #include <sys/mnttab.h>
67 #include <sys/mman.h>
68 #include <sys/fs/cachefs_fs.h>
69 #include <syslog.h>
70 #include "../common/subr.h"
71 #include "res.h"
72 
73 char *cfs_opts[] = {
74 #define		CFSOPT_PREEN		0
75 		"preen",
76 #define		CFSOPT_NOCLEAN		1
77 		"noclean",
78 #define		CFSOPT_VERBOSE		2
79 		"verbose",
80 #define		CFSOPT_NONOCLEAN	3
81 		"nonoclean",
82 
83 		NULL
84 };
85 
86 extern int dlog_ck(char *dir_path, ino64_t *maxlocalfilenop);
87 
88 /* forward references */
89 void usage(char *msgp);
90 void pr_err(char *fmt, ...);
91 int cfs_check(char *cachedirp, int noclean, int mflag, int verbose,
92     int nonoclean);
93 int cache_label_file(char *cachedirp, struct cache_label *clabelp);
94 int cache_permissions(char *cachedirp);
95 int cache_check_dir(char *cachedirp, char *namep);
96 int process_fsdir(char *cachedirp, char *namep, res *resp, int verbose);
97 int process_fsinfo(char *namep, ino64_t maxlocalfileno,
98     cachefs_fsinfo_t *fsinfop, int verbose);
99 int process_fsgroup(char *dirp, char *namep, res *resp, ino64_t base,
100     int fgsize, ino64_t fsid, int local, int verbose);
101 int tree_remove(const char *namep, const struct stat64 *statp, int type,
102     struct FTW *ftwp);
103 int cache_upgrade(char *cachedirp, int lockid);
104 int file_remove(const char *namep, const struct stat64 *statp, int verbose);
105 void cache_backmnt_cleanup(char *cachedirp, char *backmnt_namep);
106 
107 #define	FLAGS_FTW (FTW_PHYS | FTW_MOUNT | FTW_DEPTH)
108 
109 static int S_verbose = 0;
110 static char S_lostfound[MAXPATHLEN];
111 static int S_move_lostfound = 0;
112 
113 /*
114  *
115  *			main
116  *
117  * Description:
118  *	Main routine for the cachefs fsck program.
119  * Arguments:
120  *	argc	number of command line arguments
121  *	argv	list of command line arguments
122  * Returns:
123  *	Returns:
124  *		 0	file system is okay and does not need checking
125  *		 1	problem unrelated to the file system
126  *		32	file system is unmounted and needs checking  (fsck
127  *			-m only)
128  *		33	file system is already mounted
129  *		34	cannot stat device
130  *		36	uncorrectable errors detected - terminate normally
131  *		37	a signal was caught during processing
132  *		39	uncorrectable errors detected - terminate  immediately
133  *		40	for root mounted fs, same as 0
134  * Preconditions:
135  */
136 
137 int
138 main(int argc, char **argv)
139 {
140 	int xx;
141 	int c;
142 	char *optionp;
143 	char *valuep;
144 	int mflag;
145 	int noclean;
146 	char *cachedirp;
147 	int lockid;
148 	int verbose;
149 	int nonoclean;
150 
151 	(void) setlocale(LC_ALL, "");
152 #if !defined(TEXT_DOMAIN)
153 #define	TEXT_DOMAIN	"SYS_TEST"
154 #endif
155 	(void) textdomain(TEXT_DOMAIN);
156 
157 	/* verify root running command */
158 	if (getuid() != 0) {
159 		fprintf(stderr, gettext(
160 			"fsck -F cachefs: must be run by root\n"));
161 		return (1);
162 	}
163 
164 	/* process command line options */
165 	optionp = NULL;
166 	mflag = 0;
167 	noclean = 0;
168 	verbose = 0;
169 	nonoclean = 0;
170 	while ((c = getopt(argc, argv, "mnNo:yY")) != EOF) {
171 		switch (c) {
172 		case 'm':	/* check but do not repair */
173 			mflag = 1;
174 			break;
175 
176 		case 'n':	/* answer no to questions */
177 		case 'N':
178 			/* ignored */
179 			break;
180 
181 		case 'o':
182 			optionp = optarg;
183 			while (*optionp) {
184 				xx = getsubopt(&optionp, cfs_opts, &valuep);
185 				switch (xx) {
186 				case CFSOPT_PREEN:
187 					/* preen is the default mode */
188 					break;
189 				case CFSOPT_NOCLEAN:
190 					noclean = 1;
191 					break;
192 				case CFSOPT_VERBOSE:
193 					verbose++;
194 					S_verbose++;
195 					break;
196 				case CFSOPT_NONOCLEAN:
197 					nonoclean = 1;
198 					break;
199 				default:
200 				case -1:
201 					pr_err(gettext("unknown option %s"),
202 					    valuep);
203 					return (1);
204 				}
205 			}
206 			break;
207 
208 		case 'y':	/* answer yes to questions */
209 		case 'Y':
210 			/* ignored, this is the default */
211 			break;
212 
213 		default:
214 			usage("invalid option");
215 			return (1);
216 		}
217 	}
218 
219 	/* verify fsck device is specified */
220 	if (argc - optind < 1) {
221 		usage(gettext("must specify cache directory"));
222 		return (1);
223 	}
224 
225 	/* save cache directory */
226 	cachedirp = argv[argc - 1];
227 
228 	/* ensure cache directory exists */
229 	if (access(cachedirp, F_OK) != 0) {
230 		pr_err(gettext("Cache directory %s does not exist."),
231 		    cachedirp);
232 		return (39);
233 	}
234 
235 	/* lock the cache directory non-shared */
236 	lockid = cachefs_dir_lock(cachedirp, 0);
237 	if (lockid == -1) {
238 		/* exit if could not get the lock */
239 		return (1);
240 	}
241 
242 	/* is the cache directory in use */
243 	if (cachefs_inuse(cachedirp)) {
244 		if (noclean) {
245 			pr_err(gettext("Cache directory %s is in use."),
246 			    cachedirp);
247 			xx = 33;
248 		} else {
249 			/* assume if in use that it is clean */
250 			xx = 0;
251 		}
252 		cachefs_dir_unlock(lockid);
253 		return (xx);
254 	}
255 
256 	xx = cache_upgrade(cachedirp, lockid);
257 	if (xx != 0) {
258 		/* check the file system */
259 		xx = cfs_check(cachedirp, noclean, mflag, verbose, nonoclean);
260 	}
261 
262 	/* unlock the cache directory */
263 	cachefs_dir_unlock(lockid);
264 
265 	/* inform if files moved to lost+found */
266 	if (S_move_lostfound) {
267 		pr_err(gettext("Files recovered to %s"), S_lostfound);
268 	}
269 
270 	/* return the status of the file system checking */
271 	return (xx);
272 }
273 
274 /*
275  *
276  *			usage
277  *
278  * Description:
279  *	Prints a short usage message.
280  * Arguments:
281  *	msgp	message to include with the usage message
282  * Returns:
283  * Preconditions:
284  */
285 
286 void
287 usage(char *msgp)
288 {
289 	if (msgp) {
290 		pr_err("%s", msgp);
291 	}
292 
293 	(void) fprintf(stderr,
294 	    gettext("Usage: fsck -F cachefs [ -o specific_options ] [ -m ] "
295 	    "cachedir\n"));
296 }
297 
298 /*
299  *
300  *			pr_err
301  *
302  * Description:
303  *	Prints an error message to stderr.
304  * Arguments:
305  *	fmt	printf style format
306  *	...	arguments for fmt
307  * Returns:
308  * Preconditions:
309  *	precond(fmt)
310  */
311 
312 void
313 pr_err(char *fmt, ...)
314 {
315 	va_list ap;
316 
317 	va_start(ap, fmt);
318 	(void) fprintf(stderr, gettext("fsck -F cachefs: "));
319 	(void) vfprintf(stderr, fmt, ap);
320 	(void) fprintf(stderr, "\n");
321 	va_end(ap);
322 }
323 
324 /*
325  *
326  *			cache_upgrade
327  *
328  * Description:
329  *
330  *	See if the current cache is out of date.  If it is, do
331  *	whatever magic is necessary to upgrade it.  All such magic
332  *	should be encapsulated here!
333  *
334  * Arguments:
335  *
336  *	cachedirp	name of the cache directory to check
337  *
338  * Returns:
339  *	Returns:
340  *		 0	cache was upgraded and shouldn't be checked
341  *		 1	problem unrelated to the file system
342  *		36	uncorrectable errors detected - terminate normally
343  *		39	uncorrectable errors detected - terminate  immediately
344  *		50	cache was already up-to-date (maybe we should fsck it)
345  *		51	cache was upgraded (but you should do fsck)
346  * Preconditions:
347  *	precond(cachedirp)
348  */
349 
350 int
351 cache_upgrade(char *cachedirp, int lockid)
352 {
353 #ifdef CFSRLDEBUG
354 	static int canupgrade[] = {1, 2, 3, 103, 104, 105, 106, 107,
355 	    4, 5, 108, 6, 7, 8, 0};
356 #else /* CFSRLDEBUG */
357 	static int canupgrade[] = {1, 2, 3, 103, 104, 105, 106, 107,
358 	    4, 108, 5, 109, 110, 6, 111, 0};
359 #endif /* CFSRLDEBUG */
360 	char labelpath[MAXPATHLEN];
361 	struct cache_label clabel;
362 	int i;
363 
364 	if (((int)strlen(cachedirp) + (int)strlen(CACHELABEL_NAME) + 2)
365 	    >= MAXPATHLEN)
366 		return (1);
367 
368 	(void) sprintf(labelpath, "%s/%s", cachedirp, CACHELABEL_NAME);
369 
370 	if (cachefs_label_file_get(labelpath, &clabel) != 0)
371 		return (1);
372 
373 	/* nothing to do if we're current */
374 	if (clabel.cl_cfsversion == CFSVERSION)
375 		return (50);
376 
377 	/* see if it's an old version that we know how to upgrade */
378 	for (i = 0; canupgrade[i] != 0; i++)
379 		if (clabel.cl_cfsversion == canupgrade[i])
380 			break;
381 	if (canupgrade[i] == 0)
382 		return (36);
383 
384 	syslog(LOG_USER | LOG_INFO,
385 	    gettext("fsck -F cachefs: Recreating cache %s"), cachedirp);
386 
387 	/* currently, to `upgrade' we delete the old cache */
388 	if (cachefs_delete_all_cache(cachedirp) != 0)
389 		return (36);
390 
391 	/* do any magic necessary to convert the old label to the new one */
392 	clabel.cl_cfsversion = CFSVERSION;
393 
394 	/* create the new cache! */
395 	if (cachefs_create_cache(cachedirp, NULL, &clabel) != 0)
396 		return (36);
397 
398 	return (0);
399 }
400 
401 /*
402  *
403  *			cfs_check
404  *
405  * Description:
406  *	This routine performs the actual checking of the cache
407  *	file system.
408  *	The file system must be inactive when this routine is called.
409  * Arguments:
410  *	cachedirp	name of the cache directory to check
411  *	noclean		1 means ignore clean flag
412  *	mflag		1 means no fixes, only check if mountable
413  *	verbose		indicate level of verbosity for diagnostics
414  *	nonoclean	1 means honor clean flag; don't by default
415  * Returns:
416  *	Returns:
417  *		 0	file system is okay and does not need checking
418  *		 1	problem unrelated to the file system
419  *		32	file system is unmounted and needs checking
420  *		33	file system is already mounted
421  *		34	cannot stat device
422  *		36	uncorrectable errors detected - terminate normally
423  *		37	a signal was caught during processing
424  *		39	uncorrectable errors detected - terminate  immediately
425  *		40	for root mounted fs, same as 0, XXX
426  * Preconditions:
427  *	precond(cachedirp)
428  */
429 
430 int
431 cfs_check(char *cachedirp, int noclean, int mflag, int verbose, int nonoclean)
432 {
433 	DIR *dp;
434 	struct dirent64 *dep;
435 	char buf[MAXPATHLEN];
436 	struct stat64 statinfo;
437 	int xx;
438 	char *namep;
439 	res *resp;
440 	struct cache_label clabel;
441 
442 	/* if checking the clean flag is sufficient */
443 	if ((noclean == 0) && (nonoclean || mflag)) {
444 		/* if the clean flag is set */
445 		if (cachefs_clean_flag_test(cachedirp)) {
446 			if (verbose) {
447 				pr_err(gettext("Cache %s is clean"), cachedirp);
448 			}
449 			return (0);
450 		}
451 	}
452 
453 	/* if mflag specified then go no farther */
454 	if (mflag)
455 		return (32);
456 
457 	/* check the cache label file for correctness */
458 	xx = cache_label_file(cachedirp, &clabel);
459 	if (xx)
460 		return (xx);
461 
462 	/* make sure the kernel lock file exists */
463 	sprintf(buf, "%s/%s", cachedirp, CACHEFS_LOCK_FILE);
464 	xx = open(buf, O_RDWR | O_CREAT, 0700);
465 	if (xx == -1) {
466 		pr_err(gettext("Cannot create lock file %s"), buf);
467 		return (39);
468 	}
469 	close(xx);
470 
471 	/* fix permissions on the cache directory */
472 	xx = cache_permissions(cachedirp);
473 	if (xx)
474 		return (xx);
475 
476 	/* make the back file system mount directory if necessary */
477 	xx = cache_check_dir(cachedirp, BACKMNT_NAME);
478 	if (xx)
479 		return (xx);
480 
481 	/* clean out junk in the back file system mount directory */
482 	cache_backmnt_cleanup(cachedirp, BACKMNT_NAME);
483 
484 	/* make the lost+found directory if necessary */
485 	xx = cache_check_dir(cachedirp, CACHEFS_LOSTFOUND_NAME);
486 	if (xx)
487 		return (xx);
488 
489 	/* construct the path to the lost and found directory for file_remove */
490 	sprintf(S_lostfound, "%s/%s", cachedirp, CACHEFS_LOSTFOUND_NAME);
491 
492 	/* construct the path name of the resource file */
493 	namep = RESOURCE_NAME;
494 	xx = strlen(cachedirp) + strlen(namep) + 3;
495 	if (xx >= MAXPATHLEN) {
496 		pr_err(gettext("Path name too long %s/%s"),
497 		    cachedirp, namep);
498 		return (39);
499 	}
500 	sprintf(buf, "%s/%s", cachedirp, namep);
501 
502 	/* make a res object to operate on the resource file */
503 	resp = res_create(buf, clabel.cl_maxinodes, verbose);
504 	if (resp == NULL) {
505 		pr_err(gettext("Could not process resource file %s: %s"),
506 		    buf, strerror(errno));
507 		return (39);
508 	}
509 
510 	/* open the cache directory */
511 	if ((dp = opendir(cachedirp)) == NULL) {
512 		pr_err(gettext("Cannot open directory %s: %s"), cachedirp,
513 		    strerror(errno));
514 		res_destroy(resp);
515 		return (39);
516 	}
517 
518 	/* mark all directories */
519 	while ((dep = readdir64(dp)) != NULL) {
520 		/* ignore . and .. */
521 		if ((strcmp(dep->d_name, ".") == 0) ||
522 				(strcmp(dep->d_name, "..") == 0))
523 			continue;
524 
525 		/* check path length */
526 		xx = strlen(cachedirp) + strlen(dep->d_name) + 3;
527 		if (xx >= MAXPATHLEN) {
528 			pr_err(gettext("Path name too long %s/%s"),
529 			    cachedirp, dep->d_name);
530 			closedir(dp);
531 			res_destroy(resp);
532 			return (39);
533 		}
534 
535 		/* stat the file */
536 		sprintf(buf, "%s/%s", cachedirp, dep->d_name);
537 		xx = lstat64(buf, &statinfo);
538 		if (xx == -1) {
539 			if (errno != ENOENT) {
540 				pr_err(gettext("Cannot stat %s: %s"), cachedirp,
541 				    strerror(errno));
542 				closedir(dp);
543 				res_destroy(resp);
544 				return (39);
545 			}
546 			continue;
547 		}
548 
549 		/* if a directory */
550 		if (S_ISDIR(statinfo.st_mode)) {
551 			xx = chmod(buf, 0700);
552 			if (xx == -1) {
553 				pr_err(gettext("Cannot chmod %s: %s"), buf,
554 				    strerror(errno));
555 				closedir(dp);
556 				res_destroy(resp);
557 				return (39);
558 			}
559 		}
560 	}
561 
562 	/* process files in the cache directory */
563 	rewinddir(dp);
564 	while ((dep = readdir64(dp)) != NULL) {
565 		/* ignore . and .. */
566 		if ((strcmp(dep->d_name, ".") == 0) ||
567 				(strcmp(dep->d_name, "..") == 0))
568 			continue;
569 
570 		/* stat the file */
571 		sprintf(buf, "%s/%s", cachedirp, dep->d_name);
572 		xx = lstat64(buf, &statinfo);
573 		if (xx == -1) {
574 			if (errno != ENOENT) {
575 				pr_err(gettext("Cannot stat %s: %s"), cachedirp,
576 				    strerror(errno));
577 				closedir(dp);
578 				res_destroy(resp);
579 				return (39);
580 			}
581 			continue;
582 		}
583 
584 		/* ignore directories */
585 		if (S_ISDIR(statinfo.st_mode))
586 			continue;
587 
588 		/* if not a link */
589 		if (!S_ISLNK(statinfo.st_mode)) {
590 			/*
591 			 * XXX make sure a valid file
592 			 * Update file and block counts for this file.
593 			 * This file will be <2GB.
594 			 */
595 			res_addfile(resp, (long)statinfo.st_size);
596 			continue;
597 		}
598 
599 		/* process the file system cache directory */
600 		xx = process_fsdir(cachedirp, dep->d_name, resp, verbose);
601 		if (xx) {
602 			closedir(dp);
603 			res_destroy(resp);
604 			return (xx);
605 		}
606 	}
607 
608 	/* look for directories that do not belong */
609 	rewinddir(dp);
610 	while ((dep = readdir64(dp)) != NULL) {
611 		/* ignore . and .. */
612 		if ((strcmp(dep->d_name, ".") == 0) ||
613 				(strcmp(dep->d_name, "..") == 0))
614 			continue;
615 
616 		/* stat the file */
617 		sprintf(buf, "%s/%s", cachedirp, dep->d_name);
618 		xx = lstat64(buf, &statinfo);
619 		if (xx == -1) {
620 			if (errno != ENOENT) {
621 				pr_err(gettext("Cannot stat %s: %s"), cachedirp,
622 				    strerror(errno));
623 				closedir(dp);
624 				res_destroy(resp);
625 				return (39);
626 			}
627 			continue;
628 		}
629 
630 		/* XXX should we unlink extraneous regular files? */
631 
632 		/* ignore all but directories */
633 		if (!S_ISDIR(statinfo.st_mode))
634 			continue;
635 
636 		/* ignore directories we have checked */
637 		if ((statinfo.st_mode & S_IAMB) != 0700)
638 			continue;
639 
640 		/* ignore the mount directory */
641 		if (strcmp(dep->d_name, BACKMNT_NAME) == 0)
642 			continue;
643 
644 		/* ignore the lost+found directory */
645 		if (strcmp(dep->d_name, CACHEFS_LOSTFOUND_NAME) == 0)
646 			continue;
647 
648 		/* remove the directory */
649 		xx = nftw64(buf, tree_remove, 3, FLAGS_FTW);
650 		if (xx != 0) {
651 			pr_err(gettext("Error walking tree %s."), namep);
652 			closedir(dp);
653 			res_destroy(resp);
654 			return (39);
655 		}
656 
657 		if (verbose)
658 			pr_err(gettext("Directory removed: %s"), buf);
659 	}
660 
661 	/* close the directory */
662 	closedir(dp);
663 
664 	/* add one file and one block for the cache directory itself */
665 	res_addfile(resp, 1);
666 
667 	/* finish off the resource file processing */
668 	xx = res_done(resp);
669 	if (xx == -1) {
670 		pr_err(gettext("Could not finish resource file %s: %s"),
671 		    buf, strerror(errno));
672 		return (39);
673 	}
674 	res_destroy(resp);
675 
676 	/* return success */
677 	return (0);
678 }
679 
680 /*
681  *
682  *			cache_label_file
683  *
684  * Description:
685  *	This routine performs the checking and fixing up of the
686  *	cache label file.
687  * Arguments:
688  *	cachedirp	name of the cache directory to check
689  *	clabelp		cache label contents put here if not NULL
690  * Returns:
691  *		 0	file system is okay and does not need checking
692  *		 1	problem unrelated to the file system
693  *		32	file system is unmounted and needs checking
694  *		33	file system is already mounted
695  *		34	cannot stat device
696  *		36	uncorrectable errors detected - terminate normally
697  *		37	a signal was caught during processing
698  *		39	uncorrectable errors detected - terminate  immediately
699  * Preconditions:
700  *	precond(cachedirp)
701  */
702 
703 int
704 cache_label_file(char *cachedirp, struct cache_label *clabelp)
705 {
706 	int xx;
707 	char buf1[MAXPATHLEN];
708 	char buf2[MAXPATHLEN];
709 	char *namep;
710 	struct cache_label clabel1, clabel2;
711 
712 	namep = CACHELABEL_NAME;
713 
714 	/* see if path name is too long */
715 	xx = strlen(cachedirp) + strlen(namep) + 10;
716 	if (xx >= MAXPATHLEN) {
717 		pr_err(gettext("Cache directory name %s is too long"),
718 		    cachedirp);
719 		return (39);
720 	}
721 
722 	/* make a path to the cache label file and its backup copy */
723 	sprintf(buf1, "%s/%s", cachedirp, namep);
724 	sprintf(buf2, "%s/%s.dup", cachedirp, namep);
725 
726 	/* get the contents of the cache label file */
727 	xx = cachefs_label_file_get(buf1, &clabel1);
728 	if (xx == -1) {
729 		/* get the backup cache label file contents */
730 		xx = cachefs_label_file_get(buf2, &clabel2);
731 		if (xx == -1) {
732 			pr_err(gettext("Run `cfsadmin -d all %s'\n"
733 			    "and then run\n"
734 			    "`cfsadmin -c %s'\n"), cachedirp, cachedirp);
735 			return (39);
736 		}
737 
738 		/* write the cache label file */
739 		xx = cachefs_label_file_put(buf1, &clabel2);
740 		if (xx == -1) {
741 			pr_err(gettext("Run `cfsadmin -d all %s'\n"
742 			    "and then run\n"
743 			    "`cfsadmin -c %s'\n"), cachedirp, cachedirp);
744 			return (39);
745 		}
746 		pr_err(gettext("Cache label file %s repaired."), buf1);
747 
748 		/* copy out the contents to the caller */
749 		if (clabelp)
750 			*clabelp = clabel2;
751 
752 		/* return success */
753 		return (0);
754 	}
755 
756 	/* get the contents of the backup cache label file */
757 	xx = cachefs_label_file_get(buf2, &clabel2);
758 	if (xx == -1) {
759 		/* write the backup cache label file */
760 		xx = cachefs_label_file_put(buf2, &clabel1);
761 		if (xx == -1) {
762 			return (39);
763 		}
764 		pr_err(gettext("Cache label file %s repaired."), buf2);
765 	}
766 
767 	/* copy out the contents to the caller */
768 	if (clabelp)
769 		*clabelp = clabel1;
770 
771 	/* return success */
772 	return (0);
773 }
774 
775 /*
776  *
777  *			cache_permissions
778  *
779  * Description:
780  *	Checks the permissions on the cache directory and fixes
781  *	them if necessary.
782  * Arguments:
783  *	cachedirp	name of the cache directory to check
784  * Returns:
785  *		 0	file system is okay and does not need checking
786  *		 1	problem unrelated to the file system
787  *		32	file system is unmounted and needs checking
788  *		33	file system is already mounted
789  *		34	cannot stat device
790  *		36	uncorrectable errors detected - terminate normally
791  *		37	a signal was caught during processing
792  *		39	uncorrectable errors detected - terminate  immediately
793  * Preconditions:
794  *	precond(cachedirp)
795  */
796 
797 int
798 cache_permissions(char *cachedirp)
799 {
800 	int xx;
801 	struct stat64 statinfo;
802 
803 	/* get info about the cache directory */
804 	xx = lstat64(cachedirp, &statinfo);
805 	if (xx == -1) {
806 		pr_err(gettext("Could not stat %s: %s"), cachedirp,
807 		    strerror(errno));
808 		return (34);
809 	}
810 
811 	/* check the mode bits */
812 	if ((statinfo.st_mode & S_IAMB) != 0) {
813 
814 		/* fix the mode bits */
815 		xx = chmod(cachedirp, 0);
816 		if (xx == -1) {
817 			pr_err(gettext("Could not set modes bits on "
818 			    "cache directory %s: %s"),
819 			    cachedirp, strerror(errno));
820 			return (36);
821 		}
822 		pr_err(gettext("Mode bits reset on cache directory %s"),
823 		    cachedirp);
824 	}
825 
826 	/* return success */
827 	return (0);
828 }
829 
830 /*
831  *
832  *			cache_check_dir
833  *
834  * Description:
835  *	Checks for the existance of the directory
836  *	and creates it if necessary.
837  * Arguments:
838  *	cachedirp	name of the cache directory containing the dir
839  *	namep		name of dir
840  * Returns:
841  *		 0	file system is okay and does not need checking
842  *		 1	problem unrelated to the file system
843  *		32	file system is unmounted and needs checking
844  *		33	file system is already mounted
845  *		34	cannot stat device
846  *		36	uncorrectable errors detected - terminate normally
847  *		37	a signal was caught during processing
848  *		39	uncorrectable errors detected - terminate  immediately
849  * Preconditions:
850  *	precond(cachedirp)
851  *	precond(dirp)
852  */
853 
854 int
855 cache_check_dir(char *cachedirp, char *namep)
856 {
857 	int xx;
858 	char buf[MAXPATHLEN];
859 	struct stat64 statinfo;
860 
861 	/* see if path name is too long */
862 	xx = strlen(cachedirp) + strlen(namep) + 3;
863 	if (xx >= MAXPATHLEN) {
864 		pr_err(gettext("Cache directory name %s is too long"),
865 		    cachedirp);
866 		return (39);
867 	}
868 
869 	/* make the pathname of the directory */
870 	sprintf(buf, "%s/%s", cachedirp, namep);
871 
872 	/* get info on the directory */
873 	xx = lstat64(buf, &statinfo);
874 	if (xx == -1) {
875 		/* if an error other than it does not exist */
876 		if (errno != ENOENT) {
877 			pr_err(gettext("Error on lstat(2) of %s: %s"),
878 			    buf, strerror(errno));
879 			return (39);
880 		}
881 
882 		/* make the directory */
883 		xx = mkdir(buf, 0);
884 		if (xx == -1) {
885 			pr_err(gettext("Could not create directory %s"),
886 			    buf);
887 			return (39);
888 		}
889 		pr_err(gettext("Created directory %s"), buf);
890 	}
891 
892 	/* else see if really a directory */
893 	else if (!S_ISDIR(statinfo.st_mode)) {
894 		/* get rid of the file */
895 		xx = unlink(buf);
896 		if (xx == -1) {
897 			pr_err(gettext("Cannot remove %s: %s"), buf,
898 			    strerror(errno));
899 			return (39);
900 		}
901 
902 		/* make the directory */
903 		xx = mkdir(buf, 0);
904 		if (xx == -1) {
905 			pr_err(gettext("Could not create directory %s"),
906 			    buf);
907 			return (39);
908 		}
909 		pr_err(gettext("Created directory %s"), buf);
910 	}
911 
912 	/* return success */
913 	return (0);
914 }
915 
916 /*
917  *
918  *			process_fsdir
919  *
920  * Description:
921  *	Performs the necessary checking and repair on the
922  *	specified file system cache directory.
923  *	Calls res_addfile and res_addident as appropriate.
924  * Arguments:
925  *	cachedirp	name of cache directory
926  *	namep		name of link file for the file system cache
927  *	resp		res object for res_addfile and res_addident calls
928  *	verbose		indicate level of verbosity for diagnostics
929  * Returns:
930  *		 0	file system is okay and does not need checking
931  *		 1	problem unrelated to the file system
932  *		32	file system is unmounted and needs checking
933  *		33	file system is already mounted
934  *		34	cannot stat device
935  *		36	uncorrectable errors detected - terminate normally
936  *		37	a signal was caught during processing
937  *		39	uncorrectable errors detected - terminate  immediately
938  * Preconditions:
939  *	precond(cachedirp)
940  *	precond(namep && is a sym link)
941  *	precond(resp)
942  */
943 
944 int
945 process_fsdir(char *cachedirp, char *namep, res *resp, int verbose)
946 {
947 	DIR *dp;
948 	struct dirent64 *dep;
949 	char linkpath[MAXPATHLEN];
950 	char dirpath[MAXPATHLEN];
951 	char attrpath[MAXPATHLEN];
952 	char buf[MAXPATHLEN];
953 	int xx;
954 	struct stat64 statinfo;
955 	char *atp = ATTRCACHE_NAME;
956 	int fd;
957 	ino64_t base;
958 	int local;
959 	char *strp;
960 	ino64_t fsid;
961 	int error = 0;
962 	int hashsize = 0;
963 	ENTRY hitem;
964 	ino64_t maxlocalfileno;
965 	cachefs_fsinfo_t fsinfo;
966 	time32_t btime;
967 
968 	/* construct the path to the sym link */
969 	xx = strlen(cachedirp) + strlen(namep) + 3;
970 	if (xx >= MAXPATHLEN) {
971 		pr_err(gettext("Pathname too long %s/%s"), cachedirp, namep);
972 		error = 39;
973 		goto out;
974 	}
975 	sprintf(linkpath, "%s/%s", cachedirp, namep);
976 
977 	/* read the contents of the link */
978 	xx = readlink(linkpath, buf, sizeof (buf));
979 	if (xx == -1) {
980 		pr_err(gettext("Unable to read link %s: %s"), linkpath,
981 		    strerror(errno));
982 		error = 39;
983 		goto out;
984 	}
985 	buf[xx] = '\0';
986 
987 	/* do a one time check on lengths of files */
988 	xx = strlen(cachedirp) + strlen(buf) + 20 + 20;
989 	if (xx >= MAXPATHLEN) {
990 		pr_err(gettext("Pathname too long %s/%s"), cachedirp, buf);
991 		error = 39;
992 		goto out;
993 	}
994 
995 	/* construct the path to the directory */
996 	sprintf(dirpath, "%s/%s", cachedirp, buf);
997 
998 	/* stat the directory */
999 	xx = lstat64(dirpath, &statinfo);
1000 	if ((xx == -1) || (strtoull(buf, NULL, 16) != statinfo.st_ino)) {
1001 		if ((xx == -1) && (errno != ENOENT)) {
1002 			pr_err(gettext("Could not stat %s: %s"), dirpath,
1003 			    strerror(errno));
1004 			error = 39;
1005 		} else
1006 			error = -1;
1007 		goto out;
1008 	}
1009 	fsid = statinfo.st_ino;
1010 
1011 	/*
1012 	 * Check for a disconnect log(dlog) file and verify it.
1013 	 */
1014 	xx = dlog_ck(dirpath, &maxlocalfileno);
1015 	if (xx) {
1016 		error = -1;
1017 		goto out;
1018 	}
1019 
1020 	/* process the fsinfo file */
1021 	sprintf(buf, "%s/%s", dirpath, CACHEFS_FSINFO);
1022 	xx = process_fsinfo(buf, maxlocalfileno, &fsinfo, verbose);
1023 	if (xx) {
1024 		error = -1;
1025 		pr_err(gettext("Cannot update fsinfo file %s"), buf);
1026 		goto out;
1027 	}
1028 
1029 	/* create the unmount file in the cachedir */
1030 	sprintf(buf, "%s/%s", dirpath, CACHEFS_UNMNT_FILE);
1031 	/* this file will be < 2GB */
1032 	fd = open(buf, O_CREAT | O_RDWR, 0666);
1033 	if (fd == -1) {
1034 		pr_err(gettext("Cannot create unmnt file %s: %s"), buf,
1035 		    strerror(errno));
1036 		error = -1;
1037 		goto out;
1038 	}
1039 	btime = get_boottime();
1040 	if (write(fd, &btime, sizeof (btime)) == -1) {
1041 		pr_err(gettext("Cannot write cachedir unmnt file %s: %s"), buf,
1042 		    strerror(errno));
1043 		error = -1;
1044 		goto out;
1045 	}
1046 	close(fd);
1047 
1048 	/* create the unmount file */
1049 	sprintf(buf, "%s/%s", dirpath, CACHEFS_UNMNT_FILE);
1050 	/* this file will be < 2GB */
1051 	fd = open(buf, O_CREAT | O_RDWR, 0666);
1052 	if (fd == -1) {
1053 		pr_err(gettext("Cannot create unmnt file %s: %s"), buf,
1054 		    strerror(errno));
1055 		error = -1;
1056 		goto out;
1057 	}
1058 	close(fd);
1059 
1060 	/* construct the name to the attrcache directory */
1061 	sprintf(attrpath, "%s/%s", dirpath, atp);
1062 
1063 	/* open the attrcache directory */
1064 	if ((dp = opendir(attrpath)) == NULL) {
1065 		pr_err(gettext("Cannot open directory %s: %s"), attrpath,
1066 		    strerror(errno));
1067 		error = -1;
1068 		goto out;
1069 	}
1070 
1071 	/* make one pass, counting how big to make the hash table */
1072 	while (readdir64(dp) != NULL)
1073 		++hashsize;
1074 	if (hcreate(hashsize + 1000) == 0) {
1075 		pr_err(gettext("Cannot allocate heap space."));
1076 		(void) closedir(dp);
1077 		hashsize = 0;
1078 		error = 39;
1079 		goto out;
1080 	}
1081 	rewinddir(dp);
1082 
1083 	/* loop reading the contents of the directory */
1084 	while ((dep = readdir64(dp)) != NULL) {
1085 		/* ignore . and .. */
1086 		if ((strcmp(dep->d_name, ".") == 0) ||
1087 		    (strcmp(dep->d_name, "..") == 0))
1088 			continue;
1089 
1090 		/* check for a reasonable name */
1091 		xx = strlen(dep->d_name);
1092 		if ((xx != 16) && (xx != 17)) {
1093 			/* bad file */
1094 			pr_err(gettext("Unknown file %s/%s"),
1095 				attrpath, dep->d_name);
1096 			closedir(dp);
1097 			error = 39;
1098 			goto out;
1099 		}
1100 
1101 		/* derive the base number from the file name */
1102 		if (*(dep->d_name) == 'L') {
1103 			local = 1;
1104 			base = strtoull(dep->d_name + 1, &strp, 16);
1105 		} else {
1106 			local = 0;
1107 			base = strtoull(dep->d_name, &strp, 16);
1108 		}
1109 		if (*strp != '\0') {
1110 			/* bad file */
1111 			pr_err(gettext("Unknown file %s/%s"),
1112 				attrpath, dep->d_name);
1113 			closedir(dp);
1114 			error = 39;
1115 			goto out;
1116 		}
1117 
1118 		/* process the file group */
1119 		error = process_fsgroup(dirpath, dep->d_name, resp,
1120 			base, fsinfo.fi_fgsize, fsid, local, verbose);
1121 		if (error) {
1122 			closedir(dp);
1123 			goto out;
1124 		}
1125 	}
1126 	closedir(dp);
1127 
1128 	/* open the fscache directory */
1129 	if ((dp = opendir(dirpath)) == NULL) {
1130 		pr_err(gettext("Cannot open directory %s: %s"), dirpath,
1131 		    strerror(errno));
1132 		error = 39;
1133 		goto out;
1134 	}
1135 
1136 	/* loop reading the contents of the directory */
1137 	while ((dep = readdir64(dp)) != NULL) {
1138 		/* ignore . and .. */
1139 		if ((strcmp(dep->d_name, ".") == 0) ||
1140 		    (strcmp(dep->d_name, "..") == 0))
1141 			continue;
1142 
1143 		/* ignore cachefs special files */
1144 		xx = strncmp(dep->d_name, CACHEFS_PREFIX, CACHEFS_PREFIX_LEN);
1145 		if (xx == 0)
1146 			continue;
1147 
1148 		hitem.key = dep->d_name;
1149 		hitem.data = NULL;
1150 		if (hsearch(hitem, FIND) == NULL) {
1151 			sprintf(buf, "%s/%s", dirpath, dep->d_name);
1152 			if (verbose) {
1153 				printf("Unreferenced dir %s\n", buf);
1154 			}
1155 			xx = nftw64(buf, tree_remove, 3, FLAGS_FTW);
1156 			if (xx != 0) {
1157 				pr_err(gettext("Could not remove %s"), buf);
1158 				error = 39;
1159 				closedir(dp);
1160 				goto out;
1161 			}
1162 		}
1163 	}
1164 	closedir(dp);
1165 
1166 	/* add the info file to the resource */
1167 	res_addfile(resp, 1);
1168 
1169 	/* add the directory to the resources */
1170 	res_addfile(resp, 1);
1171 
1172 	/* add the sym link to the resources */
1173 	res_addfile(resp, 1);
1174 
1175 	/* change the mode on the directory to indicate we visited it */
1176 	xx = chmod(dirpath, 0777);
1177 	if (xx == -1) {
1178 		pr_err(gettext("Cannot chmod %s: %s"), dirpath,
1179 		    strerror(errno));
1180 		error = 39;
1181 		goto out;
1182 	}
1183 
1184 out:
1185 	/* free up the heap allocated by the hash functions */
1186 	if (hashsize != 0)
1187 		hdestroy();
1188 
1189 	if (error == -1) {
1190 		/* remove the sym link */
1191 		xx = unlink(linkpath);
1192 		if (xx == -1) {
1193 			pr_err(gettext("Unable to remove %s: %s"), linkpath,
1194 			    strerror(errno));
1195 			error = 39;
1196 		} else {
1197 			error = 0;
1198 		}
1199 	}
1200 
1201 	return (error);
1202 }
1203 
1204 /*
1205  * Processes and fixes up the fsinfo file.
1206  */
1207 int
1208 process_fsinfo(char *namep, ino64_t maxlocalfileno, cachefs_fsinfo_t *fsinfop,
1209     int verbose)
1210 {
1211 	int fd;
1212 	int error;
1213 	cachefs_fsinfo_t fsinfo;
1214 	int xx;
1215 
1216 	/* open the info file; this file will be <2GB */
1217 	fd = open(namep, O_RDWR);
1218 	if (fd == -1) {
1219 		error = errno;
1220 		if (verbose)
1221 			pr_err(gettext("Could not open %s: %s"),
1222 			    namep, strerror(errno));
1223 		if (error != ENOENT)
1224 			return (-1);
1225 
1226 		/* try to create the info file */
1227 		fd = open(namep, O_RDWR | O_CREAT, 0666);
1228 		if (fd == -1) {
1229 			if (verbose)
1230 				pr_err(gettext("Could not create %s: %s"),
1231 				    namep, strerror(errno));
1232 			return (-1);
1233 		}
1234 
1235 	}
1236 
1237 	/* read the contents of the info file */
1238 	xx = read(fd, &fsinfo, sizeof (fsinfo));
1239 	if (xx != sizeof (fsinfo)) {
1240 		memset(&fsinfo, 0, sizeof (fsinfo));
1241 	}
1242 
1243 	/* fix up the fields as necessary */
1244 	if (fsinfo.fi_popsize < DEF_POP_SIZE)
1245 		fsinfo.fi_popsize = DEF_POP_SIZE;
1246 	if (fsinfo.fi_fgsize < DEF_FILEGRP_SIZE)
1247 		fsinfo.fi_fgsize = DEF_FILEGRP_SIZE;
1248 	if (fsinfo.fi_localfileno < maxlocalfileno)
1249 		fsinfo.fi_localfileno = maxlocalfileno;
1250 
1251 	/* write back the info to the file */
1252 	if (lseek(fd, 0, SEEK_SET) == -1) {
1253 		if (verbose)
1254 			pr_err(gettext("Could not lseek %s: %s"),
1255 			    namep, strerror(errno));
1256 		close(fd);
1257 		return (-1);
1258 	}
1259 	xx = write(fd, &fsinfo, sizeof (fsinfo));
1260 	if (xx != sizeof (fsinfo)) {
1261 		if (verbose)
1262 			pr_err(gettext("Could not write %s: %s"),
1263 			    namep, strerror(errno));
1264 		close(fd);
1265 		return (-1);
1266 	}
1267 
1268 	if (fsync(fd) == -1) {
1269 		pr_err(gettext("Could not sync %s: %s"),
1270 		    namep, strerror(errno));
1271 		(void) close(fd);
1272 		return (-1);
1273 	}
1274 	(void) close(fd);
1275 	*fsinfop = fsinfo;
1276 	return (0);
1277 }
1278 
1279 /*
1280  *
1281  *			process_fsgroup
1282  *
1283  * Description:
1284  *	Performs the necessary checking and repair on the
1285  *	specified file group directory.
1286  *	Calls res_addfile and res_addident as appropriate.
1287  * Arguments:
1288  *	dirpath	pathname to fscache directory
1289  *	namep	name of fsgroup
1290  *	resp	res object for res_addfile and res_addident calls
1291  *	base	base offset for file numbers in this directory
1292  *	fgsize	size of the file groups
1293  *	fsid	file system id
1294  *	local	1 if fsgroup dir is a local dir
1295  *	verbose		indicate level of verbosity for diagnostics
1296  * Returns:
1297  *		 0	file system is okay and does not need checking
1298  *		 1	problem unrelated to the file system
1299  *		32	file system is unmounted and needs checking
1300  *		33	file system is already mounted
1301  *		34	cannot stat device
1302  *		36	uncorrectable errors detected - terminate normally
1303  *		37	a signal was caught during processing
1304  *		39	uncorrectable errors detected - terminate  immediately
1305  * Preconditions:
1306  *	precond(dirp)
1307  *	precond(namep)
1308  *	precond(resp)
1309  *	precond(fgsize > 0)
1310  */
1311 
1312 int
1313 process_fsgroup(char *dirp, char *namep, res *resp, ino64_t base, int fgsize,
1314     ino64_t fsid, int local, int verbose)
1315 {
1316 	DIR *dp;
1317 	struct dirent64 *dep;
1318 	char buf[MAXPATHLEN];
1319 	char attrfile[MAXPATHLEN];
1320 	char attrdir[MAXPATHLEN];
1321 	int xx;
1322 	struct stat64 statinfo;
1323 	char *atp = ATTRCACHE_NAME;
1324 	void *addrp = MAP_FAILED;
1325 	struct attrcache_header *ahp;
1326 	struct attrcache_index *startp = NULL;
1327 	struct attrcache_index *aip;
1328 	uchar_t *bitp;
1329 	int offlen;
1330 	int bitlen;
1331 	int fd;
1332 	int offentry;
1333 	int size;
1334 	struct cachefs_metadata *metap;
1335 	int index;
1336 	char *strp;
1337 	uint_t offset;
1338 	int error = 0;
1339 	ENTRY hitem;
1340 	int nffs;
1341 	int rlno;
1342 	rl_entry_t ent;
1343 	enum cachefs_rl_type which;
1344 
1345 	/* construct the name to the attribute file and front file dir */
1346 	sprintf(attrfile, "%s/%s/%s", dirp, atp, namep);
1347 	sprintf(attrdir, "%s/%s", dirp, namep);
1348 
1349 	/* get the size of the attribute file */
1350 	xx = lstat64(attrfile, &statinfo);
1351 	if (xx == -1) {
1352 		pr_err(gettext("Could not stat %s: %s"), attrfile,
1353 		    strerror(errno));
1354 		error = 39;
1355 		goto out;
1356 	}
1357 
1358 	offlen = sizeof (struct attrcache_index) * fgsize;
1359 	bitlen = (sizeof (uchar_t) * fgsize + 7) / 8;
1360 	/* attrfile will be <2GB */
1361 	size = (int)statinfo.st_size;
1362 	offentry = sizeof (struct attrcache_header) + offlen + bitlen;
1363 
1364 	/* if the attribute file is the wrong size */
1365 	if (size < offentry) {
1366 		error = -1;
1367 		goto out;
1368 	}
1369 
1370 	/* open the attribute file */
1371 	fd = open(attrfile, O_RDWR);
1372 	if (fd == -1) {
1373 		pr_err(gettext("Could not open %s: %s"),
1374 			attrfile, strerror(errno));
1375 		error = 39;
1376 		goto out;
1377 	}
1378 
1379 	/* mmap the file into our address space */
1380 	addrp = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
1381 	if (addrp == MAP_FAILED) {
1382 		pr_err(gettext("Could not map %s: %s"),
1383 			attrfile, strerror(errno));
1384 		close(fd);
1385 		error = 39;
1386 		goto out;
1387 	}
1388 	close(fd);
1389 
1390 	/* set up pointers into mapped file */
1391 	ahp = (struct attrcache_header *)addrp;
1392 	startp = (struct attrcache_index *)(ahp + 1);
1393 	bitp = (uchar_t *)((char *)startp + offlen);
1394 
1395 	/* clear the bitmap */
1396 	memset(bitp, 0, bitlen);
1397 
1398 	/* fix number of allocated blocks value if necessary */
1399 	xx = (size + MAXBSIZE - 1) / MAXBSIZE;
1400 	if (xx != ahp->ach_nblks) {
1401 		if (verbose) {
1402 			pr_err(gettext("File %s size wrong, old %d new %d:"
1403 				"corrected."),
1404 				attrfile, ahp->ach_nblks, xx);
1405 		}
1406 		ahp->ach_nblks = xx;
1407 	}
1408 	ahp->ach_nffs = 0;
1409 	nffs = 0;
1410 
1411 	/* verify sanity of attribute file */
1412 	ahp->ach_count = 0;
1413 	for (index = 0; index < fgsize; index++) {
1414 
1415 		/* get next entry to work on */
1416 		aip = startp + index;
1417 
1418 		/* save offset to data */
1419 		offset = aip->ach_offset;
1420 		aip->ach_offset = 0;
1421 
1422 		/* if entry not in use */
1423 		if (aip->ach_written == 0)
1424 			continue;
1425 		aip->ach_written = 0;
1426 
1427 		/* if offset is out of range or invalid */
1428 		if ((offset < offentry) ||
1429 		    ((size - sizeof (struct cachefs_metadata)) < offset) ||
1430 		    (offset & 3)) {
1431 			if (verbose)
1432 				pr_err(gettext("Offset %d invalid - index %d"),
1433 				    offset, index);
1434 			continue;
1435 		}
1436 
1437 		/* get pointer to meta data */
1438 		metap = (struct cachefs_metadata *)((char *)addrp + offset);
1439 
1440 		/* sanity check the meta data */
1441 		if ((metap->md_vattr.va_nodeid != (base + (ino64_t)index)) ||
1442 		    ((metap->md_flags & (MD_FILE | MD_POPULATED)) ==
1443 		    MD_POPULATED) ||
1444 		    ((metap->md_flags & MD_FILE) && (metap->md_rlno == 0)) ||
1445 		    (metap->md_rltype < CACHEFS_RL_START) ||
1446 		    (metap->md_rltype > CACHEFS_RL_END)) {
1447 			if (verbose) {
1448 				pr_err(gettext("Metadata corrupted %d"), index);
1449 			}
1450 			continue;
1451 		}
1452 
1453 		/* if there is a front file */
1454 		if (metap->md_flags & MD_FILE) {
1455 			/* make sure front file is still there */
1456 			if (local)
1457 				sprintf(buf, "%s/L%016llx", attrdir,
1458 				    base + (ino64_t)index);
1459 			else
1460 				sprintf(buf, "%s/%016llx", attrdir,
1461 				    base + (ino64_t)index);
1462 			if (access(buf, F_OK)) {
1463 				if (verbose) {
1464 					pr_err(gettext("File error %s %s"),
1465 					    buf, strerror(errno));
1466 				}
1467 				continue;
1468 			}
1469 			nffs++;
1470 
1471 			/* make sure default ACL directory holder is there */
1472 			if (metap->md_flags & MD_ACLDIR) {
1473 				sprintf(buf, (local) ?
1474 				    "%s/L%016llx.d" : "%s/%016llx.d",
1475 				    attrdir, base + (ino64_t)index);
1476 				if (access(buf, F_OK)) {
1477 					if (verbose) {
1478 						pr_err(gettext(
1479 						    "File error %s %s"),
1480 						    buf, strerror(errno));
1481 					}
1482 					continue;
1483 				}
1484 			}
1485 		}
1486 
1487 		/* if using a rl slot */
1488 		if (metap->md_rlno) {
1489 			/* make sure not on an unusable list */
1490 			if ((metap->md_rltype == CACHEFS_RL_NONE) ||
1491 			    (metap->md_rltype == CACHEFS_RL_FREE)) {
1492 				if (verbose) {
1493 					pr_err(gettext("Bad list %d, %d"),
1494 					    metap->md_rltype, index);
1495 				}
1496 				continue;
1497 			}
1498 
1499 			/* move from the active to the gc list */
1500 			if (metap->md_rltype == CACHEFS_RL_ACTIVE)
1501 				metap->md_rltype = CACHEFS_RL_GC;
1502 
1503 			/* move from the mf to the modified list */
1504 			if (metap->md_rltype == CACHEFS_RL_MF)
1505 				metap->md_rltype = CACHEFS_RL_MODIFIED;
1506 
1507 			/* add to the resource file */
1508 			ent.rl_attrc = 0;
1509 			ent.rl_local = local;
1510 			ent.rl_fsid = fsid;
1511 			ent.rl_fileno = base + (ino64_t)index;
1512 			ent.rl_current = metap->md_rltype;
1513 			xx = res_addident(resp, metap->md_rlno, &ent,
1514 			    metap->md_frontblks * MAXBSIZE,
1515 			    (metap->md_flags & MD_FILE) ? 1 : 0);
1516 			if (xx == -1) {
1517 				if (verbose) {
1518 					pr_err(gettext(
1519 					    "File %s, bad rlno"), attrfile);
1520 				}
1521 				continue;
1522 			}
1523 			ahp->ach_nffs++;
1524 		}
1525 
1526 		/* mark entry as valid */
1527 		aip->ach_written = 1;
1528 		aip->ach_offset = offset;
1529 
1530 		/* set bitmap for this entry */
1531 		xx = (offset - offentry) / sizeof (struct cachefs_metadata);
1532 		bitp[xx/8] |= 1 << (xx % 8);
1533 
1534 		/* bump number of active entries */
1535 		ahp->ach_count += 1;
1536 	}
1537 
1538 	/* loop reading the contents of the front file directory */
1539 	dp = opendir(attrdir);
1540 	while (dp && ((dep = readdir64(dp)) != NULL)) {
1541 		int acldir;
1542 
1543 		/* ignore . and .. */
1544 		if ((strcmp(dep->d_name, ".") == 0) ||
1545 		    (strcmp(dep->d_name, "..") == 0))
1546 			continue;
1547 
1548 		acldir = 0;
1549 		xx = strlen(dep->d_name);
1550 		/* check for valid ACL directory */
1551 		if ((xx > 2) && (strcmp(dep->d_name + xx - 2, ".d") == 0)) {
1552 			acldir = 1;
1553 		} else if ((xx != 16) && (xx != 17)) {
1554 			/*
1555 			 * Bad file.
1556 			 * Front file dir name is based on 64 bit inode number.
1557 			 */
1558 			pr_err(gettext("Unknown file %s/%s"),
1559 				attrdir, dep->d_name);
1560 			closedir(dp);
1561 			error = 39;
1562 			goto out;
1563 		}
1564 
1565 		sprintf(buf, "%s/%s", attrdir, dep->d_name);
1566 
1567 		/* determine index into file group */
1568 		if (*(dep->d_name) == 'L') {
1569 			index = (int)(strtoull(dep->d_name + 1, &strp,
1570 			    16) - base);
1571 		} else {
1572 			index = (int)(strtoull(dep->d_name, &strp, 16) - base);
1573 		}
1574 
1575 		/* verify a valid file */
1576 		if (((! acldir) && (*strp != '\0')) ||
1577 		    ((acldir) && (strcmp(strp, ".d") != 0)) ||
1578 		    (index < 0) || (fgsize <= index) ||
1579 		    (startp[index].ach_written == 0)) {
1580 			/* remove the file */
1581 			xx = file_remove(buf, NULL, verbose);
1582 			if (xx == -1) {
1583 				error = 39;
1584 				goto out;
1585 			}
1586 			continue;
1587 		}
1588 
1589 		/* verify file should be there */
1590 		aip = startp + index;
1591 		offset = aip->ach_offset;
1592 		metap = (struct cachefs_metadata *)((char *)addrp + offset);
1593 		if (((metap->md_flags & MD_FILE) == 0) ||
1594 		    ((acldir) && ((metap->md_flags & MD_ACLDIR) == 0))) {
1595 			/* remove the file */
1596 			if (acldir)
1597 				xx = rmdir(buf);
1598 			else
1599 				xx = file_remove(buf, NULL, verbose);
1600 			if (xx == -1) {
1601 				error = 39;
1602 				goto out;
1603 			}
1604 			continue;
1605 		}
1606 		if (! acldir)
1607 			nffs--;
1608 	}
1609 
1610 	/* close the directory */
1611 	if (dp)
1612 		closedir(dp);
1613 
1614 	/* if we did not find the correct number of front files in the dir */
1615 	rlno = ahp->ach_rlno;
1616 	if (nffs != 0) {
1617 		if (verbose) {
1618 			pr_err(gettext("Front file mismatch %d in %s"),
1619 			    nffs, attrdir);
1620 		}
1621 		error = -1;
1622 		goto out;
1623 	}
1624 
1625 	/* add the attrcache file to the resouce file */
1626 	which = (ahp->ach_nffs == 0) ? CACHEFS_RL_GC : CACHEFS_RL_ATTRFILE;
1627 	ahp->ach_rl_current = which;
1628 	ent.rl_attrc = 1;
1629 	ent.rl_local = local;
1630 	ent.rl_fsid = fsid;
1631 	ent.rl_fileno = base;
1632 	ent.rl_current = which;
1633 	error = res_addident(resp, rlno, &ent, size, 1);
1634 	if (error == -1) {
1635 		if (verbose) {
1636 			pr_err(gettext("%s bad rlno %d\n"), attrfile, rlno);
1637 		}
1638 		goto out;
1639 	} else if (ahp->ach_nffs > 0) {
1640 		/* add the directory to the resources */
1641 		res_addfile(resp, 1);
1642 
1643 		/* indicate that the file group directory is okay */
1644 		hitem.key = strdup(namep);
1645 		hitem.data = NULL;
1646 		if (hsearch(hitem, ENTER) == NULL) {
1647 			pr_err(gettext("Hash table full"));
1648 			error = 39;
1649 			goto out;
1650 		}
1651 	}
1652 
1653 out:
1654 	if (error == -1) {
1655 		if (startp) {
1656 			/* clear idents we created for this attrcache file */
1657 			for (index = 0; index < fgsize; index++) {
1658 				aip = startp + index;
1659 				if (aip->ach_written == 0)
1660 					continue;
1661 				metap = (struct cachefs_metadata *)((char *)
1662 				    addrp + aip->ach_offset);
1663 				if (metap->md_rlno != 0) {
1664 					/* clear the resource file idents */
1665 					res_clearident(resp, metap->md_rlno,
1666 					    (metap->md_frontblks * MAXBSIZE),
1667 					    (metap->md_flags & MD_FILE) ? 1:0);
1668 					if (verbose) {
1669 						pr_err(gettext("Removed %d"),
1670 							metap->md_rlno);
1671 					}
1672 				}
1673 			}
1674 		}
1675 
1676 		/* nuke the attrcache file */
1677 		xx = unlink(attrfile);
1678 		if (xx == -1) {
1679 			pr_err(gettext("Unable to remove %s"), attrfile);
1680 			error = 39;
1681 		} else {
1682 			error = 0;
1683 			if (verbose) {
1684 				pr_err(gettext("Removed attrcache %s"),
1685 					attrfile);
1686 			}
1687 		}
1688 	}
1689 
1690 	if (msync(addrp, size, MS_SYNC) == -1) {
1691 		pr_err(gettext("Unable to sync %s"), attrfile);
1692 		error = 39;
1693 	}
1694 
1695 	/* unmap the attribute file */
1696 	if (addrp != MAP_FAILED)
1697 		munmap(addrp, size);
1698 
1699 	return (error);
1700 }
1701 
1702 /*
1703  *
1704  *			tree_remove
1705  *
1706  * Description:
1707  *	Called via the nftw64(3c) routine, this routine removes
1708  *	the specified file.
1709  * Arguments:
1710  *	namep	pathname to the file
1711  *	statp	stat info on the file
1712  *	type	ftw type information
1713  *	ftwp	pointer to additional ftw information
1714  * Returns:
1715  *	Returns 0 for success or -1 if an error occurs.
1716  * Preconditions:
1717  *	precond(namep)
1718  *	precond(statp)
1719  *	precond(ftwp)
1720  */
1721 
1722 int
1723 tree_remove(const char *namep, const struct stat64 *statp, int type,
1724     struct FTW *ftwp)
1725 {
1726 	int xx;
1727 
1728 	switch (type) {
1729 	case FTW_D:
1730 	case FTW_DP:
1731 	case FTW_DNR:
1732 		xx = rmdir(namep);
1733 		if (xx != 0) {
1734 			pr_err(gettext("Could not remove directory %s: %s"),
1735 			    namep, strerror(errno));
1736 			return (-1);
1737 		}
1738 #if 0
1739 		pr_err(gettext("Directory %s removed."), namep);
1740 #endif
1741 		break;
1742 
1743 	default:
1744 		xx = file_remove(namep, statp, S_verbose);
1745 #if 0
1746 		pr_err(gettext("File %s removed."), namep);
1747 #endif
1748 		break;
1749 	}
1750 
1751 	/* return success */
1752 	return (0);
1753 }
1754 
1755 /*
1756  *
1757  *			file_remove
1758  *
1759  * Description:
1760  *	Removes the specified file.
1761  *	If the file is a local file or has been modified locally
1762  *	then it is moved to lost+found.
1763  *	Should only be called for non-directory files.
1764  * Arguments:
1765  *	namep	pathname to the file
1766  *	statp	stat info on the file or NULL
1767  *	verbose	1 means be verbose about what is being removed
1768  * Returns:
1769  *	Returns 0 for success or -1 if an error occurs.
1770  * Preconditions:
1771  *	precond(namep)
1772  */
1773 
1774 int
1775 file_remove(const char *namep, const struct stat64 *statp, int verbose)
1776 {
1777 	int xx;
1778 	int ii;
1779 	struct stat64 statinfo;
1780 	int dolf = 0;
1781 	char newname[MAXPATHLEN * 2];
1782 	char *strp;
1783 
1784 	/* get stat info on the file if we were not passed it */
1785 	if (statp == NULL) {
1786 		xx = stat64(namep, &statinfo);
1787 		if (xx) {
1788 			if (verbose) {
1789 				pr_err(gettext("stat failed %s %d"),
1790 				    namep, errno);
1791 			}
1792 			return (-1);
1793 		}
1794 		statp = &statinfo;
1795 	}
1796 
1797 	/* ignore directories */
1798 	if (S_ISDIR(statp->st_mode)) {
1799 		errno = EINVAL;
1800 		return (-1);
1801 	}
1802 
1803 	/* if a local file then move to lost+found */
1804 	strp = strrchr(namep, '/');
1805 	if (strp == NULL) {
1806 		errno = EINVAL;
1807 		return (-1);
1808 	}
1809 	strp++;
1810 	if (*strp == 'L')
1811 		dolf = 1;
1812 
1813 	/* if a modified file then move to lost+found */
1814 	if ((statp->st_mode & S_IAMB) == 0766)
1815 		dolf = 1;
1816 
1817 	/* if moving to lost+found */
1818 	if (dolf) {
1819 		sprintf(newname, "%s/%s", S_lostfound, strp);
1820 		xx = stat64(newname, &statinfo);
1821 		for (ii = 1; ((ii < 1000) && (xx == 0)); ii++) {
1822 			sprintf(newname, "%s/%s_%d", S_lostfound, strp, ii);
1823 			xx = stat64(newname, &statinfo);
1824 		}
1825 		xx = rename(namep, newname);
1826 		if (xx) {
1827 			pr_err(gettext("Could not move file %s to %s: %s"),
1828 			    namep, newname, strerror(errno));
1829 			exit(-1);
1830 		}
1831 		S_move_lostfound = 1;
1832 		return (0);
1833 	}
1834 
1835 	/* remove the file */
1836 	xx = unlink(namep);
1837 	if (xx == -1) {
1838 		pr_err(gettext("Could not remove file %s: %s"),
1839 		    namep, strerror(errno));
1840 	} else if (verbose) {
1841 		pr_err(gettext("Removed %s"), namep);
1842 	}
1843 
1844 	return (0);
1845 }
1846 
1847 void
1848 cache_backmnt_cleanup(char *cachedirp, char *backmnt_namep)
1849 {
1850 	DIR *dirp;
1851 	struct dirent *entp;
1852 	char dirname[MAXPATHLEN * 2];
1853 
1854 	/* open the directory */
1855 	sprintf(dirname, "%s/%s", cachedirp, backmnt_namep);
1856 	dirp = opendir(dirname);
1857 	if (dirp == NULL)
1858 		return;
1859 
1860 	/*
1861 	 * Try to remove everything in the directory with rmdir.
1862 	 * Should only be empty directories in here at this point.
1863 	 * If not, do not worry about it.
1864 	 */
1865 	for (;;) {
1866 		/* get the next dir entry */
1867 		entp = readdir(dirp);
1868 		if (entp == NULL)
1869 			break;
1870 
1871 		/*
1872 		 * Try and remove the directory.
1873 		 * This will fail if there is anything in the dir,
1874 		 * like a mounted file system.
1875 		 */
1876 		rmdir(entp->d_name);
1877 	}
1878 	closedir(dirp);
1879 }
1880