Home | History | Annotate | Line # | Download | only in mtree
mtree.c revision 1.47
      1  1.47  christos /*	$NetBSD: mtree.c,v 1.47 2013/02/03 19:15:17 christos 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.47  christos __RCSID("$NetBSD: mtree.c,v 1.47 2013/02/03 19:15:17 christos 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.44  christos int	bflag, cflag, Cflag, dflag, Dflag, eflag, iflag, jflag, lflag, mflag,
     63  1.44  christos     	nflag, qflag, rflag, sflag, tflag, uflag, Uflag, wflag;
     64  1.25     lukem char	fullpath[MAXPATHLEN];
     65  1.44  christos 
     66  1.44  christos static struct {
     67  1.44  christos 	enum flavor flavor;
     68  1.44  christos 	const char name[9];
     69  1.44  christos } flavors[] = {
     70  1.44  christos 	{F_MTREE, "mtree"},
     71  1.44  christos 	{F_FREEBSD9, "freebsd9"},
     72  1.44  christos 	{F_NETBSD6, "netbsd6"},
     73  1.44  christos };
     74   1.3       cgd 
     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.45   mlelstv 	unsigned int	i;
     82  1.19     lukem 	char	*dir, *p;
     83  1.41  christos 	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.41  christos 	spec1 = stdin;
     90  1.41  christos 	spec2 = NULL;
     91  1.24     lukem 
     92  1.35       apb 	while ((ch = getopt(argc, argv,
     93  1.47  christos 	    "bcCdDeE:f:F:I:ijk:K:lLmMnN:O:p:PqrR:s:StuUwWxX:"))
     94  1.24     lukem 	    != -1) {
     95   1.1       cgd 		switch((char)ch) {
     96  1.44  christos 		case 'b':
     97  1.44  christos 			bflag = 1;
     98  1.44  christos 			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.41  christos 			if (spec1 == stdin) {
    119  1.41  christos 				spec1 = fopen(optarg, "r");
    120  1.41  christos 				if (spec1 == NULL)
    121  1.41  christos 					mtree_err("%s: %s", optarg,
    122  1.41  christos 					    strerror(errno));
    123  1.41  christos 			} else if (spec2 == NULL) {
    124  1.41  christos 				spec2 = fopen(optarg, "r");
    125  1.41  christos 				if (spec2 == NULL)
    126  1.41  christos 					mtree_err("%s: %s", optarg,
    127  1.41  christos 					    strerror(errno));
    128  1.41  christos 			} else
    129  1.41  christos 				usage();
    130   1.3       cgd 			break;
    131  1.44  christos 		case 'F':
    132  1.44  christos 			for (i = 0; i < __arraycount(flavors); i++)
    133  1.44  christos 				if (strcmp(optarg, flavors[i].name) == 0) {
    134  1.44  christos 					flavor = flavors[i].flavor;
    135  1.44  christos 					break;
    136  1.44  christos 				}
    137  1.44  christos 			if (i == __arraycount(flavors))
    138  1.44  christos 				usage();
    139  1.44  christos 			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.40  christos 		case 'j':
    147  1.40  christos 			jflag = 1;
    148  1.40  christos 			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.38  christos 		case 'n':
    174  1.38  christos 			nflag = 1;
    175  1.38  christos 			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.47  christos 		case 'O':
    183  1.47  christos 			load_only(optarg);
    184  1.47  christos 			break;
    185   1.1       cgd 		case 'p':
    186   1.1       cgd 			dir = optarg;
    187   1.1       cgd 			break;
    188  1.24     lukem 		case 'P':
    189  1.24     lukem 			ftsoptions &= ~FTS_LOGICAL;
    190  1.24     lukem 			ftsoptions |= FTS_PHYSICAL;
    191  1.24     lukem 			break;
    192  1.39  christos 		case 'q':
    193  1.39  christos 			qflag = 1;
    194  1.39  christos 			break;
    195   1.1       cgd 		case 'r':
    196   1.1       cgd 			rflag = 1;
    197   1.1       cgd 			break;
    198  1.18     lukem 		case 'R':
    199  1.18     lukem 			while ((p = strsep(&optarg, " \t,")) != NULL)
    200  1.18     lukem 				if (*p != '\0')
    201  1.18     lukem 					keys &= ~parsekey(p, NULL);
    202  1.18     lukem 			break;
    203   1.3       cgd 		case 's':
    204   1.3       cgd 			sflag = 1;
    205   1.3       cgd 			crc_total = ~strtol(optarg, &p, 0);
    206   1.3       cgd 			if (*p)
    207  1.10  wsanchez 				mtree_err("illegal seed value -- %s", optarg);
    208   1.7   thorpej 			break;
    209  1.35       apb 		case 'S':
    210  1.35       apb 			mtree_Sflag = 1;
    211  1.35       apb 			break;
    212   1.6   mycroft 		case 't':
    213   1.6   mycroft 			tflag = 1;
    214   1.6   mycroft 			break;
    215  1.24     lukem 		case 'u':
    216  1.24     lukem 			uflag = 1;
    217  1.24     lukem 			break;
    218   1.8       agc 		case 'U':
    219   1.8       agc 			Uflag = uflag = 1;
    220   1.8       agc 			break;
    221  1.44  christos 		case 'w':
    222  1.44  christos 			wflag = 1;
    223  1.44  christos 			break;
    224  1.22     lukem 		case 'W':
    225  1.32     lukem 			mtree_Wflag = 1;
    226  1.22     lukem 			break;
    227   1.1       cgd 		case 'x':
    228   1.1       cgd 			ftsoptions |= FTS_XDEV;
    229   1.1       cgd 			break;
    230  1.24     lukem 		case 'X':
    231  1.24     lukem 			read_excludes_file(optarg);
    232  1.24     lukem 			break;
    233   1.1       cgd 		case '?':
    234   1.1       cgd 		default:
    235   1.1       cgd 			usage();
    236   1.1       cgd 		}
    237  1.24     lukem 	}
    238   1.1       cgd 	argc -= optind;
    239   1.3       cgd 	argv += optind;
    240   1.3       cgd 
    241   1.1       cgd 	if (argc)
    242   1.1       cgd 		usage();
    243   1.1       cgd 
    244  1.44  christos 	switch (flavor) {
    245  1.44  christos 	case F_FREEBSD9:
    246  1.44  christos 		if (cflag && iflag) {
    247  1.44  christos 			warnx("-c and -i passed, replacing -i with -j for "
    248  1.44  christos 			    "FreeBSD compatibility");
    249  1.44  christos 			iflag = 0;
    250  1.44  christos 			jflag = 1;
    251  1.44  christos 		}
    252  1.44  christos 		if (dflag && !bflag) {
    253  1.44  christos 			warnx("Adding -b to -d for FreeBSD compatibility");
    254  1.44  christos 			bflag = 1;
    255  1.44  christos 		}
    256  1.44  christos 		if (uflag && !iflag) {
    257  1.44  christos 			warnx("Adding -i to -%c for FreeBSD compatibility",
    258  1.44  christos 			    Uflag ? 'U' : 'u');
    259  1.44  christos 			iflag = 1;
    260  1.44  christos 		}
    261  1.44  christos 		if (uflag && !tflag) {
    262  1.44  christos 			warnx("Adding -t to -%c for FreeBSD compatibility",
    263  1.44  christos 			    Uflag ? 'U' : 'u');
    264  1.44  christos 			tflag = 1;
    265  1.44  christos 		}
    266  1.44  christos 		if (wflag)
    267  1.44  christos 			warnx("The -w flag is a no-op");
    268  1.44  christos 		break;
    269  1.44  christos 	default:
    270  1.44  christos 		if (wflag)
    271  1.44  christos 			usage();
    272  1.44  christos 	}
    273  1.44  christos 
    274  1.41  christos 	if (spec2 && (cflag || Cflag || Dflag))
    275  1.41  christos 		mtree_err("Double -f, -c, -C and -D flags are mutually "
    276  1.41  christos 		    "exclusive");
    277  1.41  christos 
    278  1.41  christos 	if (dir && spec2)
    279  1.41  christos 		mtree_err("Double -f and -p flags are mutually exclusive");
    280  1.41  christos 
    281   1.3       cgd 	if (dir && chdir(dir))
    282  1.10  wsanchez 		mtree_err("%s: %s", dir, strerror(errno));
    283   1.1       cgd 
    284  1.25     lukem 	if ((cflag || sflag) && !getcwd(fullpath, sizeof(fullpath)))
    285  1.13     itohy 		mtree_err("%s", strerror(errno));
    286   1.1       cgd 
    287  1.29     lukem 	if ((cflag && Cflag) || (cflag && Dflag) || (Cflag && Dflag))
    288  1.29     lukem 		mtree_err("-c, -C and -D flags are mutually exclusive");
    289  1.18     lukem 
    290  1.29     lukem 	if (iflag && mflag)
    291  1.14       mrg 		mtree_err("-i and -m flags are mutually exclusive");
    292  1.14       mrg 
    293  1.29     lukem 	if (lflag && uflag)
    294  1.17     perry 		mtree_err("-l and -u flags are mutually exclusive");
    295  1.17     perry 
    296   1.3       cgd 	if (cflag) {
    297   1.1       cgd 		cwalk();
    298   1.3       cgd 		exit(0);
    299   1.3       cgd 	}
    300  1.29     lukem 	if (Cflag || Dflag) {
    301  1.41  christos 		dump_nodes("", spec(spec1), Dflag);
    302  1.18     lukem 		exit(0);
    303  1.18     lukem 	}
    304  1.41  christos 	if (spec2 != NULL)
    305  1.41  christos 		status = mtree_specspec(spec1, spec2);
    306  1.41  christos 	else
    307  1.41  christos 		status = verify(spec1);
    308  1.33     perry 	if (Uflag && (status == MISMATCHEXIT))
    309   1.8       agc 		status = 0;
    310   1.8       agc 	exit(status);
    311  1.19     lukem }
    312  1.19     lukem 
    313   1.3       cgd static void
    314  1.15    simonb usage(void)
    315   1.1       cgd {
    316  1.45   mlelstv 	unsigned int i;
    317  1.15    simonb 
    318  1.26     lukem 	fprintf(stderr,
    319  1.44  christos 	    "usage: %s [-bCcDdejLlMnPqrStUuWx] [-i|-m] [-E tags]\n"
    320  1.41  christos 	    "\t\t[-f spec] [-f spec]\n"
    321  1.36       wiz 	    "\t\t[-I tags] [-K keywords] [-k keywords] [-N dbdir] [-p path]\n"
    322  1.44  christos 	    "\t\t[-R keywords] [-s seed] [-X exclude-file]\n"
    323  1.44  christos 	    "\t\t[-F flavor]\n",
    324  1.26     lukem 	    getprogname());
    325  1.44  christos 	fprintf(stderr, "\nflavors:");
    326  1.44  christos 	for (i = 0; i < __arraycount(flavors); i++)
    327  1.44  christos 		fprintf(stderr, " %s", flavors[i].name);
    328  1.44  christos 	fprintf(stderr, "\n");
    329   1.1       cgd 	exit(1);
    330   1.1       cgd }
    331