logo

utils-std

Collection of commonly available Unix tools git clone https://anongit.hacktivis.me/git/utils-std.git/
commit: b3ec898d44678d5fbd965e2c0477f68eb0d5e656
parent 2e30e80f9d43654b40fcc8f6b3bbe1f1ae067e6f
Author: Haelwenn (lanodan) Monnier <contact@hacktivis.me>
Date:   Sat, 22 Mar 2025 04:46:58 +0100

Gracefully handle long options when unsupported

Diffstat:

MMakefile33++++++++++++++++++++++++++++-----
Mcmd/base64.c6++++--
Mcmd/basename.c4+++-
Mcmd/cat.c3++-
Mcmd/chmod.c3++-
Mcmd/chown.c3++-
Mcmd/cmp.c4+++-
Mcmd/cut.c3++-
Mcmd/date.c5+++--
Mcmd/df.c3++-
Mcmd/env.c5++++-
Mcmd/expr.y7++++---
Mcmd/getconf.c5++++-
Mcmd/head.c5+++--
Mcmd/id.c18++++++++++--------
Mcmd/install.c3++-
Mcmd/join.c3++-
Mcmd/ln.c5++++-
Mcmd/mkdir.c3++-
Mcmd/mkfifo.c4+++-
Mcmd/mknod.c4+++-
Mcmd/mktemp.c3++-
Mcmd/mv.c3++-
Mcmd/nice.c3++-
Mcmd/nproc.c4+++-
Mcmd/paste.c3++-
Mcmd/pathchk.c5++++-
Mcmd/printf.c3++-
Mcmd/pwd.c4+++-
Mcmd/realpath.c7++++---
Mcmd/renice.c6+++++-
Mcmd/rm.c3++-
Mcmd/rmdir.c5++++-
Mcmd/seq.c5++++-
Mcmd/sha1sum.c3++-
Mcmd/sha256sum.c3++-
Mcmd/sha512sum.c3++-
Mcmd/shuf.c3++-
Mcmd/split.c5+++--
Mcmd/strings.c7+++++--
Mcmd/sync.c5++++-
Mcmd/tee.c6+++++-
Mcmd/time.c6+++++-
Mcmd/timeout.c3++-
Mcmd/touch.c7+++++--
Mcmd/tr.c3++-
Mcmd/truncate.c5+++--
Mcmd/uname.c5++++-
Mcmd/uniq.c3++-
Mcmd/wc.c5+++--
Mcmd/which.c5++++-
Mcommon.mk8++++----
Alib/getopt_nolong.c22++++++++++++++++++++++
Alib/getopt_nolong.h7+++++++
Mtest-cmd/cat.sh3+--
Mtest-cmd/tee.sh2+-
56 files changed, 221 insertions(+), 81 deletions(-)

diff --git a/Makefile b/Makefile @@ -99,40 +99,63 @@ cmd/expr: cmd/expr.tab.c lib/utils.a $(CC) -std=c99 $(CFLAGS) -D_POSIX_C_SOURCE=200809L -o cmd/expr cmd/expr.tab.c lib/utils.a $(LDFLAGS) $(LDSTATIC) # Needs -lm -cmd/seq: cmd/seq.c Makefile +cmd/seq: cmd/seq.c Makefile lib/utils.a $(RM) -f ${<:=.gcov} ${@:=.gcda} ${@:=.gcno} - $(CC) -std=c99 $(CFLAGS) -o $@ cmd/seq.c -lm $(LDFLAGS) $(LDSTATIC) + $(CC) -std=c99 $(CFLAGS) -o $@ cmd/seq.c lib/utils.a -lm $(LDFLAGS) $(LDSTATIC) cmd/getconf_vars.h: cmd/getconf_vars.m4 $(M4) cmd/getconf_vars.m4 > cmd/getconf_vars.h # cmd/getconf_vars.h needs to invalidate cmd/getconf but not be in command -cmd/getconf: cmd/getconf.c cmd/getconf_vars.h +cmd/getconf: cmd/getconf.c cmd/getconf_vars.h lib/utils.a $(RM) -f ${<:=.gcov} ${@:=.gcda} ${@:=.gcno} - $(CC) -std=c99 $(CFLAGS) -o cmd/getconf cmd/getconf.c $(LDFLAGS) $(LDSTATIC) + $(CC) -std=c99 $(CFLAGS) -o cmd/getconf cmd/getconf.c lib/utils.a $(LDFLAGS) $(LDSTATIC) +cmd/base64: cmd/base64.c lib/utils.a +cmd/basename: cmd/basename.c lib/utils.a cmd/chmod: cmd/chmod.c lib/utils.a cmd/chown: cmd/chown.c lib/utils.a +cmd/cmp: cmd/cmp.c lib/utils.a +cmd/cut: cmd/cut.c lib/utils.a cmd/date: cmd/date.c lib/utils.a cmd/df: cmd/df.c lib/utils.a +cmd/env: cmd/env.c lib/utils.a cmd/head: cmd/head.c lib/utils.a +cmd/id: cmd/id.c lib/utils.a cmd/install: cmd/install.c lib/utils.a cmd/join: cmd/join.c lib/utils.a +cmd/ln: cmd/ln.c lib/utils.a cmd/mkdir: cmd/mkdir.c lib/utils.a cmd/mkfifo: cmd/mkfifo.c lib/utils.a cmd/mknod: cmd/mknod.c lib/utils.a +cmd/mktemp: cmd/mktemp.c lib/utils.a cmd/mv: cmd/mv.c lib/utils.a +cmd/nice: cmd/nice.c lib/utils.a +cmd/nproc: cmd/nproc.c lib/utils.a +cmd/od: cmd/od.c lib/utils.a cmd/paste: cmd/paste.c lib/utils.a +cmd/pathchk: cmd/pathchk.c lib/utils.a +cmd/pwd: cmd/pwd.c lib/utils.a cmd/realpath: cmd/realpath.c lib/utils.a -cmd/seq: cmd/seq.c lib/utils.a +cmd/renice: cmd/renice.c lib/utils.a +cmd/rmdir: cmd/rmdir.c lib/utils.a cmd/sha1sum: cmd/sha1sum.c lib/utils.a cmd/sha256sum: cmd/sha256sum.c lib/utils.a cmd/sha512sum: cmd/sha512sum.c lib/utils.a +cmd/shuf: cmd/shuf.c lib/utils.a cmd/sleep: cmd/sleep.c lib/utils.a cmd/split: cmd/split.c lib/utils.a +cmd/strings: cmd/strings.c lib/utils.a +cmd/sync: cmd/sync.c lib/utils.a +cmd/tee: cmd/tee.c lib/utils.a +cmd/time: cmd/time.c lib/utils.a cmd/timeout: cmd/timeout.c lib/utils.a cmd/touch: cmd/touch.c lib/utils.a cmd/truncate: cmd/truncate.c lib/utils.a +cmd/uname: cmd/uname.c lib/utils.a +cmd/uniq: cmd/uniq.c lib/utils.a +cmd/wc: cmd/wc.c lib/utils.a +cmd/which: cmd/which.c lib/utils.a cmd/whoami: cmd/whoami.c lib/utils.a lib/mode.o: lib/mode.c lib/mode.h diff --git a/cmd/base64.c b/cmd/base64.c @@ -4,6 +4,8 @@ // SPDX-License-Identifier: MPL-2.0 AND BSD-2-Clause #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <assert.h> /* assert */ #include <ctype.h> /* isspace */ #include <errno.h> /* errno */ @@ -11,7 +13,7 @@ #include <stdio.h> /* fopen(), fprintf(), BUFSIZ */ #include <stdlib.h> /* abort */ #include <string.h> /* strerror(), strncmp() */ -#include <unistd.h> /* read(), write(), close(), getopt(), opt* */ +#include <unistd.h> /* read(), write(), close() */ // 64(26+26+10+2) + NULL static const char *b64_encmap = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/"; @@ -259,7 +261,7 @@ main(int argc, char *argv[]) int ret = 0; - for(int c = -1; (c = getopt(argc, argv, ":dw:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":dw:")) != -1;) { switch(c) { diff --git a/cmd/basename.c b/cmd/basename.c @@ -4,6 +4,8 @@ #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <libgen.h> // basename #include <stdbool.h> #include <stdio.h> // puts, perror @@ -49,7 +51,7 @@ main(int argc, char *argv[]) char *suffix = NULL; char delim = '\n'; - for(int c = -1; (c = getopt(argc, argv, ":as:z")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":as:z")) != -1;) { switch(c) { diff --git a/cmd/cat.c b/cmd/cat.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L #include "../lib/fs.h" +#include "../lib/getopt_nolong.h" #include <errno.h> #include <fcntl.h> // open, O_RDONLY @@ -24,7 +25,7 @@ usage(void) int main(int argc, char *argv[]) { - for(int c = -1; (c = getopt(argc, argv, ":u")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":u")) != -1;) { switch(c) { diff --git a/cmd/chmod.c b/cmd/chmod.c @@ -10,6 +10,7 @@ #endif #include "../config.h" // HAS_* +#include "../lib/getopt_nolong.h" #include "../lib/mode.h" #include <dirent.h> // fdopendir, readdir, closedir @@ -356,7 +357,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing c = getopt_long(argc, argv, "+:cRv", opts, NULL); #else - c = getopt(argc, argv, ":cRv"); + c = getopt_nolong(argc, argv, ":cRv"); #endif if(c == -1) break; diff --git a/cmd/chown.c b/cmd/chown.c @@ -11,6 +11,7 @@ #include "../config.h" // HAS_* #include "../lib/fs.h" +#include "../lib/getopt_nolong.h" #include "../lib/user_group_parse.h" #include <dirent.h> // fdopendir, readdir, closedir @@ -220,7 +221,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing for(int c = -1; (c = getopt_long(argc, argv, "+:hRHLPv", opts, NULL)) != -1;) #else - for(int c = -1; (c = getopt(argc, argv, ":hRHLPv")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":hRHLPv")) != -1;) #endif { switch(c) diff --git a/cmd/cmp.c b/cmd/cmp.c @@ -3,6 +3,8 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <assert.h> #include <errno.h> #include <stdbool.h> @@ -97,7 +99,7 @@ main(int argc, char *argv[]) { char *endptr = NULL; - for(int c = -1; (c = getopt(argc, argv, ":ln:s")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":ln:s")) != -1;) { switch(c) { diff --git a/cmd/cut.c b/cmd/cut.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 202405L +#include "../lib/getopt_nolong.h" #include "../lib/reallocarray.h" #include <assert.h> @@ -376,7 +377,7 @@ main(int argc, char *argv[]) } errno = 0; - for(int c = -1; (c = getopt(argc, argv, ":b:c:d:f:ns")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":b:c:d:f:ns")) != -1;) { switch(c) { diff --git a/cmd/date.c b/cmd/date.c @@ -6,6 +6,7 @@ #define _POSIX_C_SOURCE 200809L #define _XOPEN_SOURCE 700 // strptime is in XSI +#include "../lib/getopt_nolong.h" #include "../lib/iso_parse.h" /* iso_parse */ #include <assert.h> @@ -16,7 +17,7 @@ #include <stdlib.h> /* strtol() */ #include <string.h> /* strerror */ #include <time.h> /* time, localtime, tm, strftime, strptime, clock_settime */ -#include <unistd.h> /* getopt(), optarg, optind */ +#include <unistd.h> /* getopt(), opt… */ const char *argv0 = "date"; @@ -151,7 +152,7 @@ main(int argc, char *argv[]) return 1; } - for(int c = -1; (c = getopt(argc, argv, ":d:f:I:jr:Ru")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":d:f:I:jr:Ru")) != -1;) { const char *errstr = NULL; switch(c) diff --git a/cmd/df.c b/cmd/df.c @@ -5,6 +5,7 @@ #define _POSIX_C_SOURCE 200809L #define _DEFAULT_SOURCE // mntent in glibc 2.19+ +#include "../lib/getopt_nolong.h" #include "../lib/humanize.h" #include <ctype.h> // iscntrl, isspace @@ -50,7 +51,7 @@ main(int argc, char *argv[]) size_t only_count = 0; char *only[4096]; - for(int c = -1; (c = getopt(argc, argv, ":ahilPkTt:x:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":ahilPkTt:x:")) != -1;) { switch(c) { diff --git a/cmd/env.c b/cmd/env.c @@ -6,6 +6,7 @@ #define _XOPEN_SOURCE 800 // wordexp #include "../config.h" // HAS_* +#include "../lib/getopt_nolong.h" #include <assert.h> // assert #include <errno.h> // errno @@ -21,6 +22,8 @@ #include <getopt.h> #endif +const char *argv0 = "env"; + extern char **environ; char *envclear[1]; @@ -75,7 +78,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing for(int c = -1; (c = getopt_long(argc, argv, "+" OPTARGS, opts, NULL)) != -1;) #else - for(int c = -1; (c = getopt(argc, argv, OPTARGS)) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, OPTARGS)) != -1;) #endif { switch(c) diff --git a/cmd/expr.y b/cmd/expr.y @@ -19,6 +19,7 @@ #include <sys/types.h> #include <ctype.h> #include "../lib/err.h" +#include "../lib/getopt_nolong.h" #include <errno.h> #include <inttypes.h> #include <limits.h> @@ -297,16 +298,16 @@ main(int argc, char *argv[]) av = argv + 1; nonposix = 1; } else { - while ((c = getopt(argc, argv, ":e")) != -1) { + while ((c = getopt_nolong(argc, argv, ":e")) != -1) { switch (c) { case 'e': nonposix = 1; break; case ':': - utils_errx(1, "Missing operand for option: '-%c'\n", optopt); + utils_errx(1, "Missing operand for option: '-%c'", optopt); break; case '?': - utils_errx(1, "Unrecognised option: '-%c'\n", optopt); + utils_errx(1, "Unrecognised option: '-%c'", optopt); break; default: abort(); diff --git a/cmd/getconf.c b/cmd/getconf.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 202405L #define _XOPEN_SOURCE 800 +#include "../lib/getopt_nolong.h" #include <assert.h> #include <errno.h> @@ -17,6 +18,8 @@ // need to be after system headers #include "./getconf_vars.h" +const char *argv0 = "getconf"; + static int print_system_var(const char *var) { @@ -133,7 +136,7 @@ main(int argc, char *argv[]) bool o_all = false; int c = -1; - while((c = getopt(argc, argv, ":av:")) != -1) + while((c = getopt_nolong(argc, argv, ":av:")) != -1) { switch(c) { diff --git a/cmd/head.c b/cmd/head.c @@ -4,7 +4,8 @@ #define _POSIX_C_SOURCE 200809L -#include "../lib/fs.h" // auto_fd_copy +#include "../lib/fs.h" // auto_fd_copy +#include "../lib/getopt_nolong.h" #include "../lib/truncation.h" // apply_size_suffix #include <ctype.h> // isdigit @@ -168,7 +169,7 @@ main(int argc, char *argv[]) continue; } - int c = getopt(argc, argv, ":qvc:n:z"); + int c = getopt_nolong(argc, argv, ":qvc:n:z"); if(c == -1) break; switch(c) diff --git a/cmd/id.c b/cmd/id.c @@ -3,13 +3,15 @@ // SPDX-License-Identifier: MPL-2.0 #define _DEFAULT_SOURCE // getgrouplist (4.4BSD+) -#include <grp.h> // getgrgid, getgroups, getgrouplist(glibc) -#include <pwd.h> // getpwuid -#include <stdbool.h> // bool -#include <stdio.h> // printf, perror -#include <stdlib.h> // calloc, free -#include <sys/types.h> // uid_t -#include <unistd.h> // getuid, getgid, getopt, opt*, getgrouplist(FreeBSD, NetBSD) +#include "../lib/getopt_nolong.h" + +#include <grp.h> // getgrgid, getgroups, getgrouplist(glibc) +#include <pwd.h> // getpwuid +#include <stdbool.h> // bool +#include <stdio.h> // printf, perror +#include <stdlib.h> // calloc, free +#include <sys/types.h> // uid_t +#include <unistd.h> // getuid, getgid, getopt, opt*, getgrouplist(FreeBSD, NetBSD) const char *argv0 = "id"; bool name_flag = false; @@ -188,7 +190,7 @@ main(int argc, char *argv[]) struct passwd pw = {.pw_uid = uid, .pw_gid = gid}; struct passwd epw = {.pw_uid = euid, .pw_gid = egid}; - for(int c = -1; (c = getopt(argc, argv, ":Ggunr")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":Ggunr")) != -1;) { switch(c) { diff --git a/cmd/install.c b/cmd/install.c @@ -8,6 +8,7 @@ #include "../config.h" // HAS_* #include "../lib/fs.h" +#include "../lib/getopt_nolong.h" #include "../lib/lib_mkdir.h" #include "../lib/mode.h" #include "../lib/user_group_parse.h" @@ -227,7 +228,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing for(int c = -1; (c = getopt_long(argc, argv, "+:CcDdpTt:g:m:o:v", opts, NULL)) != -1;) #else - for(int c = -1; (c = getopt(argc, argv, ":CcDdpTt:g:m:o:v")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":CcDdpTt:g:m:o:v")) != -1;) #endif { switch(c) diff --git a/cmd/join.c b/cmd/join.c @@ -36,6 +36,7 @@ #define _BSD_SOURCE // fgetln, strsep #include "../lib/err.h" +#include "../lib/getopt_nolong.h" #include <errno.h> #include <limits.h> @@ -127,7 +128,7 @@ main(int argc, char *argv[]) F2 = &input2; aflag = vflag = 0; - while((ch = getopt(argc, argv, ":a:e:j:1:2:o:t:v:")) != -1) + while((ch = getopt_nolong(argc, argv, ":a:e:j:1:2:o:t:v:")) != -1) { switch(ch) { diff --git a/cmd/ln.c b/cmd/ln.c @@ -5,6 +5,7 @@ #define _POSIX_C_SOURCE 200809L #include "../lib/bitmasks.h" +#include "../lib/getopt_nolong.h" // NetBSD (9.3 and 10) hides symlink behind _XOPEN_SOURCE / _NETBSD_SOURCE #ifdef __NetBSD__ @@ -21,6 +22,8 @@ #include <sys/stat.h> #include <unistd.h> // getopt, symlink, link +const char *argv0 = "ln"; + static bool opt_s = false, force = false; static int link_flags = 0; static int open_dir_flags = O_RDONLY | O_DIRECTORY; @@ -136,7 +139,7 @@ main(int argc, char *argv[]) { bool verbose = false; - for(int c = -1; (c = getopt(argc, argv, ":fnsLPv")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":fnsLPv")) != -1;) { switch(c) { diff --git a/cmd/mkdir.c b/cmd/mkdir.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include "../lib/lib_mkdir.h" #include "../lib/mode.h" @@ -53,7 +54,7 @@ main(int argc, char *argv[]) const char *errstr = NULL; // clang-format on - for(int c = -1; (c = getopt(argc, argv, ":pvm:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":pvm:")) != -1;) { switch(c) { diff --git a/cmd/mkfifo.c b/cmd/mkfifo.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include "../lib/mode.h" #include <errno.h> @@ -14,6 +15,7 @@ #include <unistd.h> // getopt mode_t filemask; +const char *argv0 = "mkfifo"; static void usage(void) @@ -28,7 +30,7 @@ main(int argc, char *argv[]) const char *errstr = NULL; - for(int c = -1; (c = getopt(argc, argv, ":m:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":m:")) != -1;) { switch(c) { diff --git a/cmd/mknod.c b/cmd/mknod.c @@ -5,6 +5,7 @@ #define _POSIX_C_SOURCE 200809L #define _XOPEN_SOURCE 700 // mknod is in XSI +#include "../lib/getopt_nolong.h" #include "../lib/mode.h" #include <errno.h> @@ -16,6 +17,7 @@ #include <unistd.h> // getopt mode_t filemask; +const char *argv0 = "mknod"; static void usage(void) @@ -52,7 +54,7 @@ main(int argc, char *argv[]) const char *errstr = NULL; - for(int c = -1; (c = getopt(argc, argv, ":m:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":m:")) != -1;) { switch(c) { diff --git a/cmd/mktemp.c b/cmd/mktemp.c @@ -5,6 +5,7 @@ // getentropy got in POSIX.1-2024 but defining _POSIX_C_SOURCE causes too much side-effects #define _DEFAULT_SOURCE // getentropy #include "../config.h" // HAS_* +#include "../lib/getopt_nolong.h" #include <errno.h> #include <limits.h> // PATH_MAX @@ -96,7 +97,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing for(int c = -1; (c = getopt_long(argc, argv, "+:dqp:tu", opts, NULL)) != -1;) #else - for(int c = -1; (c = getopt(argc, argv, ":dqp:tu")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":dqp:tu")) != -1;) #endif { switch(c) diff --git a/cmd/mv.c b/cmd/mv.c @@ -13,6 +13,7 @@ #include "../lib/consent.h" #include "../lib/fs.h" +#include "../lib/getopt_nolong.h" #include <dirent.h> // fdopendir #include <errno.h> @@ -392,7 +393,7 @@ main(int argc, char *argv[]) .sep = "", }; - for(int c = -1; (c = getopt(argc, argv, ":fint:v")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":fint:v")) != -1;) { switch(c) { diff --git a/cmd/nice.c b/cmd/nice.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L #define _XOPEN_SOURCE 700 // nice() is in XSI +#include "../lib/getopt_nolong.h" #include <assert.h> #include <errno.h> @@ -25,7 +26,7 @@ main(int argc, char *argv[]) { long incr = 0; - for(int c = -1; (c = getopt(argc, argv, ":n:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":n:")) != -1;) { char *endptr = NULL; diff --git a/cmd/nproc.c b/cmd/nproc.c @@ -9,6 +9,8 @@ // Sadly {Free,Net}BSD hides _SC_NPROCESSORS_{CONF,ONLN} if _POSIX_C_SOURCE is defined *sigh* // #define _POSIX_C_SOURCE 202405L +#include "../lib/getopt_nolong.h" + #include <errno.h> #include <stdio.h> // printf #include <string.h> // strerror @@ -29,7 +31,7 @@ main(int argc, char *argv[]) int target = _SC_NPROCESSORS_ONLN; const char *target_str = "_SC_NPROCESSORS_ONLN"; - for(int c = -1; (c = getopt(argc, argv, ":a")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":a")) != -1;) { switch(c) { diff --git a/cmd/paste.c b/cmd/paste.c @@ -35,6 +35,7 @@ #define _POSIX_C_SOURCE 200809L #include "../lib/err.h" +#include "../lib/getopt_nolong.h" #include <errno.h> #include <limits.h> @@ -223,7 +224,7 @@ main(int argc, char *argv[]) int seq = 0; - for(int c = -1; (c = getopt(argc, argv, ":d:sz")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":d:sz")) != -1;) { switch(c) { diff --git a/cmd/pathchk.c b/cmd/pathchk.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include <errno.h> #include <limits.h> // PATH_MAX @@ -12,6 +13,8 @@ #include <sys/stat.h> // lstat #include <unistd.h> // getopt +const char *argv0 = "pathchk"; + // POSIX Portable Character Set // Returns 0 on success, or the first invalid character found static char @@ -42,7 +45,7 @@ main(int argc, char *argv[]) size_t path_max = PATH_MAX - 1; size_t name_max = NAME_MAX; - for(int c = -1; (c = getopt(argc, argv, ":pP")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":pP")) != -1;) { switch(c) { diff --git a/cmd/printf.c b/cmd/printf.c @@ -39,6 +39,7 @@ #define _POSIX_C_SOURCE 200809L #include "../lib/err.h" +#include "../lib/getopt_nolong.h" #include <assert.h> #include <ctype.h> @@ -105,7 +106,7 @@ main(int argc, char *argv[]) strerror(errno)); } - for(int c = -1; (c = getopt(argc, argv, "")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, "")) != -1;) { switch(c) { diff --git a/cmd/pwd.c b/cmd/pwd.c @@ -3,6 +3,8 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <limits.h> // PATH_MAX #include <stdbool.h> #include <stdio.h> // puts, perror, printf @@ -74,7 +76,7 @@ main(int argc, char *argv[]) { enum pwd mode = PWD_L; - for(int c = -1; (c = getopt(argc, argv, ":LP")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":LP")) != -1;) { switch(c) { diff --git a/cmd/realpath.c b/cmd/realpath.c @@ -6,6 +6,7 @@ #define _XOPEN_SOURCE 700 // realpath is in XSI #include "../lib/fs.h" +#include "../lib/getopt_nolong.h" #include <errno.h> #include <limits.h> // PATH_MAX @@ -18,7 +19,7 @@ static bool must_exists = false; static bool offline = false; -static char *argv0 = NULL; +const char *argv0 = NULL; static char sep = '\n'; static int @@ -97,7 +98,7 @@ main_realpath(int argc, char *argv[]) int offset_sep = 0; - for(int c = -1; (c = getopt(argc, argv, ":Eemnsz")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":Eemnsz")) != -1;) { switch(c) { @@ -161,7 +162,7 @@ main_readlink(int argc, char *argv[]) int offset_sep = 0; - for(int c = -1; (c = getopt(argc, argv, ":femnz")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":femnz")) != -1;) { switch(c) { diff --git a/cmd/renice.c b/cmd/renice.c @@ -5,6 +5,8 @@ #define _POSIX_C_SOURCE 200809L #define _XOPEN_SOURCE 700 // getpriority, setpriority +#include "../lib/getopt_nolong.h" + #include <errno.h> #include <stdio.h> // fprintf #include <stdlib.h> // abort @@ -12,6 +14,8 @@ #include <sys/resource.h> // getpriority, setpriority #include <unistd.h> // getopt, nice +const char *argv0 = "renice"; + static int renice(int which, id_t who, int adj) { @@ -71,7 +75,7 @@ main(int argc, char *argv[]) long adj = 0; int which = PRIO_PROCESS; - for(int c = -1; (c = getopt(argc, argv, ":gpun:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":gpun:")) != -1;) { char *endptr = NULL; diff --git a/cmd/rm.c b/cmd/rm.c @@ -10,6 +10,7 @@ #endif #include "../lib/consent.h" +#include "../lib/getopt_nolong.h" #include <ctype.h> // isprint #include <dirent.h> // fdopendir, readdir, closedir @@ -169,7 +170,7 @@ usage(void) int main(int argc, char *argv[]) { - for(int c = -1; (c = getopt(argc, argv, ":dfirRv")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":dfirRv")) != -1;) { switch(c) { diff --git a/cmd/rmdir.c b/cmd/rmdir.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L #include "../config.h" // HAS_* +#include "../lib/getopt_nolong.h" #include <errno.h> #include <stdbool.h> @@ -15,6 +16,8 @@ #include <getopt.h> #endif +const char *argv0 = "rmdir"; + static void usage(void) { @@ -46,7 +49,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing for(int c = -1; (c = getopt_long(argc, argv, "+:pv", opts, NULL)) != -1;) #else - for(int c = -1; (c = getopt(argc, argv, ":pv")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":pv")) != -1;) #endif { switch(c) diff --git a/cmd/seq.c b/cmd/seq.c @@ -3,6 +3,8 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <ctype.h> // isdigit #include <errno.h> // errno #include <math.h> // fabs @@ -12,6 +14,7 @@ #include <string.h> // strerror #include <unistd.h> // getopt, optarg, optind +const char *argv0 = "seq"; const char *sep = "\n"; const char *term = "\n"; @@ -78,7 +81,7 @@ usage(void) int main(int argc, char *argv[]) { - for(int c = -1; (c = getopt(argc, argv, ":ws:t:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":ws:t:")) != -1;) { switch(c) { diff --git a/cmd/sha1sum.c b/cmd/sha1sum.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include "../lib/sha1.h" // sha1_* #include "../lib/strconv.h" // bytes2hex @@ -167,7 +168,7 @@ main(int argc, char *argv[]) { bool opt_c = false; - for(int c = -1; (c = getopt(argc, argv, ":c")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":c")) != -1;) { switch(c) { diff --git a/cmd/sha256sum.c b/cmd/sha256sum.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include "../lib/sha256.h" // sha256_* #include "../lib/strconv.h" // bytes2hex @@ -167,7 +168,7 @@ main(int argc, char *argv[]) { bool opt_c = false; - for(int c = -1; (c = getopt(argc, argv, ":c")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":c")) != -1;) { switch(c) { diff --git a/cmd/sha512sum.c b/cmd/sha512sum.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include "../lib/sha512.h" // sha512_* #include "../lib/strconv.h" // bytes2hex @@ -167,7 +168,7 @@ main(int argc, char *argv[]) { bool opt_c = false; - for(int c = -1; (c = getopt(argc, argv, ":c")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":c")) != -1;) { switch(c) { diff --git a/cmd/shuf.c b/cmd/shuf.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include <errno.h> #include <stdbool.h> @@ -92,7 +93,7 @@ main(int argc, char *argv[]) bool e_flag = false; srand((int)time(NULL)); - for(int c = -1; (c = getopt(argc, argv, ":en:z")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":en:z")) != -1;) { char *endptr = NULL; diff --git a/cmd/split.c b/cmd/split.c @@ -4,7 +4,8 @@ #define _POSIX_C_SOURCE 200809L -#include "../lib/fs.h" // auto_file_copy +#include "../lib/fs.h" // auto_file_copy +#include "../lib/getopt_nolong.h" #include "../lib/truncation.h" // apply_size_suffix #include <errno.h> @@ -233,7 +234,7 @@ static const char *error_opt_b_l = "%s: error: Options -b and -l are mutually ex int main(int argc, char *argv[]) { - for(int c = -1; (c = getopt(argc, argv, ":a:b:l:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":a:b:l:")) != -1;) { char *endptr = NULL; diff --git a/cmd/strings.c b/cmd/strings.c @@ -3,6 +3,8 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <ctype.h> /* isprint() */ #include <errno.h> /* errno */ #include <fcntl.h> /* open(), O_RDONLY */ @@ -10,8 +12,9 @@ #include <stdio.h> /* fprintf(), BUFSIZ */ #include <stdlib.h> /* strtol() */ #include <string.h> /* strerror(), strncmp() */ -#include <unistd.h> /* read(), write(), close(), getopt(), optarg, optind */ +#include <unistd.h> /* read(), write(), close() */ +const char *argv0 = "strings"; size_t opt_min_strlen = 4; const char *opt_offset_format = NULL; int delim = '\n'; @@ -102,7 +105,7 @@ usage(void) int main(int argc, char *argv[]) { - for(int c = -1; (c = getopt(argc, argv, ":an:t:z")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":an:t:z")) != -1;) { char *endptr = NULL; diff --git a/cmd/sync.c b/cmd/sync.c @@ -7,6 +7,7 @@ #define _XOPEN_SOURCE 700 // sync #include "../config.h" // HAS_* +#include "../lib/getopt_nolong.h" #include <errno.h> #include <fcntl.h> // open, O_* @@ -16,13 +17,15 @@ #include <string.h> // strerror #include <unistd.h> // fsync, sync, getopt, syncfs +const char *argv0 = "sync"; + int main(int argc, char *argv[]) { int err = 0; int (*sync_func)(int) = fsync; - for(int c = -1; (c = getopt(argc, argv, ":df")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":df")) != -1;) { switch(c) { diff --git a/cmd/tee.c b/cmd/tee.c @@ -3,6 +3,8 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <assert.h> /* assert() */ #include <errno.h> /* errno */ #include <signal.h> /* signal() */ @@ -11,6 +13,8 @@ #include <string.h> /* strerror() */ #include <unistd.h> /* getopt(), opt… */ +const char *argv0 = "tee"; + static void cleanup(FILE **fds) { @@ -27,7 +31,7 @@ main(int argc, char *argv[]) FILE **fds = {NULL}; // Shut up GCC int c; - for(int c = -1; (c = getopt(argc, argv, ":ai")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":ai")) != -1;) { switch(c) { diff --git a/cmd/time.c b/cmd/time.c @@ -3,6 +3,8 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" + #include <errno.h> // errno #include <stdio.h> // perror, fprintf #include <stdlib.h> // abort @@ -12,6 +14,8 @@ #include <sys/wait.h> // waitpid #include <unistd.h> // sysconf, fork, execvp, getopt +const char *argv0 = "time"; + enum cmd_time_mode { CMD_TIME_POSIX = 0, @@ -37,7 +41,7 @@ main(int argc, char *argv[]) return 0; } - for(int c = -1; (c = getopt(argc, argv, ":pv")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":pv")) != -1;) { switch(c) { diff --git a/cmd/timeout.c b/cmd/timeout.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L #define _DEFAULT_SOURCE // For NSIG in sys_signame.h, thanks glibc +#include "../lib/getopt_nolong.h" #include "../lib/strtodur.h" #include "../lib/sys_signame.h" @@ -60,7 +61,7 @@ main(int argc, char *argv[]) char *arg = NULL; - for(int c = -1; (c = getopt(argc, argv, ":fk:ps:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":fk:ps:")) != -1;) { switch(c) { diff --git a/cmd/touch.c b/cmd/touch.c @@ -6,7 +6,8 @@ #define _XOPEN_SOURCE 700 // strptime (NetBSD) #define _POSIX_C_SOURCE 200809L // O_NOFOLLOW, st_atim/st_mtim -#include "../lib/bitmasks.h" /* FIELD_* */ +#include "../lib/bitmasks.h" /* FIELD_* */ +#include "../lib/getopt_nolong.h" #include "../lib/iso_parse.h" /* iso_parse */ #include <assert.h> @@ -19,6 +20,8 @@ #include <time.h> /* mktime */ #include <unistd.h> /* getopt, opt*, close */ +const char *argv0 = "touch"; + // [[CC]YY]MMDDhhmm[.SS] static struct timespec opt_t_parse(char *arg, const char **errstr) @@ -123,7 +126,7 @@ main(int argc, char *argv[]) int open_flags = O_WRONLY | O_CREAT | O_NOCTTY; int utimensat_flags = 0; - for(int c = -1; (c = getopt(argc, argv, ":acfhmr:t:d:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":acfhmr:t:d:")) != -1;) { const char *errstr = NULL; diff --git a/cmd/tr.c b/cmd/tr.c @@ -33,6 +33,7 @@ #define _DEFAULT_SOURCE #include "../lib/err.h" +#include "../lib/getopt_nolong.h" #include "../lib/tr_str.h" #include <stdio.h> @@ -77,7 +78,7 @@ main(int argc, char *argv[]) int cflag, dflag, sflag; cflag = dflag = sflag = 0; - for(int c = -1; (c = getopt(argc, argv, "Ccds")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, "Ccds")) != -1;) { switch(c) { diff --git a/cmd/truncate.c b/cmd/truncate.c @@ -4,7 +4,8 @@ #define _POSIX_C_SOURCE 200809L -#include "../lib/bitmasks.h" // FIELD_CLR +#include "../lib/bitmasks.h" // FIELD_CLR +#include "../lib/getopt_nolong.h" #include "../lib/truncation.h" // parse_size #include <errno.h> @@ -36,7 +37,7 @@ main(int argc, char *argv[]) }; char *ref_file = NULL; - for(int c = -1; (c = getopt(argc, argv, ":cr:s:")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":cr:s:")) != -1;) { switch(c) { diff --git a/cmd/uname.c b/cmd/uname.c @@ -4,6 +4,7 @@ #define _POSIX_C_SOURCE 200809L #include "../lib/bitmasks.h" +#include "../lib/getopt_nolong.h" #include <errno.h> #include <stdio.h> // printf @@ -12,6 +13,8 @@ #include <sys/utsname.h> // uname #include <unistd.h> // getopt +const char *argv0 = "uname"; + enum uname_names { CMD_UNAME_SYSNAME = 1 << 1, @@ -49,7 +52,7 @@ usage(void) int main(int argc, char *argv[]) { - for(int c = -1; (c = getopt(argc, argv, ":amnprsv")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":amnprsv")) != -1;) { switch(c) { diff --git a/cmd/uniq.c b/cmd/uniq.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include <ctype.h> // isblank #include <errno.h> @@ -30,7 +31,7 @@ main(int argc, char *argv[]) char *endptr = NULL; - for(int c = -1; (c = getopt(argc, argv, ":cdf:s:u")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":cdf:s:u")) != -1;) { switch(c) { diff --git a/cmd/wc.c b/cmd/wc.c @@ -5,6 +5,7 @@ #define _POSIX_C_SOURCE 200809L #include "../config.h" // HAS_* #include "../lib/bitmasks.h" +#include "../lib/getopt_nolong.h" #include <ctype.h> // isspace #include <errno.h> @@ -26,7 +27,7 @@ #define WC_BUFSIZ 16320 static char buf[WC_BUFSIZ] = ""; -static const char *argv0 = "wc"; +const char *argv0 = "wc"; static enum { WC_OPT_C = 1 << 0, @@ -237,7 +238,7 @@ main(int argc, char *argv[]) // Need + as first character to get POSIX-style option parsing for(int c = -1; (c = getopt_long(argc, argv, "+:clmw", opts, NULL)) != -1;) #else - for(int c = -1; (c = getopt(argc, argv, ":clmw")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, ":clmw")) != -1;) #endif { switch(c) diff --git a/cmd/which.c b/cmd/which.c @@ -3,6 +3,7 @@ // SPDX-License-Identifier: MPL-2.0 #define _POSIX_C_SOURCE 200809L +#include "../lib/getopt_nolong.h" #include <limits.h> // PATH_MAX #include <stdbool.h> @@ -11,6 +12,8 @@ #include <string.h> // strtok #include <unistd.h> // access, getopt +const char *argv0 = "which"; + int main(int argc, char *argv[]) { @@ -24,7 +27,7 @@ main(int argc, char *argv[]) return 1; } - for(int c = -1; (c = getopt(argc, argv, "as")) != -1;) + for(int c = -1; (c = getopt_nolong(argc, argv, "as")) != -1;) { switch(c) { diff --git a/common.mk b/common.mk @@ -6,7 +6,7 @@ lib/fs.o: lib/fs.c lib/fs.h config.h lib/err.o: lib/err.c lib/err.h lib/consent.o: lib/consent.c lib/consent.h lib/tr_str.o: lib/tr_str.c lib/tr_str.h -cmd/cat: cmd/cat.c lib/fs.o -cmd/printf: cmd/printf.c lib/err.o -cmd/rm: cmd/rm.c lib/consent.o -cmd/tr: cmd/tr.c lib/tr_str.o lib/err.o +cmd/cat: cmd/cat.c lib/fs.o lib/getopt_nolong.o +cmd/printf: cmd/printf.c lib/err.o lib/getopt_nolong.o +cmd/rm: cmd/rm.c lib/consent.o lib/getopt_nolong.o +cmd/tr: cmd/tr.c lib/tr_str.o lib/err.o lib/getopt_nolong.o diff --git a/lib/getopt_nolong.c b/lib/getopt_nolong.c @@ -0,0 +1,22 @@ +// utils-std: Collection of commonly available Unix tools +// SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me> +// SPDX-License-Identifier: MPL-2.0 + +#define _POSIX_C_SOURCE 200809L + +#include "getopt_nolong.h" + +#include <stdio.h> // fprintf +#include <unistd.h> // getopt + +int +getopt_nolong(int argc, char *const argv[], const char *optstring) +{ + if(argv[optind][0] == '-' && argv[optind][1] == '-' && argv[optind][2] != '\0') + { + fprintf(stderr, "%s: error: Long options unsupported: '%s'\n", argv0, argv[optind]); + return -1; + } + + return getopt(argc, argv, optstring); +} diff --git a/lib/getopt_nolong.h b/lib/getopt_nolong.h @@ -0,0 +1,7 @@ +// utils-std: Collection of commonly available Unix tools +// SPDX-FileCopyrightText: 2017 Haelwenn (lanodan) Monnier <contact+utils@hacktivis.me> +// SPDX-License-Identifier: MPL-2.0 + +extern const char *argv0; + +int getopt_nolong(int argc, char *const argv[], const char *optstring); diff --git a/test-cmd/cat.sh b/test-cmd/cat.sh @@ -47,6 +47,5 @@ t --exit=1 enoent /var/empty/e/no/ent "cat: error: Failed opening file '/var/emp t_file doubledash "$WD/inputs/all_bytes" -- "$WD/inputs/all_bytes" -t --exit=1 tripledash "--- $WD/inputs/all_bytes" "cat: error: Unrecognised option: '--' -Usage: cat [-u] [files ...] +t --exit=1 tripledash "--- $WD/inputs/all_bytes" "cat: error: Long options unsupported: '---' " diff --git a/test-cmd/tee.sh b/test-cmd/tee.sh @@ -45,5 +45,5 @@ t --exit=1 --input='foo' enoent '/var/empty/e/no/ent' 'tee: error: Failed openin t_file --infile="${WD}/inputs/all_bytes" doubledash "${WD}/inputs/all_bytes" -- -t --exit=1 --input='foo' tripledash '---' "tee: error: Unrecognised option: '--' +t --exit=1 --input='foo' tripledash '---' "tee: error: Long options unsupported: '---' "