]> git.saurik.com Git - apple/shell_cmds.git/blob - find/function.c
01e45eaac107abbf91b47091c24a9d966eba2dc6
[apple/shell_cmds.git] / find / function.c
1 /*-
2 * Copyright (c) 1990, 1993
3 * The Regents of the University of California. All rights reserved.
4 *
5 * This code is derived from software contributed to Berkeley by
6 * Cimarron D. Taylor of the University of California, Berkeley.
7 *
8 * Redistribution and use in source and binary forms, with or without
9 * modification, are permitted provided that the following conditions
10 * are met:
11 * 1. Redistributions of source code must retain the above copyright
12 * notice, this list of conditions and the following disclaimer.
13 * 2. Redistributions in binary form must reproduce the above copyright
14 * notice, this list of conditions and the following disclaimer in the
15 * documentation and/or other materials provided with the distribution.
16 * 3. All advertising materials mentioning features or use of this software
17 * must display the following acknowledgement:
18 * This product includes software developed by the University of
19 * California, Berkeley and its contributors.
20 * 4. Neither the name of the University nor the names of its contributors
21 * may be used to endorse or promote products derived from this software
22 * without specific prior written permission.
23 *
24 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
25 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
26 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34 * SUCH DAMAGE.
35 */
36
37 #ifndef lint
38 #if 0
39 static const char sccsid[] = "@(#)function.c 8.10 (Berkeley) 5/4/95";
40 #endif
41 #endif /* not lint */
42
43 #include <sys/cdefs.h>
44 __FBSDID("$FreeBSD: src/usr.bin/find/function.c,v 1.63 2009/09/04 20:01:16 trasz Exp $");
45
46 #include <sys/param.h>
47 #include <sys/ucred.h>
48 #include <sys/stat.h>
49 #include <sys/types.h>
50 #include <sys/acl.h>
51 #include <sys/wait.h>
52 #include <sys/mount.h>
53 #include <sys/timeb.h>
54 #ifdef __APPLE__
55 #include <sys/xattr.h>
56 #endif /* __APPLE__ */
57
58 #include <dirent.h>
59 #include <err.h>
60 #include <errno.h>
61 #include <fnmatch.h>
62 #include <fts.h>
63 #include <grp.h>
64 #include <limits.h>
65 #include <pwd.h>
66 #include <regex.h>
67 #include <stdio.h>
68 #include <stdlib.h>
69 #include <string.h>
70 #include <unistd.h>
71 #include <ctype.h>
72
73 #include "find.h"
74
75 #ifdef __APPLE__
76 #include <sys/sysctl.h>
77 #include <libgen.h>
78 #include <get_compat.h>
79 #else
80 #define COMPAT_MODE(func, mode) 1
81 #endif
82
83 static PLAN *palloc(OPTION *);
84 static long long find_parsenum(PLAN *, const char *, char *, char *);
85 static long long find_parsetime(PLAN *, const char *, char *);
86 static char *nextarg(OPTION *, char ***);
87
88 extern char **environ;
89
90 static PLAN *lastexecplus = NULL;
91
92 #define COMPARE(a, b) do { \
93 switch (plan->flags & F_ELG_MASK) { \
94 case F_EQUAL: \
95 return (a == b); \
96 case F_LESSTHAN: \
97 return (a < b); \
98 case F_GREATER: \
99 return (a > b); \
100 default: \
101 abort(); \
102 } \
103 } while(0)
104
105 static PLAN *
106 palloc(OPTION *option)
107 {
108 PLAN *new;
109
110 if ((new = malloc(sizeof(PLAN))) == NULL)
111 err(1, NULL);
112 new->execute = option->execute;
113 new->flags = option->flags;
114 new->next = NULL;
115 return new;
116 }
117
118 /*
119 * find_parsenum --
120 * Parse a string of the form [+-]# and return the value.
121 */
122 static long long
123 find_parsenum(PLAN *plan, const char *option, char *vp, char *endch)
124 {
125 long long value;
126 char *endchar, *str; /* Pointer to character ending conversion. */
127
128 /* Determine comparison from leading + or -. */
129 str = vp;
130 switch (*str) {
131 case '+':
132 ++str;
133 plan->flags |= F_GREATER;
134 break;
135 case '-':
136 ++str;
137 plan->flags |= F_LESSTHAN;
138 break;
139 default:
140 plan->flags |= F_EQUAL;
141 break;
142 }
143
144 /*
145 * Convert the string with strtoq(). Note, if strtoq() returns zero
146 * and endchar points to the beginning of the string we know we have
147 * a syntax error.
148 */
149 value = strtoq(str, &endchar, 10);
150 if (value == 0 && endchar == str)
151 errx(1, "%s: %s: illegal numeric value", option, vp);
152 if (endchar[0] && endch == NULL)
153 errx(1, "%s: %s: illegal trailing character", option, vp);
154 if (endch)
155 *endch = endchar[0];
156 return value;
157 }
158
159 /*
160 * find_parsetime --
161 * Parse a string of the form [+-]([0-9]+[smhdw]?)+ and return the value.
162 */
163 static long long
164 find_parsetime(PLAN *plan, const char *option, char *vp)
165 {
166 long long secs, value;
167 char *str, *unit; /* Pointer to character ending conversion. */
168
169 /* Determine comparison from leading + or -. */
170 str = vp;
171 switch (*str) {
172 case '+':
173 ++str;
174 plan->flags |= F_GREATER;
175 break;
176 case '-':
177 ++str;
178 plan->flags |= F_LESSTHAN;
179 break;
180 default:
181 plan->flags |= F_EQUAL;
182 break;
183 }
184
185 value = strtoq(str, &unit, 10);
186 if (value == 0 && unit == str) {
187 errx(1, "%s: %s: illegal time value", option, vp);
188 /* NOTREACHED */
189 }
190 if (*unit == '\0')
191 return value;
192
193 /* Units syntax. */
194 secs = 0;
195 for (;;) {
196 switch(*unit) {
197 case 's': /* seconds */
198 secs += value;
199 break;
200 case 'm': /* minutes */
201 secs += value * 60;
202 break;
203 case 'h': /* hours */
204 secs += value * 3600;
205 break;
206 case 'd': /* days */
207 secs += value * 86400;
208 break;
209 case 'w': /* weeks */
210 secs += value * 604800;
211 break;
212 default:
213 errx(1, "%s: %s: bad unit '%c'", option, vp, *unit);
214 /* NOTREACHED */
215 }
216 str = unit + 1;
217 if (*str == '\0') /* EOS */
218 break;
219 value = strtoq(str, &unit, 10);
220 if (value == 0 && unit == str) {
221 errx(1, "%s: %s: illegal time value", option, vp);
222 /* NOTREACHED */
223 }
224 if (*unit == '\0') {
225 errx(1, "%s: %s: missing trailing unit", option, vp);
226 /* NOTREACHED */
227 }
228 }
229 plan->flags |= F_EXACTTIME;
230 return secs;
231 }
232
233 /*
234 * nextarg --
235 * Check that another argument still exists, return a pointer to it,
236 * and increment the argument vector pointer.
237 */
238 static char *
239 nextarg(OPTION *option, char ***argvp)
240 {
241 char *arg;
242
243 if ((arg = **argvp) == 0)
244 errx(1, "%s: requires additional arguments", option->name);
245 (*argvp)++;
246 return arg;
247 } /* nextarg() */
248
249 /*
250 * The value of n for the inode times (atime, birthtime, ctime, mtime) is a
251 * range, i.e. n matches from (n - 1) to n 24 hour periods. This interacts
252 * with -n, such that "-mtime -1" would be less than 0 days, which isn't what
253 * the user wanted. Correct so that -1 is "less than 1".
254 */
255 #define TIME_CORRECT(p) \
256 if (((p)->flags & F_ELG_MASK) == F_LESSTHAN) \
257 ++((p)->t_data);
258
259 /*
260 * -[acm]min n functions --
261 *
262 * True if the difference between the
263 * file access time (-amin)
264 * file birth time (-Bmin)
265 * last change of file status information (-cmin)
266 * file modification time (-mmin)
267 * and the current time is n min periods.
268 */
269 int
270 f_Xmin(PLAN *plan, FTSENT *entry)
271 {
272 if (plan->flags & F_TIME_C) {
273 COMPARE((now - entry->fts_statp->st_ctime +
274 60 - 1) / 60, plan->t_data);
275 } else if (plan->flags & F_TIME_A) {
276 COMPARE((now - entry->fts_statp->st_atime +
277 60 - 1) / 60, plan->t_data);
278 } else if (plan->flags & F_TIME_B) {
279 COMPARE((now - entry->fts_statp->st_birthtime +
280 60 - 1) / 60, plan->t_data);
281 } else {
282 COMPARE((now - entry->fts_statp->st_mtime +
283 60 - 1) / 60, plan->t_data);
284 }
285 }
286
287 PLAN *
288 c_Xmin(OPTION *option, char ***argvp)
289 {
290 char *nmins;
291 PLAN *new;
292
293 nmins = nextarg(option, argvp);
294 ftsoptions &= ~FTS_NOSTAT;
295
296 new = palloc(option);
297 new->t_data = find_parsenum(new, option->name, nmins, NULL);
298 TIME_CORRECT(new);
299 return new;
300 }
301
302 /*
303 * -[acm]time n functions --
304 *
305 * True if the difference between the
306 * file access time (-atime)
307 * file birth time (-Btime)
308 * last change of file status information (-ctime)
309 * file modification time (-mtime)
310 * and the current time is n 24 hour periods.
311 */
312
313 int
314 f_Xtime(PLAN *plan, FTSENT *entry)
315 {
316 time_t xtime;
317
318 if (plan->flags & F_TIME_A)
319 xtime = entry->fts_statp->st_atime;
320 else if (plan->flags & F_TIME_B)
321 xtime = entry->fts_statp->st_birthtime;
322 else if (plan->flags & F_TIME_C)
323 xtime = entry->fts_statp->st_ctime;
324 else
325 xtime = entry->fts_statp->st_mtime;
326
327 if (plan->flags & F_EXACTTIME)
328 COMPARE(now - xtime, plan->t_data);
329 else
330 COMPARE((now - xtime + (COMPAT_MODE("bin/find", "unix2003") ? 0 : 86400 - 1)) / 86400, plan->t_data);
331 }
332
333 PLAN *
334 c_Xtime(OPTION *option, char ***argvp)
335 {
336 char *value;
337 PLAN *new;
338
339 value = nextarg(option, argvp);
340 ftsoptions &= ~FTS_NOSTAT;
341
342 new = palloc(option);
343 new->t_data = find_parsetime(new, option->name, value);
344 if (!(new->flags & F_EXACTTIME) && !COMPAT_MODE("bin/find", "unix2003"))
345 TIME_CORRECT(new);
346 return new;
347 }
348
349 /*
350 * -maxdepth/-mindepth n functions --
351 *
352 * Does the same as -prune if the level of the current file is
353 * greater/less than the specified maximum/minimum depth.
354 *
355 * Note that -maxdepth and -mindepth are handled specially in
356 * find_execute() so their f_* functions are set to f_always_true().
357 */
358 PLAN *
359 c_mXXdepth(OPTION *option, char ***argvp)
360 {
361 char *dstr;
362 PLAN *new;
363
364 dstr = nextarg(option, argvp);
365 if (dstr[0] == '-')
366 /* all other errors handled by find_parsenum() */
367 errx(1, "%s: %s: value must be positive", option->name, dstr);
368
369 new = palloc(option);
370 if (option->flags & F_MAXDEPTH)
371 maxdepth = find_parsenum(new, option->name, dstr, NULL);
372 else
373 mindepth = find_parsenum(new, option->name, dstr, NULL);
374 return new;
375 }
376
377 /*
378 * -acl function --
379 *
380 * Show files with EXTENDED ACL attributes.
381 */
382 #ifdef __APPLE__
383 int
384 f_acl(PLAN *plan __unused, FTSENT *entry)
385 {
386 acl_t facl;
387 int match;
388 acl_entry_t ae;
389
390 match = 0;
391 if ((facl = acl_get_link_np(entry->fts_accpath, ACL_TYPE_EXTENDED)) != NULL) {
392 if (acl_get_entry(facl, ACL_FIRST_ENTRY, &ae) == 0) {
393 match = 1;
394 }
395 acl_free(facl);
396 }
397 return match;
398 }
399 #else /* !__APPLE__ */
400 int
401 f_acl(PLAN *plan __unused, FTSENT *entry)
402 {
403 acl_t facl;
404 acl_type_t acl_type;
405 int acl_supported = 0, ret, trivial;
406
407 if (S_ISLNK(entry->fts_statp->st_mode))
408 return 0;
409 ret = pathconf(entry->fts_accpath, _PC_ACL_NFS4);
410 if (ret > 0) {
411 acl_supported = 1;
412 acl_type = ACL_TYPE_NFS4;
413 } else if (ret < 0 && errno != EINVAL) {
414 warn("%s", entry->fts_accpath);
415 return (0);
416 }
417 if (acl_supported == 0) {
418 ret = pathconf(entry->fts_accpath, _PC_ACL_EXTENDED);
419 if (ret > 0) {
420 acl_supported = 1;
421 acl_type = ACL_TYPE_ACCESS;
422 } else if (ret < 0 && errno != EINVAL) {
423 warn("%s", entry->fts_accpath);
424 return (0);
425 }
426 }
427 if (acl_supported == 0)
428 return (0);
429
430 facl = acl_get_file(entry->fts_accpath, acl_type);
431 if (facl == NULL) {
432 warn("%s", entry->fts_accpath);
433 return (0);
434 }
435 ret = acl_is_trivial_np(facl, &trivial);
436 acl_free(facl);
437 if (ret) {
438 warn("%s", entry->fts_accpath);
439 acl_free(facl);
440 return (0);
441 }
442 if (trivial)
443 return (0);
444 return (1);
445 }
446 #endif /* __APPLE__ */
447
448 PLAN *
449 c_acl(OPTION *option, char ***argvp __unused)
450 {
451 #ifndef __APPLE__
452 ftsoptions &= ~FTS_NOSTAT;
453 #endif /* !__APPLE__ */
454 return (palloc(option));
455 }
456
457 #ifdef __APPLE__
458 int
459 f_xattr(PLAN *plan __unused, FTSENT *entry)
460 {
461 ssize_t xattr;
462 int match;
463
464 match = 0;
465 xattr = listxattr(entry->fts_accpath, NULL, 0, XATTR_NOFOLLOW);
466 if (xattr > 0) {
467 match = 1;
468 }
469 return match;
470 }
471
472 int
473 f_xattrname(PLAN *plan, FTSENT *entry)
474 {
475 ssize_t xattr;
476 int match;
477
478 match = 0;
479 xattr = getxattr(entry->fts_accpath, plan->c_data, NULL, 0, 0, XATTR_NOFOLLOW);
480 if (xattr > 0) {
481 match = 1;
482 }
483 return match;
484 }
485 #endif /* __APPLE__ */
486
487 /*
488 * -delete functions --
489 *
490 * True always. Makes its best shot and continues on regardless.
491 */
492 int
493 f_delete(PLAN *plan __unused, FTSENT *entry)
494 {
495 /* ignore these from fts */
496 if (strcmp(entry->fts_accpath, ".") == 0 ||
497 strcmp(entry->fts_accpath, "..") == 0)
498 return 1;
499
500 /* sanity check */
501 if (isdepth == 0 || /* depth off */
502 (ftsoptions & FTS_NOSTAT)) /* not stat()ing */
503 errx(1, "-delete: insecure options got turned on");
504
505 if (!(ftsoptions & FTS_PHYSICAL) || /* physical off */
506 (ftsoptions & FTS_LOGICAL)) /* or finally, logical on */
507 errx(1, "-delete: forbidden when symlinks are followed");
508
509 /* Potentially unsafe - do not accept relative paths whatsoever */
510 if (strchr(entry->fts_accpath, '/') != NULL)
511 errx(1, "-delete: %s: relative path potentially not safe",
512 entry->fts_accpath);
513
514 /* Turn off user immutable bits if running as root */
515 if ((entry->fts_statp->st_flags & (UF_APPEND|UF_IMMUTABLE)) &&
516 !(entry->fts_statp->st_flags & (SF_APPEND|SF_IMMUTABLE)) &&
517 geteuid() == 0)
518 lchflags(entry->fts_accpath,
519 entry->fts_statp->st_flags &= ~(UF_APPEND|UF_IMMUTABLE));
520
521 /* rmdir directories, unlink everything else */
522 if (S_ISDIR(entry->fts_statp->st_mode)) {
523 if (rmdir(entry->fts_accpath) < 0 && errno != ENOTEMPTY)
524 warn("-delete: rmdir(%s)", entry->fts_path);
525 } else {
526 if (unlink(entry->fts_accpath) < 0)
527 warn("-delete: unlink(%s)", entry->fts_path);
528 }
529
530 /* "succeed" */
531 return 1;
532 }
533
534 PLAN *
535 c_delete(OPTION *option, char ***argvp __unused)
536 {
537
538 ftsoptions &= ~FTS_NOSTAT; /* no optimise */
539 isoutput = 1; /* possible output */
540 isdepth = 1; /* -depth implied */
541
542 return palloc(option);
543 }
544
545
546 /*
547 * always_true --
548 *
549 * Always true, used for -maxdepth, -mindepth, -xdev, -follow, and -true
550 */
551 int
552 f_always_true(PLAN *plan __unused, FTSENT *entry __unused)
553 {
554 return 1;
555 }
556
557 /*
558 * -depth functions --
559 *
560 * With argument: True if the file is at level n.
561 * Without argument: Always true, causes descent of the directory hierarchy
562 * to be done so that all entries in a directory are acted on before the
563 * directory itself.
564 */
565 int
566 f_depth(PLAN *plan, FTSENT *entry)
567 {
568 if (plan->flags & F_DEPTH)
569 COMPARE(entry->fts_level, plan->d_data);
570 else
571 return 1;
572 }
573
574 PLAN *
575 c_depth(OPTION *option, char ***argvp)
576 {
577 PLAN *new;
578 char *str;
579
580 new = palloc(option);
581
582 str = **argvp;
583 if (str && !(new->flags & F_DEPTH)) {
584 /* skip leading + or - */
585 if (*str == '+' || *str == '-')
586 str++;
587 /* skip sign */
588 if (*str == '+' || *str == '-')
589 str++;
590 if (isdigit(*str))
591 new->flags |= F_DEPTH;
592 }
593
594 if (new->flags & F_DEPTH) { /* -depth n */
595 char *ndepth;
596
597 ndepth = nextarg(option, argvp);
598 new->d_data = find_parsenum(new, option->name, ndepth, NULL);
599 } else { /* -d */
600 isdepth = 1;
601 }
602
603 return new;
604 }
605
606 /*
607 * -empty functions --
608 *
609 * True if the file or directory is empty
610 */
611 int
612 f_empty(PLAN *plan __unused, FTSENT *entry)
613 {
614 if (S_ISREG(entry->fts_statp->st_mode) &&
615 entry->fts_statp->st_size == 0)
616 return 1;
617 if (S_ISDIR(entry->fts_statp->st_mode)) {
618 struct dirent *dp;
619 int empty;
620 DIR *dir;
621
622 empty = 1;
623 dir = opendir(entry->fts_accpath);
624 if (dir == NULL)
625 err(1, "%s", entry->fts_accpath);
626 for (dp = readdir(dir); dp; dp = readdir(dir))
627 if (dp->d_name[0] != '.' ||
628 (dp->d_name[1] != '\0' &&
629 (dp->d_name[1] != '.' || dp->d_name[2] != '\0'))) {
630 empty = 0;
631 break;
632 }
633 closedir(dir);
634 return empty;
635 }
636 return 0;
637 }
638
639 PLAN *
640 c_empty(OPTION *option, char ***argvp __unused)
641 {
642 ftsoptions &= ~FTS_NOSTAT;
643
644 return palloc(option);
645 }
646
647 /*
648 * [-exec | -execdir | -ok] utility [arg ... ] ; functions --
649 *
650 * True if the executed utility returns a zero value as exit status.
651 * The end of the primary expression is delimited by a semicolon. If
652 * "{}" occurs anywhere, it gets replaced by the current pathname,
653 * or, in the case of -execdir, the current basename (filename
654 * without leading directory prefix). For -exec and -ok,
655 * the current directory for the execution of utility is the same as
656 * the current directory when the find utility was started, whereas
657 * for -execdir, it is the directory the file resides in.
658 *
659 * The primary -ok differs from -exec in that it requests affirmation
660 * of the user before executing the utility.
661 */
662 int
663 f_exec(PLAN *plan, FTSENT *entry)
664 {
665 int cnt;
666 pid_t pid;
667 int status;
668 char *file;
669
670 if (entry == NULL && plan->flags & F_EXECPLUS) {
671 if (plan->e_ppos == plan->e_pbnum)
672 return (1);
673 plan->e_argv[plan->e_ppos] = NULL;
674 goto doexec;
675 }
676
677 /* XXX - if file/dir ends in '/' this will not work -- can it? */
678 if ((plan->flags & F_EXECDIR) && \
679 (file = strrchr(entry->fts_path, '/')))
680 file++;
681 else
682 file = entry->fts_path;
683
684 if (plan->flags & F_EXECPLUS) {
685 if ((plan->e_argv[plan->e_ppos] = strdup(file)) == NULL)
686 err(1, NULL);
687 plan->e_len[plan->e_ppos] = strlen(file);
688 plan->e_psize += plan->e_len[plan->e_ppos];
689 if (++plan->e_ppos < plan->e_pnummax &&
690 plan->e_psize < plan->e_psizemax)
691 return (1);
692 plan->e_argv[plan->e_ppos] = NULL;
693 } else {
694 for (cnt = 0; plan->e_argv[cnt]; ++cnt)
695 if (plan->e_len[cnt])
696 brace_subst(plan->e_orig[cnt],
697 &plan->e_argv[cnt], file,
698 plan->e_len[cnt]);
699 }
700
701 doexec: if ((plan->flags & F_NEEDOK) && !queryuser(plan->e_argv))
702 return 0;
703
704 /* make sure find output is interspersed correctly with subprocesses */
705 fflush(stdout);
706 fflush(stderr);
707
708 switch (pid = fork()) {
709 case -1:
710 err(1, "fork");
711 /* NOTREACHED */
712 case 0:
713 /* change dir back from where we started */
714 if (!(plan->flags & F_EXECDIR) && fchdir(dotfd)) {
715 warn("chdir");
716 _exit(1);
717 }
718 execvp(plan->e_argv[0], plan->e_argv);
719 warn("%s", plan->e_argv[0]);
720 _exit(1);
721 }
722 if (plan->flags & F_EXECPLUS) {
723 while (--plan->e_ppos >= plan->e_pbnum)
724 free(plan->e_argv[plan->e_ppos]);
725 plan->e_ppos = plan->e_pbnum;
726 plan->e_psize = plan->e_pbsize;
727 }
728 pid = waitpid(pid, &status, 0);
729 if (plan->flags & F_EXECPLUS && WIFEXITED(status) && WEXITSTATUS(status))
730 _exit(WEXITSTATUS(status));
731 return (pid != -1 && WIFEXITED(status) && !WEXITSTATUS(status));
732 }
733
734 /*
735 * c_exec, c_execdir, c_ok --
736 * build three parallel arrays, one with pointers to the strings passed
737 * on the command line, one with (possibly duplicated) pointers to the
738 * argv array, and one with integer values that are lengths of the
739 * strings, but also flags meaning that the string has to be massaged.
740 */
741 PLAN *
742 c_exec(OPTION *option, char ***argvp)
743 {
744 PLAN *new; /* node returned */
745 long argmax;
746 int cnt, i;
747 char **argv, **ap, **ep, *p;
748
749 /* XXX - was in c_execdir, but seems unnecessary!?
750 ftsoptions &= ~FTS_NOSTAT;
751 */
752 isoutput = 1;
753
754 /* XXX - this is a change from the previous coding */
755 new = palloc(option);
756
757 for (ap = argv = *argvp;; ++ap) {
758 if (!*ap)
759 errx(1,
760 "%s: no terminating \";\" or \"+\"", option->name);
761 if (**ap == ';')
762 break;
763 if (**ap == '+' && ap != argv && strcmp(*(ap - 1), "{}") == 0) {
764 new->flags |= F_EXECPLUS;
765 break;
766 }
767 }
768
769 if (ap == argv)
770 errx(1, "%s: no command specified", option->name);
771
772 cnt = ap - *argvp + 1;
773 if (new->flags & F_EXECPLUS) {
774 new->e_ppos = new->e_pbnum = cnt - 2;
775 if ((argmax = sysconf(_SC_ARG_MAX)) == -1) {
776 warn("sysconf(_SC_ARG_MAX)");
777 argmax = _POSIX_ARG_MAX;
778 }
779 argmax -= 1024;
780 for (ep = environ; *ep != NULL; ep++)
781 argmax -= strlen(*ep) + 1 + sizeof(*ep);
782 argmax -= 1 + sizeof(*ep);
783 new->e_pnummax = argmax / 16;
784 argmax -= sizeof(char *) * new->e_pnummax;
785 if (argmax <= 0)
786 errx(1, "no space for arguments");
787 new->e_psizemax = argmax;
788 new->e_pbsize = 0;
789 cnt += new->e_pnummax + 1;
790 new->e_next = lastexecplus;
791 lastexecplus = new;
792 }
793 if ((new->e_argv = malloc(cnt * sizeof(char *))) == NULL)
794 err(1, NULL);
795 if ((new->e_orig = malloc(cnt * sizeof(char *))) == NULL)
796 err(1, NULL);
797 if ((new->e_len = malloc(cnt * sizeof(int))) == NULL)
798 err(1, NULL);
799
800 for (argv = *argvp, cnt = 0; argv < ap; ++argv, ++cnt) {
801 new->e_orig[cnt] = *argv;
802 if (new->flags & F_EXECPLUS)
803 new->e_pbsize += strlen(*argv) + 1;
804 for (p = *argv; *p; ++p)
805 if (!(new->flags & F_EXECPLUS) && p[0] == '{' &&
806 p[1] == '}') {
807 if ((new->e_argv[cnt] =
808 malloc(MAXPATHLEN)) == NULL)
809 err(1, NULL);
810 new->e_len[cnt] = MAXPATHLEN;
811 break;
812 }
813 if (!*p) {
814 new->e_argv[cnt] = *argv;
815 new->e_len[cnt] = 0;
816 }
817 }
818 if (new->flags & F_EXECPLUS) {
819 new->e_psize = new->e_pbsize;
820 cnt--;
821 for (i = 0; i < new->e_pnummax; i++) {
822 new->e_argv[cnt] = NULL;
823 new->e_len[cnt] = 0;
824 cnt++;
825 }
826 argv = ap;
827 goto done;
828 }
829 new->e_argv[cnt] = new->e_orig[cnt] = NULL;
830
831 done: *argvp = argv + 1;
832 return new;
833 }
834
835 /* Finish any pending -exec ... {} + functions. */
836 void
837 finish_execplus()
838 {
839 PLAN *p;
840
841 p = lastexecplus;
842 while (p != NULL) {
843 (p->execute)(p, NULL);
844 p = p->e_next;
845 }
846 }
847
848 int
849 f_flags(PLAN *plan, FTSENT *entry)
850 {
851 u_long flags;
852
853 flags = entry->fts_statp->st_flags;
854 if (plan->flags & F_ATLEAST)
855 return (flags | plan->fl_flags) == flags &&
856 !(flags & plan->fl_notflags);
857 else if (plan->flags & F_ANY)
858 return (flags & plan->fl_flags) ||
859 (flags | plan->fl_notflags) != flags;
860 else
861 return flags == plan->fl_flags &&
862 !(plan->fl_flags & plan->fl_notflags);
863 }
864
865 PLAN *
866 c_flags(OPTION *option, char ***argvp)
867 {
868 char *flags_str;
869 PLAN *new;
870 u_long flags, notflags;
871
872 flags_str = nextarg(option, argvp);
873 ftsoptions &= ~FTS_NOSTAT;
874
875 new = palloc(option);
876
877 if (*flags_str == '-') {
878 new->flags |= F_ATLEAST;
879 flags_str++;
880 } else if (*flags_str == '+') {
881 new->flags |= F_ANY;
882 flags_str++;
883 }
884 if (strtofflags(&flags_str, &flags, &notflags) == 1)
885 errx(1, "%s: %s: illegal flags string", option->name, flags_str);
886
887 new->fl_flags = flags;
888 new->fl_notflags = notflags;
889 return new;
890 }
891
892 /*
893 * -follow functions --
894 *
895 * Always true, causes symbolic links to be followed on a global
896 * basis.
897 */
898 PLAN *
899 c_follow(OPTION *option, char ***argvp __unused)
900 {
901 ftsoptions &= ~FTS_PHYSICAL;
902 ftsoptions |= FTS_LOGICAL;
903
904 return palloc(option);
905 }
906
907 /*
908 * -fstype functions --
909 *
910 * True if the file is of a certain type.
911 */
912 int
913 f_fstype(PLAN *plan, FTSENT *entry)
914 {
915 static dev_t curdev; /* need a guaranteed illegal dev value */
916 static int first = 1;
917 struct statfs sb;
918 static int val_type, val_flags;
919 char *p, save[2] = {0,0};
920
921 if ((plan->flags & F_MTMASK) == F_MTUNKNOWN)
922 return 0;
923
924 /* Only check when we cross mount point. */
925 if (first || curdev != entry->fts_statp->st_dev) {
926 curdev = entry->fts_statp->st_dev;
927
928 /*
929 * Statfs follows symlinks; find wants the link's filesystem,
930 * not where it points.
931 */
932 if (entry->fts_info == FTS_SL ||
933 entry->fts_info == FTS_SLNONE) {
934 if ((p = strrchr(entry->fts_accpath, '/')) != NULL)
935 ++p;
936 else
937 p = entry->fts_accpath;
938 save[0] = p[0];
939 p[0] = '.';
940 save[1] = p[1];
941 p[1] = '\0';
942 } else
943 p = NULL;
944
945 if (statfs(entry->fts_accpath, &sb))
946 err(1, "%s", entry->fts_accpath);
947
948 if (p) {
949 p[0] = save[0];
950 p[1] = save[1];
951 }
952
953 first = 0;
954
955 /*
956 * Further tests may need both of these values, so
957 * always copy both of them.
958 */
959 val_flags = sb.f_flags;
960 val_type = sb.f_type;
961 }
962 switch (plan->flags & F_MTMASK) {
963 case F_MTFLAG:
964 return val_flags & plan->mt_data;
965 case F_MTTYPE:
966 return val_type == plan->mt_data;
967 default:
968 abort();
969 }
970 }
971
972 PLAN *
973 c_fstype(OPTION *option, char ***argvp)
974 {
975 char *fsname;
976 PLAN *new;
977 struct vfsconf vfc;
978
979 fsname = nextarg(option, argvp);
980 ftsoptions &= ~FTS_NOSTAT;
981
982 new = palloc(option);
983
984 /*
985 * Check first for a filesystem name.
986 */
987 if (getvfsbyname(fsname, &vfc) == 0) {
988 new->flags |= F_MTTYPE;
989 new->mt_data = vfc.vfc_typenum;
990 return new;
991 }
992
993 switch (*fsname) {
994 case 'l':
995 if (!strcmp(fsname, "local")) {
996 new->flags |= F_MTFLAG;
997 new->mt_data = MNT_LOCAL;
998 return new;
999 }
1000 break;
1001 case 'r':
1002 if (!strcmp(fsname, "rdonly")) {
1003 new->flags |= F_MTFLAG;
1004 new->mt_data = MNT_RDONLY;
1005 return new;
1006 }
1007 break;
1008 }
1009
1010 /*
1011 * We need to make filesystem checks for filesystems
1012 * that exists but aren't in the kernel work.
1013 */
1014 fprintf(stderr, "Warning: Unknown filesystem type %s\n", fsname);
1015 new->flags |= F_MTUNKNOWN;
1016 return new;
1017 }
1018
1019 /*
1020 * -group gname functions --
1021 *
1022 * True if the file belongs to the group gname. If gname is numeric and
1023 * an equivalent of the getgrnam() function does not return a valid group
1024 * name, gname is taken as a group ID.
1025 */
1026 int
1027 f_group(PLAN *plan, FTSENT *entry)
1028 {
1029 COMPARE(entry->fts_statp->st_gid, plan->g_data);
1030 }
1031
1032 PLAN *
1033 c_group(OPTION *option, char ***argvp)
1034 {
1035 char *gname;
1036 PLAN *new;
1037 struct group *g;
1038 gid_t gid;
1039
1040 gname = nextarg(option, argvp);
1041 ftsoptions &= ~FTS_NOSTAT;
1042
1043 new = palloc(option);
1044 g = getgrnam(gname);
1045 if (g == NULL) {
1046 char* cp = gname;
1047 if (gname[0] == '-' || gname[0] == '+')
1048 gname++;
1049 gid = atoi(gname);
1050 if (gid == 0 && gname[0] != '0')
1051 errx(1, "%s: %s: no such group", option->name, gname);
1052 gid = find_parsenum(new, option->name, cp, NULL);
1053 } else
1054 gid = g->gr_gid;
1055
1056 new->g_data = gid;
1057 return new;
1058 }
1059
1060 /*
1061 * -inum n functions --
1062 *
1063 * True if the file has inode # n.
1064 */
1065 int
1066 f_inum(PLAN *plan, FTSENT *entry)
1067 {
1068 COMPARE(entry->fts_statp->st_ino, plan->i_data);
1069 }
1070
1071 PLAN *
1072 c_inum(OPTION *option, char ***argvp)
1073 {
1074 char *inum_str;
1075 PLAN *new;
1076
1077 inum_str = nextarg(option, argvp);
1078 ftsoptions &= ~FTS_NOSTAT;
1079
1080 new = palloc(option);
1081 new->i_data = find_parsenum(new, option->name, inum_str, NULL);
1082 return new;
1083 }
1084
1085 /*
1086 * -samefile FN
1087 *
1088 * True if the file has the same inode (eg hard link) FN
1089 */
1090
1091 /* f_samefile is just f_inum */
1092 PLAN *
1093 c_samefile(OPTION *option, char ***argvp)
1094 {
1095 char *fn;
1096 PLAN *new;
1097 struct stat sb;
1098
1099 fn = nextarg(option, argvp);
1100 ftsoptions &= ~FTS_NOSTAT;
1101
1102 new = palloc(option);
1103 if (stat(fn, &sb))
1104 err(1, "%s", fn);
1105 new->i_data = sb.st_ino;
1106 return new;
1107 }
1108
1109 /*
1110 * -links n functions --
1111 *
1112 * True if the file has n links.
1113 */
1114 int
1115 f_links(PLAN *plan, FTSENT *entry)
1116 {
1117 COMPARE(entry->fts_statp->st_nlink, plan->l_data);
1118 }
1119
1120 PLAN *
1121 c_links(OPTION *option, char ***argvp)
1122 {
1123 char *nlinks;
1124 PLAN *new;
1125
1126 nlinks = nextarg(option, argvp);
1127 ftsoptions &= ~FTS_NOSTAT;
1128
1129 new = palloc(option);
1130 new->l_data = (nlink_t)find_parsenum(new, option->name, nlinks, NULL);
1131 return new;
1132 }
1133
1134 /*
1135 * -ls functions --
1136 *
1137 * Always true - prints the current entry to stdout in "ls" format.
1138 */
1139 int
1140 f_ls(PLAN *plan __unused, FTSENT *entry)
1141 {
1142 printlong(entry->fts_path, entry->fts_accpath, entry->fts_statp);
1143 return 1;
1144 }
1145
1146 PLAN *
1147 c_ls(OPTION *option, char ***argvp __unused)
1148 {
1149 ftsoptions &= ~FTS_NOSTAT;
1150 isoutput = 1;
1151
1152 return palloc(option);
1153 }
1154
1155 /*
1156 * -name functions --
1157 *
1158 * True if the basename of the filename being examined
1159 * matches pattern using Pattern Matching Notation S3.14
1160 */
1161 int
1162 f_name(PLAN *plan, FTSENT *entry)
1163 {
1164 char fn[PATH_MAX];
1165 const char *name;
1166
1167 if (plan->flags & F_LINK) {
1168 name = fn;
1169 if (readlink(entry->fts_accpath, fn, sizeof(fn)) == -1)
1170 return 0;
1171 } else if (entry->fts_namelen == 0) {
1172 name = basename(entry->fts_path);
1173 } else
1174 name = entry->fts_name;
1175 return !fnmatch(plan->c_data, name,
1176 plan->flags & F_IGNCASE ? FNM_CASEFOLD : 0);
1177 }
1178
1179 PLAN *
1180 c_name(OPTION *option, char ***argvp)
1181 {
1182 char *pattern;
1183 PLAN *new;
1184
1185 pattern = nextarg(option, argvp);
1186 new = palloc(option);
1187 new->c_data = pattern;
1188 return new;
1189 }
1190
1191 /*
1192 * -newer file functions --
1193 *
1194 * True if the current file has been modified more recently
1195 * then the modification time of the file named by the pathname
1196 * file.
1197 */
1198 int
1199 f_newer(PLAN *plan, FTSENT *entry)
1200 {
1201 if (plan->flags & F_TIME_C)
1202 return entry->fts_statp->st_ctime > plan->t_data;
1203 else if (plan->flags & F_TIME_A)
1204 return entry->fts_statp->st_atime > plan->t_data;
1205 else if (plan->flags & F_TIME_B)
1206 return entry->fts_statp->st_birthtime > plan->t_data;
1207 else
1208 return entry->fts_statp->st_mtime > plan->t_data;
1209 }
1210
1211 PLAN *
1212 c_newer(OPTION *option, char ***argvp)
1213 {
1214 char *fn_or_tspec;
1215 PLAN *new;
1216 struct stat sb;
1217
1218 fn_or_tspec = nextarg(option, argvp);
1219 ftsoptions &= ~FTS_NOSTAT;
1220
1221 new = palloc(option);
1222 /* compare against what */
1223 if (option->flags & F_TIME2_T) {
1224 new->t_data = get_date(fn_or_tspec, (struct timeb *) 0);
1225 if (new->t_data == (time_t) -1)
1226 errx(1, "Can't parse date/time: %s", fn_or_tspec);
1227 } else {
1228 if (stat(fn_or_tspec, &sb))
1229 err(1, "%s", fn_or_tspec);
1230 if (option->flags & F_TIME2_C)
1231 new->t_data = sb.st_ctime;
1232 else if (option->flags & F_TIME2_A)
1233 new->t_data = sb.st_atime;
1234 else
1235 new->t_data = sb.st_mtime;
1236 }
1237 return new;
1238 }
1239
1240 /*
1241 * -nogroup functions --
1242 *
1243 * True if file belongs to a user ID for which the equivalent
1244 * of the getgrnam() 9.2.1 [POSIX.1] function returns NULL.
1245 */
1246 int
1247 f_nogroup(PLAN *plan __unused, FTSENT *entry)
1248 {
1249 return group_from_gid(entry->fts_statp->st_gid, 1) == NULL;
1250 }
1251
1252 PLAN *
1253 c_nogroup(OPTION *option, char ***argvp __unused)
1254 {
1255 ftsoptions &= ~FTS_NOSTAT;
1256
1257 return palloc(option);
1258 }
1259
1260 /*
1261 * -nouser functions --
1262 *
1263 * True if file belongs to a user ID for which the equivalent
1264 * of the getpwuid() 9.2.2 [POSIX.1] function returns NULL.
1265 */
1266 int
1267 f_nouser(PLAN *plan __unused, FTSENT *entry)
1268 {
1269 return user_from_uid(entry->fts_statp->st_uid, 1) == NULL;
1270 }
1271
1272 PLAN *
1273 c_nouser(OPTION *option, char ***argvp __unused)
1274 {
1275 ftsoptions &= ~FTS_NOSTAT;
1276
1277 return palloc(option);
1278 }
1279
1280 /*
1281 * -path functions --
1282 *
1283 * True if the path of the filename being examined
1284 * matches pattern using Pattern Matching Notation S3.14
1285 */
1286 int
1287 f_path(PLAN *plan, FTSENT *entry)
1288 {
1289 return !fnmatch(plan->c_data, entry->fts_path,
1290 plan->flags & F_IGNCASE ? FNM_CASEFOLD : 0);
1291 }
1292
1293 /* c_path is the same as c_name */
1294
1295 /*
1296 * -perm functions --
1297 *
1298 * The mode argument is used to represent file mode bits. If it starts
1299 * with a leading digit, it's treated as an octal mode, otherwise as a
1300 * symbolic mode.
1301 */
1302 int
1303 f_perm(PLAN *plan, FTSENT *entry)
1304 {
1305 mode_t mode;
1306
1307 mode = entry->fts_statp->st_mode &
1308 (S_ISUID|S_ISGID|S_ISTXT|S_IRWXU|S_IRWXG|S_IRWXO);
1309 if (plan->flags & F_ATLEAST)
1310 return (plan->m_data | mode) == mode;
1311 else if (plan->flags & F_ANY)
1312 return (mode & plan->m_data);
1313 else
1314 return mode == plan->m_data;
1315 /* NOTREACHED */
1316 }
1317
1318 PLAN *
1319 c_perm(OPTION *option, char ***argvp)
1320 {
1321 char *perm;
1322 PLAN *new;
1323 mode_t *set;
1324
1325 perm = nextarg(option, argvp);
1326 ftsoptions &= ~FTS_NOSTAT;
1327
1328 new = palloc(option);
1329
1330 if (*perm == '-') {
1331 new->flags |= F_ATLEAST;
1332 ++perm;
1333 } else if (*perm == '+') {
1334 if ((set = setmode(perm + 1)) != NULL) {
1335 new->flags |= F_ANY;
1336 ++perm;
1337 free(set);
1338 }
1339 }
1340
1341 if ((set = setmode(perm)) == NULL)
1342 errx(1, "%s: %s: illegal mode string", option->name, perm);
1343
1344 new->m_data = getmode(set, 0);
1345 free(set);
1346 return new;
1347 }
1348
1349 /*
1350 * -print functions --
1351 *
1352 * Always true, causes the current pathname to be written to
1353 * standard output.
1354 */
1355 int
1356 f_print(PLAN *plan __unused, FTSENT *entry)
1357 {
1358 (void)puts(entry->fts_path);
1359 return 1;
1360 }
1361
1362 PLAN *
1363 c_print(OPTION *option, char ***argvp __unused)
1364 {
1365 isoutput = 1;
1366
1367 return palloc(option);
1368 }
1369
1370 /*
1371 * -print0 functions --
1372 *
1373 * Always true, causes the current pathname to be written to
1374 * standard output followed by a NUL character
1375 */
1376 int
1377 f_print0(PLAN *plan __unused, FTSENT *entry)
1378 {
1379 fputs(entry->fts_path, stdout);
1380 fputc('\0', stdout);
1381 return 1;
1382 }
1383
1384 /* c_print0 is the same as c_print */
1385
1386 /*
1387 * -prune functions --
1388 *
1389 * Prune a portion of the hierarchy.
1390 */
1391 int
1392 f_prune(PLAN *plan __unused, FTSENT *entry)
1393 {
1394 if (fts_set(tree, entry, FTS_SKIP))
1395 err(1, "%s", entry->fts_path);
1396 return 1;
1397 }
1398
1399 /* c_prune == c_simple */
1400
1401 /*
1402 * -regex functions --
1403 *
1404 * True if the whole path of the file matches pattern using
1405 * regular expression.
1406 */
1407 int
1408 f_regex(PLAN *plan, FTSENT *entry)
1409 {
1410 char *str;
1411 int len;
1412 regex_t *pre;
1413 regmatch_t pmatch;
1414 int errcode;
1415 char errbuf[LINE_MAX];
1416 int matched;
1417
1418 pre = plan->re_data;
1419 str = entry->fts_path;
1420 len = strlen(str);
1421 matched = 0;
1422
1423 pmatch.rm_so = 0;
1424 pmatch.rm_eo = len;
1425
1426 errcode = regexec(pre, str, 1, &pmatch, REG_STARTEND);
1427
1428 if (errcode != 0 && errcode != REG_NOMATCH) {
1429 regerror(errcode, pre, errbuf, sizeof errbuf);
1430 errx(1, "%s: %s",
1431 plan->flags & F_IGNCASE ? "-iregex" : "-regex", errbuf);
1432 }
1433
1434 if (errcode == 0 && pmatch.rm_so == 0 && pmatch.rm_eo == len)
1435 matched = 1;
1436
1437 return matched;
1438 }
1439
1440 PLAN *
1441 c_regex(OPTION *option, char ***argvp)
1442 {
1443 PLAN *new;
1444 char *pattern;
1445 regex_t *pre;
1446 int errcode;
1447 char errbuf[LINE_MAX];
1448
1449 if ((pre = malloc(sizeof(regex_t))) == NULL)
1450 err(1, NULL);
1451
1452 pattern = nextarg(option, argvp);
1453
1454 if ((errcode = regcomp(pre, pattern,
1455 regexp_flags | (option->flags & F_IGNCASE ? REG_ICASE : 0))) != 0) {
1456 regerror(errcode, pre, errbuf, sizeof errbuf);
1457 errx(1, "%s: %s: %s",
1458 option->flags & F_IGNCASE ? "-iregex" : "-regex",
1459 pattern, errbuf);
1460 }
1461
1462 new = palloc(option);
1463 new->re_data = pre;
1464
1465 return new;
1466 }
1467
1468 /* c_simple covers c_prune, c_openparen, c_closeparen, c_not, c_or, c_true, c_false */
1469
1470 PLAN *
1471 c_simple(OPTION *option, char ***argvp __unused)
1472 {
1473 return palloc(option);
1474 }
1475
1476 /*
1477 * -size n[c] functions --
1478 *
1479 * True if the file size in bytes, divided by an implementation defined
1480 * value and rounded up to the next integer, is n. If n is followed by
1481 * one of c k M G T P, the size is in bytes, kilobytes,
1482 * megabytes, gigabytes, terabytes or petabytes respectively.
1483 */
1484 #define FIND_SIZE 512
1485 static int divsize = 1;
1486
1487 int
1488 f_size(PLAN *plan, FTSENT *entry)
1489 {
1490 off_t size;
1491
1492 size = divsize ? (entry->fts_statp->st_size + FIND_SIZE - 1) /
1493 FIND_SIZE : entry->fts_statp->st_size;
1494 COMPARE(size, plan->o_data);
1495 }
1496
1497 PLAN *
1498 c_size(OPTION *option, char ***argvp)
1499 {
1500 char *size_str;
1501 PLAN *new;
1502 char endch;
1503 off_t scale;
1504
1505 size_str = nextarg(option, argvp);
1506 ftsoptions &= ~FTS_NOSTAT;
1507
1508 new = palloc(option);
1509 endch = 'c';
1510 new->o_data = find_parsenum(new, option->name, size_str, &endch);
1511 if (endch != '\0') {
1512 divsize = 0;
1513
1514 switch (endch) {
1515 case 'c': /* characters */
1516 scale = 0x1LL;
1517 break;
1518 case 'k': /* kilobytes 1<<10 */
1519 scale = 0x400LL;
1520 break;
1521 case 'M': /* megabytes 1<<20 */
1522 scale = 0x100000LL;
1523 break;
1524 case 'G': /* gigabytes 1<<30 */
1525 scale = 0x40000000LL;
1526 break;
1527 case 'T': /* terabytes 1<<40 */
1528 scale = 0x1000000000LL;
1529 break;
1530 case 'P': /* petabytes 1<<50 */
1531 scale = 0x4000000000000LL;
1532 break;
1533 default:
1534 errx(1, "%s: %s: illegal trailing character",
1535 option->name, size_str);
1536 break;
1537 }
1538 if (new->o_data > QUAD_MAX / scale)
1539 errx(1, "%s: %s: value too large",
1540 option->name, size_str);
1541 new->o_data *= scale;
1542 }
1543 return new;
1544 }
1545
1546 /*
1547 * -type c functions --
1548 *
1549 * True if the type of the file is c, where c is b, c, d, p, f or w
1550 * for block special file, character special file, directory, FIFO,
1551 * regular file or whiteout respectively.
1552 */
1553 int
1554 f_type(PLAN *plan, FTSENT *entry)
1555 {
1556 return (entry->fts_statp->st_mode & S_IFMT) == plan->m_data;
1557 }
1558
1559 PLAN *
1560 c_type(OPTION *option, char ***argvp)
1561 {
1562 char *typestring;
1563 PLAN *new;
1564 mode_t mask;
1565
1566 typestring = nextarg(option, argvp);
1567 ftsoptions &= ~FTS_NOSTAT;
1568
1569 switch (typestring[0]) {
1570 case 'b':
1571 mask = S_IFBLK;
1572 break;
1573 case 'c':
1574 mask = S_IFCHR;
1575 break;
1576 case 'd':
1577 mask = S_IFDIR;
1578 break;
1579 case 'f':
1580 mask = S_IFREG;
1581 break;
1582 case 'l':
1583 mask = S_IFLNK;
1584 break;
1585 case 'p':
1586 mask = S_IFIFO;
1587 break;
1588 case 's':
1589 mask = S_IFSOCK;
1590 break;
1591 #ifdef FTS_WHITEOUT
1592 case 'w':
1593 mask = S_IFWHT;
1594 ftsoptions |= FTS_WHITEOUT;
1595 break;
1596 #endif /* FTS_WHITEOUT */
1597 default:
1598 errx(1, "%s: %s: unknown type", option->name, typestring);
1599 }
1600
1601 new = palloc(option);
1602 new->m_data = mask;
1603 return new;
1604 }
1605
1606 /*
1607 * -user uname functions --
1608 *
1609 * True if the file belongs to the user uname. If uname is numeric and
1610 * an equivalent of the getpwnam() S9.2.2 [POSIX.1] function does not
1611 * return a valid user name, uname is taken as a user ID.
1612 */
1613 int
1614 f_user(PLAN *plan, FTSENT *entry)
1615 {
1616 COMPARE(entry->fts_statp->st_uid, plan->u_data);
1617 }
1618
1619 PLAN *
1620 c_user(OPTION *option, char ***argvp)
1621 {
1622 char *username;
1623 PLAN *new;
1624 struct passwd *p;
1625 uid_t uid;
1626
1627 username = nextarg(option, argvp);
1628 ftsoptions &= ~FTS_NOSTAT;
1629
1630 new = palloc(option);
1631 p = getpwnam(username);
1632 if (p == NULL) {
1633 char* cp = username;
1634 if( username[0] == '-' || username[0] == '+' )
1635 username++;
1636 uid = atoi(username);
1637 if (uid == 0 && username[0] != '0')
1638 errx(1, "%s: %s: no such user", option->name, username);
1639 uid = find_parsenum(new, option->name, cp, NULL);
1640 } else
1641 uid = p->pw_uid;
1642
1643 new->u_data = uid;
1644 return new;
1645 }
1646
1647 /*
1648 * -xdev functions --
1649 *
1650 * Always true, causes find not to descend past directories that have a
1651 * different device ID (st_dev, see stat() S5.6.2 [POSIX.1])
1652 */
1653 PLAN *
1654 c_xdev(OPTION *option, char ***argvp __unused)
1655 {
1656 ftsoptions |= FTS_XDEV;
1657
1658 return palloc(option);
1659 }
1660
1661 /*
1662 * ( expression ) functions --
1663 *
1664 * True if expression is true.
1665 */
1666 int
1667 f_expr(PLAN *plan, FTSENT *entry)
1668 {
1669 PLAN *p;
1670 int state = 0;
1671
1672 for (p = plan->p_data[0];
1673 p && (state = (p->execute)(p, entry)); p = p->next);
1674 return state;
1675 }
1676
1677 /*
1678 * f_openparen and f_closeparen nodes are temporary place markers. They are
1679 * eliminated during phase 2 of find_formplan() --- the '(' node is converted
1680 * to a f_expr node containing the expression and the ')' node is discarded.
1681 * The functions themselves are only used as constants.
1682 */
1683
1684 int
1685 f_openparen(PLAN *plan __unused, FTSENT *entry __unused)
1686 {
1687 abort();
1688 }
1689
1690 int
1691 f_closeparen(PLAN *plan __unused, FTSENT *entry __unused)
1692 {
1693 abort();
1694 }
1695
1696 /* c_openparen == c_simple */
1697 /* c_closeparen == c_simple */
1698
1699 /*
1700 * AND operator. Since AND is implicit, no node is allocated.
1701 */
1702 PLAN *
1703 c_and(OPTION *option __unused, char ***argvp __unused)
1704 {
1705 return NULL;
1706 }
1707
1708 /*
1709 * ! expression functions --
1710 *
1711 * Negation of a primary; the unary NOT operator.
1712 */
1713 int
1714 f_not(PLAN *plan, FTSENT *entry)
1715 {
1716 PLAN *p;
1717 int state = 0;
1718
1719 for (p = plan->p_data[0];
1720 p && (state = (p->execute)(p, entry)); p = p->next);
1721 return !state;
1722 }
1723
1724 /* c_not == c_simple */
1725
1726 /*
1727 * expression -o expression functions --
1728 *
1729 * Alternation of primaries; the OR operator. The second expression is
1730 * not evaluated if the first expression is true.
1731 */
1732 int
1733 f_or(PLAN *plan, FTSENT *entry)
1734 {
1735 PLAN *p;
1736 int state = 0;
1737
1738 for (p = plan->p_data[0];
1739 p && (state = (p->execute)(p, entry)); p = p->next);
1740
1741 if (state)
1742 return 1;
1743
1744 for (p = plan->p_data[1];
1745 p && (state = (p->execute)(p, entry)); p = p->next);
1746 return state;
1747 }
1748
1749 /* c_or == c_simple */
1750
1751 /*
1752 * -false
1753 *
1754 * Always false.
1755 */
1756 int
1757 f_false(PLAN *plan __unused, FTSENT *entry __unused)
1758 {
1759 return 0;
1760 }
1761
1762 /* c_false == c_simple */
1763
1764 /*
1765 * -quit
1766 *
1767 * Exits the program
1768 */
1769 int
1770 f_quit(PLAN *plan __unused, FTSENT *entry __unused)
1771 {
1772 exit(0);
1773 }
1774
1775 /* c_quit == c_simple */