getopt32: remove opt_complementary
[oweals/busybox.git] / debianutils / which.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
4  * Copyright (C) 2006 Gabriel Somlo <somlo at cmu.edu>
5  *
6  * Licensed under GPLv2 or later, see file LICENSE in this source tree.
7  */
8 //config:config WHICH
9 //config:       bool "which (3.7 kb)"
10 //config:       default y
11 //config:       help
12 //config:       which is used to find programs in your PATH and
13 //config:       print out their pathnames.
14
15 //applet:IF_WHICH(APPLET_NOFORK(which, which, BB_DIR_USR_BIN, BB_SUID_DROP, which))
16
17 //kbuild:lib-$(CONFIG_WHICH) += which.o
18
19 //usage:#define which_trivial_usage
20 //usage:       "[COMMAND]..."
21 //usage:#define which_full_usage "\n\n"
22 //usage:       "Locate a COMMAND"
23 //usage:
24 //usage:#define which_example_usage
25 //usage:       "$ which login\n"
26 //usage:       "/bin/login\n"
27
28 #include "libbb.h"
29
30 int which_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
31 int which_main(int argc UNUSED_PARAM, char **argv)
32 {
33         const char *env_path;
34         int status = 0;
35
36         env_path = getenv("PATH");
37         if (!env_path)
38                 env_path = bb_default_root_path;
39
40         getopt32(argv, "^" "a" "\0" "-1"/*at least one arg*/);
41         argv += optind;
42
43         do {
44                 int missing = 1;
45
46                 /* If file contains a slash don't use PATH */
47                 if (strchr(*argv, '/')) {
48                         if (file_is_executable(*argv)) {
49                                 missing = 0;
50                                 puts(*argv);
51                         }
52                 } else {
53                         char *path;
54                         char *tmp;
55                         char *p;
56
57                         path = tmp = xstrdup(env_path);
58 //NOFORK FIXME: nested xmallocs (one is inside find_executable())
59 //can leak memory on failure
60                         while ((p = find_executable(*argv, &tmp)) != NULL) {
61                                 missing = 0;
62                                 puts(p);
63                                 free(p);
64                                 if (!option_mask32) /* -a not set */
65                                         break;
66                         }
67                         free(path);
68                 }
69                 status |= missing;
70         } while (*++argv);
71
72         return status;
73 }