/usr.bin/find/main.c

https://bitbucket.org/freebsd/freebsd-head/ · C · 167 lines · 112 code · 19 blank · 36 comment · 17 complexity · da28a5bbe30bd8e0a48246181a05f369 MD5 · raw file

  1. /*-
  2. * Copyright (c) 1990, 1993, 1994
  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. * 4. Neither the name of the University nor the names of its contributors
  17. * may be used to endorse or promote products derived from this software
  18. * without specific prior written permission.
  19. *
  20. * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
  21. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  22. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  23. * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
  24. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  25. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  26. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  27. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  28. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  29. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  30. * SUCH DAMAGE.
  31. */
  32. #ifndef lint
  33. static const char copyright[] =
  34. "@(#) Copyright (c) 1990, 1993, 1994\n\
  35. The Regents of the University of California. All rights reserved.\n";
  36. #endif /* not lint */
  37. #ifndef lint
  38. #if 0
  39. static char sccsid[] = "@(#)main.c 8.4 (Berkeley) 5/4/95";
  40. #endif
  41. #endif /* not lint */
  42. #include <sys/cdefs.h>
  43. __FBSDID("$FreeBSD$");
  44. #include <sys/types.h>
  45. #include <sys/stat.h>
  46. #include <err.h>
  47. #include <errno.h>
  48. #include <fcntl.h>
  49. #include <fts.h>
  50. #include <locale.h>
  51. #include <regex.h>
  52. #include <stdio.h>
  53. #include <stdlib.h>
  54. #include <time.h>
  55. #include <unistd.h>
  56. #include "find.h"
  57. time_t now; /* time find was run */
  58. int dotfd; /* starting directory */
  59. int ftsoptions; /* options for the ftsopen(3) call */
  60. int ignore_readdir_race; /* ignore readdir race */
  61. int isdeprecated; /* using deprecated syntax */
  62. int isdepth; /* do directories on post-order visit */
  63. int isoutput; /* user specified output operator */
  64. int issort; /* do hierarchies in lexicographical order */
  65. int isxargs; /* don't permit xargs delimiting chars */
  66. int mindepth = -1, maxdepth = -1; /* minimum and maximum depth */
  67. int regexp_flags = REG_BASIC; /* use the "basic" regexp by default*/
  68. static void usage(void);
  69. int
  70. main(int argc, char *argv[])
  71. {
  72. char **p, **start;
  73. int Hflag, Lflag, ch;
  74. (void)setlocale(LC_ALL, "");
  75. (void)time(&now); /* initialize the time-of-day */
  76. p = start = argv;
  77. Hflag = Lflag = 0;
  78. ftsoptions = FTS_NOSTAT | FTS_PHYSICAL;
  79. while ((ch = getopt(argc, argv, "EHLPXdf:sx")) != -1)
  80. switch (ch) {
  81. case 'E':
  82. regexp_flags |= REG_EXTENDED;
  83. break;
  84. case 'H':
  85. Hflag = 1;
  86. Lflag = 0;
  87. break;
  88. case 'L':
  89. Lflag = 1;
  90. Hflag = 0;
  91. break;
  92. case 'P':
  93. Hflag = Lflag = 0;
  94. break;
  95. case 'X':
  96. isxargs = 1;
  97. break;
  98. case 'd':
  99. isdepth = 1;
  100. break;
  101. case 'f':
  102. *p++ = optarg;
  103. break;
  104. case 's':
  105. issort = 1;
  106. break;
  107. case 'x':
  108. ftsoptions |= FTS_XDEV;
  109. break;
  110. case '?':
  111. default:
  112. usage();
  113. }
  114. argc -= optind;
  115. argv += optind;
  116. if (Hflag)
  117. ftsoptions |= FTS_COMFOLLOW;
  118. if (Lflag) {
  119. ftsoptions &= ~FTS_PHYSICAL;
  120. ftsoptions |= FTS_LOGICAL;
  121. }
  122. /*
  123. * Find first option to delimit the file list. The first argument
  124. * that starts with a -, or is a ! or a ( must be interpreted as a
  125. * part of the find expression, according to POSIX .2.
  126. */
  127. for (; *argv != NULL; *p++ = *argv++) {
  128. if (argv[0][0] == '-')
  129. break;
  130. if ((argv[0][0] == '!' || argv[0][0] == '(') &&
  131. argv[0][1] == '\0')
  132. break;
  133. }
  134. if (p == start)
  135. usage();
  136. *p = NULL;
  137. if ((dotfd = open(".", O_RDONLY | O_CLOEXEC, 0)) < 0)
  138. err(1, ".");
  139. exit(find_execute(find_formplan(argv), start));
  140. }
  141. static void
  142. usage(void)
  143. {
  144. (void)fprintf(stderr, "%s\n%s\n",
  145. "usage: find [-H | -L | -P] [-EXdsx] [-f path] path ... [expression]",
  146. " find [-H | -L | -P] [-EXdsx] -f path [path ...] [expression]");
  147. exit(1);
  148. }