which.c 1.8 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
  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. //applet:IF_WHICH(APPLET_NOFORK(which, which, BB_DIR_USR_BIN, BB_SUID_DROP, which))
  15. //kbuild:lib-$(CONFIG_WHICH) += which.o
  16. //usage:#define which_trivial_usage
  17. //usage: "[COMMAND]..."
  18. //usage:#define which_full_usage "\n\n"
  19. //usage: "Locate a COMMAND"
  20. //usage:
  21. //usage:#define which_example_usage
  22. //usage: "$ which login\n"
  23. //usage: "/bin/login\n"
  24. #include "libbb.h"
  25. int which_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
  26. int which_main(int argc UNUSED_PARAM, char **argv)
  27. {
  28. char *env_path;
  29. int status = 0;
  30. /* This sizeof(): bb_default_root_path is shorter than BB_PATH_ROOT_PATH */
  31. char buf[sizeof(BB_PATH_ROOT_PATH)];
  32. env_path = getenv("PATH");
  33. if (!env_path)
  34. /* env_path must be writable, and must not alloc, so... */
  35. env_path = strcpy(buf, bb_default_root_path);
  36. getopt32(argv, "^" "a" "\0" "-1"/*at least one arg*/);
  37. argv += optind;
  38. do {
  39. int missing = 1;
  40. /* If file contains a slash don't use PATH */
  41. if (strchr(*argv, '/')) {
  42. if (file_is_executable(*argv)) {
  43. missing = 0;
  44. puts(*argv);
  45. }
  46. } else {
  47. char *path;
  48. char *p;
  49. path = env_path;
  50. /* NOFORK NB: xmalloc inside find_executable(), must have no allocs above! */
  51. while ((p = find_executable(*argv, &path)) != NULL) {
  52. missing = 0;
  53. puts(p);
  54. free(p);
  55. if (!option_mask32) /* -a not set */
  56. break;
  57. }
  58. }
  59. status |= missing;
  60. } while (*++argv);
  61. return status;
  62. }