Home | History | Annotate | Line # | Download | only in mtree
mtree.c revision 1.44
      1  1.44  christos /*	$NetBSD: mtree.c,v 1.44 2012/12/20 16:43:16 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.44  christos __RCSID("$NetBSD: mtree.c,v 1.44 2012/12/20 16:43:16 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 enum flavor	flavor = F_MTREE;
     66  1.44  christos 
     67  1.44  christos static struct {
     68  1.44  christos 	enum flavor flavor;
     69  1.44  christos 	const char name[9];
     70  1.44  christos } flavors[] = {
     71  1.44  christos 	{F_MTREE, "mtree"},
     72  1.44  christos 	{F_FREEBSD9, "freebsd9"},
     73  1.44  christos 	{F_NETBSD6, "netbsd6"},
     74  1.44  christos };
     75   1.3       cgd 
     76  1.37     joerg __dead static	void	usage(void);
     77   1.3       cgd 
     78   1.3       cgd int
     79  1.15    simonb main(int argc, char **argv)
     80   1.1       cgd {
     81  1.20     lukem 	int	ch, status;
     82  1.44  christos 	uint	i;
     83  1.19     lukem 	char	*dir, *p;
     84  1.41  christos 	FILE	*spec1, *spec2;
     85  1.16       cgd 
     86  1.16       cgd 	setprogname(argv[0]);
     87   1.1       cgd 
     88   1.3       cgd 	dir = NULL;
     89  1.24     lukem 	init_excludes();
     90  1.41  christos 	spec1 = stdin;
     91  1.41  christos 	spec2 = NULL;
     92  1.24     lukem 
     93  1.35       apb 	while ((ch = getopt(argc, argv,
     94  1.44  christos 	    "bcCdDeE:f:F:I:ijk:K:lLmMnN:p:PqrR:s:StuUwWxX:"))
     95  1.24     lukem 	    != -1) {
     96   1.1       cgd 		switch((char)ch) {
     97  1.44  christos 		case 'b':
     98  1.44  christos 			bflag = 1;
     99  1.44  christos 			break;
    100   1.1       cgd 		case 'c':
    101   1.1       cgd 			cflag = 1;
    102   1.1       cgd 			break;
    103  1.29     lukem 		case 'C':
    104  1.29     lukem 			Cflag = 1;
    105  1.29     lukem 			break;
    106   1.1       cgd 		case 'd':
    107   1.1       cgd 			dflag = 1;
    108   1.1       cgd 			break;
    109  1.18     lukem 		case 'D':
    110  1.18     lukem 			Dflag = 1;
    111  1.18     lukem 			break;
    112  1.19     lukem 		case 'E':
    113  1.20     lukem 			parsetags(&excludetags, optarg);
    114  1.19     lukem 			break;
    115   1.1       cgd 		case 'e':
    116   1.1       cgd 			eflag = 1;
    117   1.1       cgd 			break;
    118   1.1       cgd 		case 'f':
    119  1.41  christos 			if (spec1 == stdin) {
    120  1.41  christos 				spec1 = fopen(optarg, "r");
    121  1.41  christos 				if (spec1 == NULL)
    122  1.41  christos 					mtree_err("%s: %s", optarg,
    123  1.41  christos 					    strerror(errno));
    124  1.41  christos 			} else if (spec2 == NULL) {
    125  1.41  christos 				spec2 = fopen(optarg, "r");
    126  1.41  christos 				if (spec2 == NULL)
    127  1.41  christos 					mtree_err("%s: %s", optarg,
    128  1.41  christos 					    strerror(errno));
    129  1.41  christos 			} else
    130  1.41  christos 				usage();
    131   1.3       cgd 			break;
    132  1.44  christos 		case 'F':
    133  1.44  christos 			for (i = 0; i < __arraycount(flavors); i++)
    134  1.44  christos 				if (strcmp(optarg, flavors[i].name) == 0) {
    135  1.44  christos 					flavor = flavors[i].flavor;
    136  1.44  christos 					break;
    137  1.44  christos 				}
    138  1.44  christos 			if (i == __arraycount(flavors))
    139  1.44  christos 				usage();
    140  1.44  christos 			break;
    141  1.24     lukem 		case 'i':
    142  1.24     lukem 			iflag = 1;
    143  1.24     lukem 			break;
    144  1.19     lukem 		case 'I':
    145  1.20     lukem 			parsetags(&includetags, optarg);
    146  1.19     lukem 			break;
    147  1.40  christos 		case 'j':
    148  1.40  christos 			jflag = 1;
    149  1.40  christos 			break;
    150  1.24     lukem 		case 'k':
    151  1.24     lukem 			keys = F_TYPE;
    152   1.3       cgd 			while ((p = strsep(&optarg, " \t,")) != NULL)
    153   1.3       cgd 				if (*p != '\0')
    154   1.3       cgd 					keys |= parsekey(p, NULL);
    155   1.3       cgd 			break;
    156  1.24     lukem 		case 'K':
    157   1.3       cgd 			while ((p = strsep(&optarg, " \t,")) != NULL)
    158   1.3       cgd 				if (*p != '\0')
    159   1.3       cgd 					keys |= parsekey(p, NULL);
    160   1.1       cgd 			break;
    161  1.17     perry 		case 'l':
    162  1.17     perry 			lflag = 1;
    163  1.17     perry 			break;
    164  1.24     lukem 		case 'L':
    165  1.24     lukem 			ftsoptions &= ~FTS_PHYSICAL;
    166  1.24     lukem 			ftsoptions |= FTS_LOGICAL;
    167  1.24     lukem 			break;
    168  1.14       mrg 		case 'm':
    169  1.14       mrg 			mflag = 1;
    170  1.14       mrg 			break;
    171  1.32     lukem 		case 'M':
    172  1.32     lukem 			mtree_Mflag = 1;
    173  1.32     lukem 			break;
    174  1.38  christos 		case 'n':
    175  1.38  christos 			nflag = 1;
    176  1.38  christos 			break;
    177  1.26     lukem 		case 'N':
    178  1.26     lukem 			if (! setup_getid(optarg))
    179  1.26     lukem 				mtree_err(
    180  1.26     lukem 			    "Unable to use user and group databases in `%s'",
    181  1.26     lukem 				    optarg);
    182  1.26     lukem 			break;
    183   1.1       cgd 		case 'p':
    184   1.1       cgd 			dir = optarg;
    185   1.1       cgd 			break;
    186  1.24     lukem 		case 'P':
    187  1.24     lukem 			ftsoptions &= ~FTS_LOGICAL;
    188  1.24     lukem 			ftsoptions |= FTS_PHYSICAL;
    189  1.24     lukem 			break;
    190  1.39  christos 		case 'q':
    191  1.39  christos 			qflag = 1;
    192  1.39  christos 			break;
    193   1.1       cgd 		case 'r':
    194   1.1       cgd 			rflag = 1;
    195   1.1       cgd 			break;
    196  1.18     lukem 		case 'R':
    197  1.18     lukem 			while ((p = strsep(&optarg, " \t,")) != NULL)
    198  1.18     lukem 				if (*p != '\0')
    199  1.18     lukem 					keys &= ~parsekey(p, NULL);
    200  1.18     lukem 			break;
    201   1.3       cgd 		case 's':
    202   1.3       cgd 			sflag = 1;
    203   1.3       cgd 			crc_total = ~strtol(optarg, &p, 0);
    204   1.3       cgd 			if (*p)
    205  1.10  wsanchez 				mtree_err("illegal seed value -- %s", optarg);
    206   1.7   thorpej 			break;
    207  1.35       apb 		case 'S':
    208  1.35       apb 			mtree_Sflag = 1;
    209  1.35       apb 			break;
    210   1.6   mycroft 		case 't':
    211   1.6   mycroft 			tflag = 1;
    212   1.6   mycroft 			break;
    213  1.24     lukem 		case 'u':
    214  1.24     lukem 			uflag = 1;
    215  1.24     lukem 			break;
    216   1.8       agc 		case 'U':
    217   1.8       agc 			Uflag = uflag = 1;
    218   1.8       agc 			break;
    219  1.44  christos 		case 'w':
    220  1.44  christos 			wflag = 1;
    221  1.44  christos 			break;
    222  1.22     lukem 		case 'W':
    223  1.32     lukem 			mtree_Wflag = 1;
    224  1.22     lukem 			break;
    225   1.1       cgd 		case 'x':
    226   1.1       cgd 			ftsoptions |= FTS_XDEV;
    227   1.1       cgd 			break;
    228  1.24     lukem 		case 'X':
    229  1.24     lukem 			read_excludes_file(optarg);
    230  1.24     lukem 			break;
    231   1.1       cgd 		case '?':
    232   1.1       cgd 		default:
    233   1.1       cgd 			usage();
    234   1.1       cgd 		}
    235  1.24     lukem 	}
    236   1.1       cgd 	argc -= optind;
    237   1.3       cgd 	argv += optind;
    238   1.3       cgd 
    239   1.1       cgd 	if (argc)
    240   1.1       cgd 		usage();
    241   1.1       cgd 
    242  1.44  christos 	switch (flavor) {
    243  1.44  christos 	case F_FREEBSD9:
    244  1.44  christos 		if (cflag && iflag) {
    245  1.44  christos 			warnx("-c and -i passed, replacing -i with -j for "
    246  1.44  christos 			    "FreeBSD compatibility");
    247  1.44  christos 			iflag = 0;
    248  1.44  christos 			jflag = 1;
    249  1.44  christos 		}
    250  1.44  christos 		if (dflag && !bflag) {
    251  1.44  christos 			warnx("Adding -b to -d for FreeBSD compatibility");
    252  1.44  christos 			bflag = 1;
    253  1.44  christos 		}
    254  1.44  christos 		if (uflag && !iflag) {
    255  1.44  christos 			warnx("Adding -i to -%c for FreeBSD compatibility",
    256  1.44  christos 			    Uflag ? 'U' : 'u');
    257  1.44  christos 			iflag = 1;
    258  1.44  christos 		}
    259  1.44  christos 		if (uflag && !tflag) {
    260  1.44  christos 			warnx("Adding -t to -%c for FreeBSD compatibility",
    261  1.44  christos 			    Uflag ? 'U' : 'u');
    262  1.44  christos 			tflag = 1;
    263  1.44  christos 		}
    264  1.44  christos 		if (wflag)
    265  1.44  christos 			warnx("The -w flag is a no-op");
    266  1.44  christos 		break;
    267  1.44  christos 	default:
    268  1.44  christos 		if (wflag)
    269  1.44  christos 			usage();
    270  1.44  christos 	}
    271  1.44  christos 
    272  1.41  christos 	if (spec2 && (cflag || Cflag || Dflag))
    273  1.41  christos 		mtree_err("Double -f, -c, -C and -D flags are mutually "
    274  1.41  christos 		    "exclusive");
    275  1.41  christos 
    276  1.41  christos 	if (dir && spec2)
    277  1.41  christos 		mtree_err("Double -f and -p flags are mutually exclusive");
    278  1.41  christos 
    279   1.3       cgd 	if (dir && chdir(dir))
    280  1.10  wsanchez 		mtree_err("%s: %s", dir, strerror(errno));
    281   1.1       cgd 
    282  1.25     lukem 	if ((cflag || sflag) && !getcwd(fullpath, sizeof(fullpath)))
    283  1.13     itohy 		mtree_err("%s", strerror(errno));
    284   1.1       cgd 
    285  1.29     lukem 	if ((cflag && Cflag) || (cflag && Dflag) || (Cflag && Dflag))
    286  1.29     lukem 		mtree_err("-c, -C and -D flags are mutually exclusive");
    287  1.18     lukem 
    288  1.29     lukem 	if (iflag && mflag)
    289  1.14       mrg 		mtree_err("-i and -m flags are mutually exclusive");
    290  1.14       mrg 
    291  1.29     lukem 	if (lflag && uflag)
    292  1.17     perry 		mtree_err("-l and -u flags are mutually exclusive");
    293  1.17     perry 
    294   1.3       cgd 	if (cflag) {
    295   1.1       cgd 		cwalk();
    296   1.3       cgd 		exit(0);
    297   1.3       cgd 	}
    298  1.29     lukem 	if (Cflag || Dflag) {
    299  1.41  christos 		dump_nodes("", spec(spec1), Dflag);
    300  1.18     lukem 		exit(0);
    301  1.18     lukem 	}
    302  1.41  christos 	if (spec2 != NULL)
    303  1.41  christos 		status = mtree_specspec(spec1, spec2);
    304  1.41  christos 	else
    305  1.41  christos 		status = verify(spec1);
    306  1.33     perry 	if (Uflag && (status == MISMATCHEXIT))
    307   1.8       agc 		status = 0;
    308   1.8       agc 	exit(status);
    309  1.19     lukem }
    310  1.19     lukem 
    311   1.3       cgd static void
    312  1.15    simonb usage(void)
    313   1.1       cgd {
    314  1.44  christos 	uint i;
    315  1.15    simonb 
    316  1.26     lukem 	fprintf(stderr,
    317  1.44  christos 	    "usage: %s [-bCcDdejLlMnPqrStUuWx] [-i|-m] [-E tags]\n"
    318  1.41  christos 	    "\t\t[-f spec] [-f spec]\n"
    319  1.36       wiz 	    "\t\t[-I tags] [-K keywords] [-k keywords] [-N dbdir] [-p path]\n"
    320  1.44  christos 	    "\t\t[-R keywords] [-s seed] [-X exclude-file]\n"
    321  1.44  christos 	    "\t\t[-F flavor]\n",
    322  1.26     lukem 	    getprogname());
    323  1.44  christos 	fprintf(stderr, "\nflavors:");
    324  1.44  christos 	for (i = 0; i < __arraycount(flavors); i++)
    325  1.44  christos 		fprintf(stderr, " %s", flavors[i].name);
    326  1.44  christos 	fprintf(stderr, "\n");
    327   1.1       cgd 	exit(1);
    328   1.1       cgd }
    329