Home | History | Annotate | Line # | Download | only in mtree
mtree.c revision 1.37.2.3
      1  1.37.2.3      yamt /*	$NetBSD: mtree.c,v 1.37.2.3 2013/01/23 00:06:43 yamt Exp $	*/
      2       1.5       cgd 
      3       1.1       cgd /*-
      4       1.4       cgd  * Copyright (c) 1989, 1990, 1993
      5       1.4       cgd  *	The Regents of the University of California.  All rights reserved.
      6       1.1       cgd  *
      7       1.1       cgd  * Redistribution and use in source and binary forms, with or without
      8       1.1       cgd  * modification, are permitted provided that the following conditions
      9       1.1       cgd  * are met:
     10       1.1       cgd  * 1. Redistributions of source code must retain the above copyright
     11       1.1       cgd  *    notice, this list of conditions and the following disclaimer.
     12       1.1       cgd  * 2. Redistributions in binary form must reproduce the above copyright
     13       1.1       cgd  *    notice, this list of conditions and the following disclaimer in the
     14       1.1       cgd  *    documentation and/or other materials provided with the distribution.
     15      1.30       agc  * 3. Neither the name of the University nor the names of its contributors
     16       1.1       cgd  *    may be used to endorse or promote products derived from this software
     17       1.1       cgd  *    without specific prior written permission.
     18       1.1       cgd  *
     19       1.1       cgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
     20       1.1       cgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     21       1.1       cgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     22       1.1       cgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
     23       1.1       cgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     24       1.1       cgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     25       1.1       cgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     26       1.1       cgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     27       1.1       cgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     28       1.1       cgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     29       1.1       cgd  * SUCH DAMAGE.
     30       1.1       cgd  */
     31       1.1       cgd 
     32      1.31       jmc #if HAVE_NBTOOL_CONFIG_H
     33      1.31       jmc #include "nbtool_config.h"
     34      1.31       jmc #endif
     35      1.31       jmc 
     36       1.9     lukem #include <sys/cdefs.h>
     37      1.28        tv #if defined(__COPYRIGHT) && !defined(lint)
     38      1.34     lukem __COPYRIGHT("@(#) Copyright (c) 1989, 1990, 1993\
     39      1.34     lukem  The Regents of the University of California.  All rights reserved.");
     40       1.1       cgd #endif /* not lint */
     41       1.1       cgd 
     42      1.28        tv #if defined(__RCSID) && !defined(lint)
     43       1.5       cgd #if 0
     44       1.4       cgd static char sccsid[] = "@(#)mtree.c	8.1 (Berkeley) 6/6/93";
     45       1.5       cgd #else
     46  1.37.2.3      yamt __RCSID("$NetBSD: mtree.c,v 1.37.2.3 2013/01/23 00:06:43 yamt Exp $");
     47       1.5       cgd #endif
     48       1.1       cgd #endif /* not lint */
     49       1.1       cgd 
     50       1.1       cgd #include <sys/param.h>
     51       1.1       cgd #include <sys/stat.h>
     52      1.15    simonb 
     53       1.1       cgd #include <errno.h>
     54      1.15    simonb #include <stdio.h>
     55      1.19     lukem #include <stdlib.h>
     56      1.21     lukem #include <string.h>
     57       1.3       cgd #include <unistd.h>
     58      1.15    simonb 
     59       1.3       cgd #include "extern.h"
     60       1.1       cgd 
     61      1.18     lukem int	ftsoptions = FTS_PHYSICAL;
     62  1.37.2.3      yamt int	bflag, cflag, Cflag, dflag, Dflag, eflag, iflag, jflag, lflag, mflag,
     63  1.37.2.3      yamt     	nflag, qflag, rflag, sflag, tflag, uflag, Uflag, wflag;
     64      1.25     lukem char	fullpath[MAXPATHLEN];
     65       1.3       cgd 
     66  1.37.2.3      yamt static struct {
     67  1.37.2.3      yamt 	enum flavor flavor;
     68  1.37.2.3      yamt 	const char name[9];
     69  1.37.2.3      yamt } flavors[] = {
     70  1.37.2.3      yamt 	{F_MTREE, "mtree"},
     71  1.37.2.3      yamt 	{F_FREEBSD9, "freebsd9"},
     72  1.37.2.3      yamt 	{F_NETBSD6, "netbsd6"},
     73  1.37.2.3      yamt };
     74  1.37.2.3      yamt 
     75      1.37     joerg __dead static	void	usage(void);
     76       1.3       cgd 
     77       1.3       cgd int
     78      1.15    simonb main(int argc, char **argv)
     79       1.1       cgd {
     80      1.20     lukem 	int	ch, status;
     81  1.37.2.3      yamt 	unsigned int	i;
     82      1.19     lukem 	char	*dir, *p;
     83  1.37.2.1      yamt 	FILE	*spec1, *spec2;
     84      1.16       cgd 
     85      1.16       cgd 	setprogname(argv[0]);
     86       1.1       cgd 
     87       1.3       cgd 	dir = NULL;
     88      1.24     lukem 	init_excludes();
     89  1.37.2.1      yamt 	spec1 = stdin;
     90  1.37.2.1      yamt 	spec2 = NULL;
     91      1.24     lukem 
     92      1.35       apb 	while ((ch = getopt(argc, argv,
     93  1.37.2.3      yamt 	    "bcCdDeE:f:F:I:ijk:K:lLmMnN:p:PqrR:s:StuUwWxX:"))
     94      1.24     lukem 	    != -1) {
     95       1.1       cgd 		switch((char)ch) {
     96  1.37.2.3      yamt 		case 'b':
     97  1.37.2.3      yamt 			bflag = 1;
     98  1.37.2.3      yamt 			break;
     99       1.1       cgd 		case 'c':
    100       1.1       cgd 			cflag = 1;
    101       1.1       cgd 			break;
    102      1.29     lukem 		case 'C':
    103      1.29     lukem 			Cflag = 1;
    104      1.29     lukem 			break;
    105       1.1       cgd 		case 'd':
    106       1.1       cgd 			dflag = 1;
    107       1.1       cgd 			break;
    108      1.18     lukem 		case 'D':
    109      1.18     lukem 			Dflag = 1;
    110      1.18     lukem 			break;
    111      1.19     lukem 		case 'E':
    112      1.20     lukem 			parsetags(&excludetags, optarg);
    113      1.19     lukem 			break;
    114       1.1       cgd 		case 'e':
    115       1.1       cgd 			eflag = 1;
    116       1.1       cgd 			break;
    117       1.1       cgd 		case 'f':
    118  1.37.2.1      yamt 			if (spec1 == stdin) {
    119  1.37.2.1      yamt 				spec1 = fopen(optarg, "r");
    120  1.37.2.1      yamt 				if (spec1 == NULL)
    121  1.37.2.1      yamt 					mtree_err("%s: %s", optarg,
    122  1.37.2.1      yamt 					    strerror(errno));
    123  1.37.2.1      yamt 			} else if (spec2 == NULL) {
    124  1.37.2.1      yamt 				spec2 = fopen(optarg, "r");
    125  1.37.2.1      yamt 				if (spec2 == NULL)
    126  1.37.2.1      yamt 					mtree_err("%s: %s", optarg,
    127  1.37.2.1      yamt 					    strerror(errno));
    128  1.37.2.1      yamt 			} else
    129  1.37.2.1      yamt 				usage();
    130       1.3       cgd 			break;
    131  1.37.2.3      yamt 		case 'F':
    132  1.37.2.3      yamt 			for (i = 0; i < __arraycount(flavors); i++)
    133  1.37.2.3      yamt 				if (strcmp(optarg, flavors[i].name) == 0) {
    134  1.37.2.3      yamt 					flavor = flavors[i].flavor;
    135  1.37.2.3      yamt 					break;
    136  1.37.2.3      yamt 				}
    137  1.37.2.3      yamt 			if (i == __arraycount(flavors))
    138  1.37.2.3      yamt 				usage();
    139  1.37.2.3      yamt 			break;
    140      1.24     lukem 		case 'i':
    141      1.24     lukem 			iflag = 1;
    142      1.24     lukem 			break;
    143      1.19     lukem 		case 'I':
    144      1.20     lukem 			parsetags(&includetags, optarg);
    145      1.19     lukem 			break;
    146  1.37.2.1      yamt 		case 'j':
    147  1.37.2.1      yamt 			jflag = 1;
    148  1.37.2.1      yamt 			break;
    149      1.24     lukem 		case 'k':
    150      1.24     lukem 			keys = F_TYPE;
    151       1.3       cgd 			while ((p = strsep(&optarg, " \t,")) != NULL)
    152       1.3       cgd 				if (*p != '\0')
    153       1.3       cgd 					keys |= parsekey(p, NULL);
    154       1.3       cgd 			break;
    155      1.24     lukem 		case 'K':
    156       1.3       cgd 			while ((p = strsep(&optarg, " \t,")) != NULL)
    157       1.3       cgd 				if (*p != '\0')
    158       1.3       cgd 					keys |= parsekey(p, NULL);
    159       1.1       cgd 			break;
    160      1.17     perry 		case 'l':
    161      1.17     perry 			lflag = 1;
    162      1.17     perry 			break;
    163      1.24     lukem 		case 'L':
    164      1.24     lukem 			ftsoptions &= ~FTS_PHYSICAL;
    165      1.24     lukem 			ftsoptions |= FTS_LOGICAL;
    166      1.24     lukem 			break;
    167      1.14       mrg 		case 'm':
    168      1.14       mrg 			mflag = 1;
    169      1.14       mrg 			break;
    170      1.32     lukem 		case 'M':
    171      1.32     lukem 			mtree_Mflag = 1;
    172      1.32     lukem 			break;
    173  1.37.2.1      yamt 		case 'n':
    174  1.37.2.1      yamt 			nflag = 1;
    175  1.37.2.1      yamt 			break;
    176      1.26     lukem 		case 'N':
    177      1.26     lukem 			if (! setup_getid(optarg))
    178      1.26     lukem 				mtree_err(
    179      1.26     lukem 			    "Unable to use user and group databases in `%s'",
    180      1.26     lukem 				    optarg);
    181      1.26     lukem 			break;
    182       1.1       cgd 		case 'p':
    183       1.1       cgd 			dir = optarg;
    184       1.1       cgd 			break;
    185      1.24     lukem 		case 'P':
    186      1.24     lukem 			ftsoptions &= ~FTS_LOGICAL;
    187      1.24     lukem 			ftsoptions |= FTS_PHYSICAL;
    188      1.24     lukem 			break;
    189  1.37.2.1      yamt 		case 'q':
    190  1.37.2.1      yamt 			qflag = 1;
    191  1.37.2.1      yamt 			break;
    192       1.1       cgd 		case 'r':
    193       1.1       cgd 			rflag = 1;
    194       1.1       cgd 			break;
    195      1.18     lukem 		case 'R':
    196      1.18     lukem 			while ((p = strsep(&optarg, " \t,")) != NULL)
    197      1.18     lukem 				if (*p != '\0')
    198      1.18     lukem 					keys &= ~parsekey(p, NULL);
    199      1.18     lukem 			break;
    200       1.3       cgd 		case 's':
    201       1.3       cgd 			sflag = 1;
    202       1.3       cgd 			crc_total = ~strtol(optarg, &p, 0);
    203       1.3       cgd 			if (*p)
    204      1.10  wsanchez 				mtree_err("illegal seed value -- %s", optarg);
    205       1.7   thorpej 			break;
    206      1.35       apb 		case 'S':
    207      1.35       apb 			mtree_Sflag = 1;
    208      1.35       apb 			break;
    209       1.6   mycroft 		case 't':
    210       1.6   mycroft 			tflag = 1;
    211       1.6   mycroft 			break;
    212      1.24     lukem 		case 'u':
    213      1.24     lukem 			uflag = 1;
    214      1.24     lukem 			break;
    215       1.8       agc 		case 'U':
    216       1.8       agc 			Uflag = uflag = 1;
    217       1.8       agc 			break;
    218  1.37.2.3      yamt 		case 'w':
    219  1.37.2.3      yamt 			wflag = 1;
    220  1.37.2.3      yamt 			break;
    221      1.22     lukem 		case 'W':
    222      1.32     lukem 			mtree_Wflag = 1;
    223      1.22     lukem 			break;
    224       1.1       cgd 		case 'x':
    225       1.1       cgd 			ftsoptions |= FTS_XDEV;
    226       1.1       cgd 			break;
    227      1.24     lukem 		case 'X':
    228      1.24     lukem 			read_excludes_file(optarg);
    229      1.24     lukem 			break;
    230       1.1       cgd 		case '?':
    231       1.1       cgd 		default:
    232       1.1       cgd 			usage();
    233       1.1       cgd 		}
    234      1.24     lukem 	}
    235       1.1       cgd 	argc -= optind;
    236       1.3       cgd 	argv += optind;
    237       1.3       cgd 
    238       1.1       cgd 	if (argc)
    239       1.1       cgd 		usage();
    240       1.1       cgd 
    241  1.37.2.3      yamt 	switch (flavor) {
    242  1.37.2.3      yamt 	case F_FREEBSD9:
    243  1.37.2.3      yamt 		if (cflag && iflag) {
    244  1.37.2.3      yamt 			warnx("-c and -i passed, replacing -i with -j for "
    245  1.37.2.3      yamt 			    "FreeBSD compatibility");
    246  1.37.2.3      yamt 			iflag = 0;
    247  1.37.2.3      yamt 			jflag = 1;
    248  1.37.2.3      yamt 		}
    249  1.37.2.3      yamt 		if (dflag && !bflag) {
    250  1.37.2.3      yamt 			warnx("Adding -b to -d for FreeBSD compatibility");
    251  1.37.2.3      yamt 			bflag = 1;
    252  1.37.2.3      yamt 		}
    253  1.37.2.3      yamt 		if (uflag && !iflag) {
    254  1.37.2.3      yamt 			warnx("Adding -i to -%c for FreeBSD compatibility",
    255  1.37.2.3      yamt 			    Uflag ? 'U' : 'u');
    256  1.37.2.3      yamt 			iflag = 1;
    257  1.37.2.3      yamt 		}
    258  1.37.2.3      yamt 		if (uflag && !tflag) {
    259  1.37.2.3      yamt 			warnx("Adding -t to -%c for FreeBSD compatibility",
    260  1.37.2.3      yamt 			    Uflag ? 'U' : 'u');
    261  1.37.2.3      yamt 			tflag = 1;
    262  1.37.2.3      yamt 		}
    263  1.37.2.3      yamt 		if (wflag)
    264  1.37.2.3      yamt 			warnx("The -w flag is a no-op");
    265  1.37.2.3      yamt 		break;
    266  1.37.2.3      yamt 	default:
    267  1.37.2.3      yamt 		if (wflag)
    268  1.37.2.3      yamt 			usage();
    269  1.37.2.3      yamt 	}
    270  1.37.2.3      yamt 
    271  1.37.2.1      yamt 	if (spec2 && (cflag || Cflag || Dflag))
    272  1.37.2.1      yamt 		mtree_err("Double -f, -c, -C and -D flags are mutually "
    273  1.37.2.1      yamt 		    "exclusive");
    274  1.37.2.1      yamt 
    275  1.37.2.1      yamt 	if (dir && spec2)
    276  1.37.2.1      yamt 		mtree_err("Double -f and -p flags are mutually exclusive");
    277  1.37.2.1      yamt 
    278       1.3       cgd 	if (dir && chdir(dir))
    279      1.10  wsanchez 		mtree_err("%s: %s", dir, strerror(errno));
    280       1.1       cgd 
    281      1.25     lukem 	if ((cflag || sflag) && !getcwd(fullpath, sizeof(fullpath)))
    282      1.13     itohy 		mtree_err("%s", strerror(errno));
    283       1.1       cgd 
    284      1.29     lukem 	if ((cflag && Cflag) || (cflag && Dflag) || (Cflag && Dflag))
    285      1.29     lukem 		mtree_err("-c, -C and -D flags are mutually exclusive");
    286      1.18     lukem 
    287      1.29     lukem 	if (iflag && mflag)
    288      1.14       mrg 		mtree_err("-i and -m flags are mutually exclusive");
    289      1.14       mrg 
    290      1.29     lukem 	if (lflag && uflag)
    291      1.17     perry 		mtree_err("-l and -u flags are mutually exclusive");
    292      1.17     perry 
    293       1.3       cgd 	if (cflag) {
    294       1.1       cgd 		cwalk();
    295       1.3       cgd 		exit(0);
    296       1.3       cgd 	}
    297      1.29     lukem 	if (Cflag || Dflag) {
    298  1.37.2.1      yamt 		dump_nodes("", spec(spec1), Dflag);
    299      1.18     lukem 		exit(0);
    300      1.18     lukem 	}
    301  1.37.2.1      yamt 	if (spec2 != NULL)
    302  1.37.2.1      yamt 		status = mtree_specspec(spec1, spec2);
    303  1.37.2.1      yamt 	else
    304  1.37.2.1      yamt 		status = verify(spec1);
    305      1.33     perry 	if (Uflag && (status == MISMATCHEXIT))
    306       1.8       agc 		status = 0;
    307       1.8       agc 	exit(status);
    308      1.19     lukem }
    309      1.19     lukem 
    310       1.3       cgd static void
    311      1.15    simonb usage(void)
    312       1.1       cgd {
    313  1.37.2.3      yamt 	unsigned int i;
    314      1.15    simonb 
    315      1.26     lukem 	fprintf(stderr,
    316  1.37.2.3      yamt 	    "usage: %s [-bCcDdejLlMnPqrStUuWx] [-i|-m] [-E tags]\n"
    317  1.37.2.1      yamt 	    "\t\t[-f spec] [-f spec]\n"
    318      1.36       wiz 	    "\t\t[-I tags] [-K keywords] [-k keywords] [-N dbdir] [-p path]\n"
    319  1.37.2.3      yamt 	    "\t\t[-R keywords] [-s seed] [-X exclude-file]\n"
    320  1.37.2.3      yamt 	    "\t\t[-F flavor]\n",
    321      1.26     lukem 	    getprogname());
    322  1.37.2.3      yamt 	fprintf(stderr, "\nflavors:");
    323  1.37.2.3      yamt 	for (i = 0; i < __arraycount(flavors); i++)
    324  1.37.2.3      yamt 		fprintf(stderr, " %s", flavors[i].name);
    325  1.37.2.3      yamt 	fprintf(stderr, "\n");
    326       1.1       cgd 	exit(1);
    327       1.1       cgd }
    328