basename.c 2.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. /* vi: set sw=4 ts=4: */
  2. /*
  3. * Mini basename implementation for busybox
  4. *
  5. * Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
  6. *
  7. * Licensed under GPLv2 or later, see file LICENSE in this source tree.
  8. */
  9. /* Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org)
  10. *
  11. * Changes:
  12. * 1) Now checks for too many args. Need at least one and at most two.
  13. * 2) Don't check for options, as per SUSv3.
  14. * 3) Save some space by using strcmp(). Calling strncmp() here was silly.
  15. */
  16. //config:config BASENAME
  17. //config: bool "basename (438 bytes)"
  18. //config: default y
  19. //config: help
  20. //config: basename is used to strip the directory and suffix from filenames,
  21. //config: leaving just the filename itself. Enable this option if you wish
  22. //config: to enable the 'basename' utility.
  23. //applet:IF_BASENAME(APPLET_NOFORK(basename, basename, BB_DIR_USR_BIN, BB_SUID_DROP, basename))
  24. //kbuild:lib-$(CONFIG_BASENAME) += basename.o
  25. /* BB_AUDIT SUSv3 compliant */
  26. /* http://www.opengroup.org/onlinepubs/007904975/utilities/basename.html */
  27. //usage:#define basename_trivial_usage
  28. //usage: "FILE [SUFFIX] | -a FILE... | -s SUFFIX FILE..."
  29. //usage:#define basename_full_usage "\n\n"
  30. //usage: "Strip directory path and SUFFIX from FILE\n"
  31. //usage: "\n -a All arguments are FILEs"
  32. //usage: "\n -s SUFFIX Remove SUFFIX (implies -a)"
  33. //usage:
  34. //usage:#define basename_example_usage
  35. //usage: "$ basename /usr/local/bin/foo\n"
  36. //usage: "foo\n"
  37. //usage: "$ basename /usr/local/bin/\n"
  38. //usage: "bin\n"
  39. //usage: "$ basename /foo/bar.txt .txt\n"
  40. //usage: "bar"
  41. #include "libbb.h"
  42. /* This is a NOFORK applet. Be very careful! */
  43. int basename_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
  44. int basename_main(int argc UNUSED_PARAM, char **argv)
  45. {
  46. unsigned opts;
  47. const char *suffix = NULL;
  48. /* '+': stop at first non-option */
  49. opts = getopt32(argv, "^+" "as:"
  50. "\0" "-1" /* At least one argument */
  51. , &suffix
  52. );
  53. argv += optind;
  54. do {
  55. char *s;
  56. size_t m;
  57. /* It should strip slash: /abc/def/ -> def */
  58. s = bb_get_last_path_component_strip(*argv++);
  59. m = strlen(s);
  60. if (!opts) {
  61. if (*argv) {
  62. suffix = *argv;
  63. if (argv[1])
  64. bb_show_usage();
  65. }
  66. }
  67. if (suffix) {
  68. size_t n = strlen(suffix);
  69. if ((m > n) && (strcmp(s + m - n, suffix) == 0)) {
  70. m -= n;
  71. /*s[m] = '\0'; - redundant */
  72. }
  73. }
  74. /* puts(s) will do, but we can do without stdio this way: */
  75. s[m++] = '\n';
  76. /* NB: != is correct here: */
  77. if (full_write(STDOUT_FILENO, s, m) != (ssize_t)m)
  78. return EXIT_FAILURE;
  79. } while (opts && *argv);
  80. return EXIT_SUCCESS;
  81. }