1 /* basename -- strip directory and suffix from file names
2 Copyright (C) 1990-2023 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
20 #include <sys/types.h>
25 /* The official name of this program (e.g., no 'g' prefix). */
26 #define PROGRAM_NAME "basename"
28 #define AUTHORS proper_name ("David MacKenzie")
30 static struct option
const longopts
[] =
32 {"multiple", no_argument
, nullptr, 'a'},
33 {"suffix", required_argument
, nullptr, 's'},
34 {"zero", no_argument
, nullptr, 'z'},
35 {GETOPT_HELP_OPTION_DECL
},
36 {GETOPT_VERSION_OPTION_DECL
},
37 {nullptr, 0, nullptr, 0}
43 if (status
!= EXIT_SUCCESS
)
48 Usage: %s NAME [SUFFIX]\n\
49 or: %s OPTION... NAME...\n\
51 program_name
, program_name
);
53 Print NAME with any leading directory components removed.\n\
54 If specified, also remove a trailing SUFFIX.\n\
57 emit_mandatory_arg_note ();
60 -a, --multiple support multiple arguments and treat each as a NAME\n\
61 -s, --suffix=SUFFIX remove a trailing SUFFIX; implies -a\n\
62 -z, --zero end each output line with NUL, not newline\n\
64 fputs (HELP_OPTION_DESCRIPTION
, stdout
);
65 fputs (VERSION_OPTION_DESCRIPTION
, stdout
);
69 %s /usr/bin/sort -> \"sort\"\n\
70 %s include/stdio.h .h -> \"stdio\"\n\
71 %s -s .h include/stdio.h -> \"stdio\"\n\
72 %s -a any/str1 any/str2 -> \"str1\" followed by \"str2\"\n\
74 program_name
, program_name
, program_name
, program_name
);
75 emit_ancillary_info (PROGRAM_NAME
);
80 /* Remove SUFFIX from the end of NAME if it is there, unless NAME
81 consists entirely of SUFFIX. */
84 remove_suffix (char *name
, char const *suffix
)
89 np
= name
+ strlen (name
);
90 sp
= suffix
+ strlen (suffix
);
92 while (np
> name
&& sp
> suffix
)
99 /* Perform the basename operation on STRING. If SUFFIX is non-null, remove
100 the trailing SUFFIX. Finally, output the result string. */
103 perform_basename (char const *string
, char const *suffix
, bool use_nuls
)
105 char *name
= base_name (string
);
106 strip_trailing_slashes (name
);
108 /* Per POSIX, 'basename // /' must return '//' on platforms with
109 distinct //. On platforms with drive letters, this generalizes
110 to making 'basename c: :' return 'c:'. This rule is captured by
111 skipping suffix stripping if base_name returned an absolute path
112 or a drive letter (only possible if name is a file-system
114 if (suffix
&& IS_RELATIVE_FILE_NAME (name
) && ! FILE_SYSTEM_PREFIX_LEN (name
))
115 remove_suffix (name
, suffix
);
117 fputs (name
, stdout
);
118 putchar (use_nuls
? '\0' : '\n');
123 main (int argc
, char **argv
)
125 bool multiple_names
= false;
126 bool use_nuls
= false;
127 char const *suffix
= nullptr;
129 initialize_main (&argc
, &argv
);
130 set_program_name (argv
[0]);
131 setlocale (LC_ALL
, "");
132 bindtextdomain (PACKAGE
, LOCALEDIR
);
133 textdomain (PACKAGE
);
135 atexit (close_stdout
);
139 int c
= getopt_long (argc
, argv
, "+as:z", longopts
, nullptr);
148 /* -s implies -a, so... */
152 multiple_names
= true;
159 case_GETOPT_HELP_CHAR
;
160 case_GETOPT_VERSION_CHAR (PROGRAM_NAME
, AUTHORS
);
163 usage (EXIT_FAILURE
);
167 if (argc
< optind
+ 1)
169 error (0, 0, _("missing operand"));
170 usage (EXIT_FAILURE
);
173 if (!multiple_names
&& optind
+ 2 < argc
)
175 error (0, 0, _("extra operand %s"), quote (argv
[optind
+ 2]));
176 usage (EXIT_FAILURE
);
181 for (; optind
< argc
; optind
++)
182 perform_basename (argv
[optind
], suffix
, use_nuls
);
185 perform_basename (argv
[optind
],
186 optind
+ 2 == argc
? argv
[optind
+ 1] : nullptr,