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