From d76eb9dd46081279642d3353f51ebc09fe4bbe9f Mon Sep 17 00:00:00 2001 From: David Oberhollenzer Date: Thu, 2 Apr 2020 20:03:32 +0200 Subject: Add a libcompat implementation of getopt Limited to the subset actually needed. Vaguely inspired by the NetBSD libc implementation. Signed-off-by: David Oberhollenzer --- configure.ac | 2 +- include/compat.h | 7 ++++ lib/compat/Makemodule.am | 2 +- lib/compat/getopt.c | 70 ++++++++++++++++++++++++++++++++++++ lib/compat/libcompat.vcxproj | 1 + lib/compat/libcompat.vcxproj.filters | 3 ++ 6 files changed, 83 insertions(+), 2 deletions(-) create mode 100644 lib/compat/getopt.c diff --git a/configure.ac b/configure.ac index 9da88e0..0162334 100644 --- a/configure.ac +++ b/configure.ac @@ -236,7 +236,7 @@ AX_COMPILE_CHECK_SIZEOF(long long) AC_CHECK_HEADERS([sys/xattr.h], [], []) AC_CHECK_HEADERS([sys/sysinfo.h], [], []) -AC_CHECK_FUNCS([strndup getline getsubopt]) +AC_CHECK_FUNCS([strndup getline getopt getsubopt]) ##### generate output ##### diff --git a/include/compat.h b/include/compat.h index 5d7a052..6f8a77d 100644 --- a/include/compat.h +++ b/include/compat.h @@ -197,6 +197,13 @@ char *strndup(const char *str, size_t max_len); int getsubopt(char **opt, char *const *keys, char **val); #endif +#ifndef HAVE_GETOPT +extern int optind; +extern char *optarg; + +int getopt(int argc, char *const argv[], const char *optstr); +#endif + #if defined(_WIN32) || defined(__WINDOWS__) WCHAR *path_to_windows(const char *input); #endif diff --git a/lib/compat/Makemodule.am b/lib/compat/Makemodule.am index a54059e..f1bb2de 100644 --- a/lib/compat/Makemodule.am +++ b/lib/compat/Makemodule.am @@ -1,7 +1,7 @@ libcompat_a_SOURCES = lib/compat/getline.c lib/compat/getsubopt.c libcompat_a_SOURCES += lib/compat/strndup.c lib/compat/mockups.c libcompat_a_SOURCES += lib/compat/chdir.c include/compat.h -libcompat_a_SOURCES += lib/compat/path_to_windows.c +libcompat_a_SOURCES += lib/compat/path_to_windows.c lib/compat/getopt.c noinst_LIBRARIES += libcompat.a diff --git a/lib/compat/getopt.c b/lib/compat/getopt.c new file mode 100644 index 0000000..23ede4c --- /dev/null +++ b/lib/compat/getopt.c @@ -0,0 +1,70 @@ +/* SPDX-License-Identifier: GPL-3.0-or-later */ +/* + * getopt.c + * + * Copyright (C) 2019 David Oberhollenzer + */ +#include "config.h" +#include "compat.h" + +#include +#include +#include + +#ifndef HAVE_GETOPT +static char *current = ""; +int optind = 1; +char *optarg = NULL; + +int getopt(int argc, char *const argv[], const char *optstr) +{ + char optchr, *ptr; + + if (*current == '\0') { + if (optind >= argc || argv[optind][0] != '-') + return -1; + + if (argv[optind][1] == '-' && argv[optind][2] == '\0') { + ++optind; + return -1; + } + + if (argv[optind][1] == '\0') + return -1; + + current = argv[optind] + 1; + } + + optchr = *(current++); + if (optchr == ':' || (ptr = strchr(optstr, optchr)) == NULL) + goto fail_unknown; + + if (ptr[1] == ':') { + if (*current != '\0') { + optarg = current; + } else { + if (++optind >= argc) + goto fail_arg; + + optarg = argv[optind]; + } + + current = ""; + ++optind; + } else { + optarg = NULL; + + if (*current == '\0') + ++optind; + } + + return optchr; +fail_unknown: + fprintf(stderr, "%s: unknown option `-%c`\n", argv[0], optchr); + return '?'; +fail_arg: + fprintf(stderr, "%s: missing argument for option `-%c`\n", + argv[0], optchr); + return '?'; +} +#endif diff --git a/lib/compat/libcompat.vcxproj b/lib/compat/libcompat.vcxproj index b77a39c..307754e 100644 --- a/lib/compat/libcompat.vcxproj +++ b/lib/compat/libcompat.vcxproj @@ -156,6 +156,7 @@ + diff --git a/lib/compat/libcompat.vcxproj.filters b/lib/compat/libcompat.vcxproj.filters index aa169a5..f2851d1 100644 --- a/lib/compat/libcompat.vcxproj.filters +++ b/lib/compat/libcompat.vcxproj.filters @@ -14,6 +14,9 @@ Source + + Source + Source -- cgit v1.2.3