123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612 |
- #include <errno.h>
- #include "getopt_long.h"
- #include <stdlib.h>
- #include <stdio.h>
- #include <string.h>
- #include <stdarg.h>
- #define GNU_COMPATIBLE
- #define PRINT_ERROR ((opterr) && (*options != ':'))
- #define FLAG_PERMUTE 0x01
- #define FLAG_ALLARGS 0x02
- #define FLAG_LONGONLY 0x04
- #define BADCH (int)'?'
- #define BADARG ((*options == ':') ? (int)':' : (int)'?')
- #define INORDER (int)1
- #define EMSG ""
- #ifdef GNU_COMPATIBLE
- #define NO_PREFIX (-1)
- #define D_PREFIX 0
- #define DD_PREFIX 1
- #define W_PREFIX 2
- #endif
- char *optarg;
- int optind, opterr = 1, optopt;
- static int getopt_internal(int, char * const *, const char *,
- const struct option *, int *, int);
- static int parse_long_options(char * const *, const char *,
- const struct option *, int *, int, int);
- static int gcd(int, int);
- static void permute_args(int, int, int, char * const *);
- static const char *place = EMSG;
- static int nonopt_start = -1;
- static int nonopt_end = -1;
- static const char recargchar[] = "option requires an argument -- %c";
- static const char illoptchar[] = "illegal option -- %c";
- #ifdef GNU_COMPATIBLE
- static int dash_prefix = NO_PREFIX;
- static const char gnuoptchar[] = "invalid option -- %c";
- static const char recargstring[] = "option `%s%s' requires an argument";
- static const char ambig[] = "option `%s%.*s' is ambiguous";
- static const char noarg[] = "option `%s%.*s' doesn't allow an argument";
- static const char illoptstring[] = "unrecognized option `%s%s'";
- #else
- static const char recargstring[] = "option requires an argument -- %s";
- static const char ambig[] = "ambiguous option -- %.*s";
- static const char noarg[] = "option doesn't take an argument -- %.*s";
- static const char illoptstring[] = "unknown option -- %s";
- #endif
- static int
- gcd(int a, int b)
- {
- int c;
- c = a % b;
- while (c != 0) {
- a = b;
- b = c;
- c = a % b;
- }
- return (b);
- }
- static void
- permute_args(int panonopt_start, int panonopt_end, int opt_end,
- char * const *nargv)
- {
- int cstart, cyclelen, i, j, ncycle, nnonopts, nopts, pos;
- char *swap;
-
- nnonopts = panonopt_end - panonopt_start;
- nopts = opt_end - panonopt_end;
- ncycle = gcd(nnonopts, nopts);
- cyclelen = (opt_end - panonopt_start) / ncycle;
- for (i = 0; i < ncycle; i++) {
- cstart = panonopt_end+i;
- pos = cstart;
- for (j = 0; j < cyclelen; j++) {
- if (pos >= panonopt_end)
- pos -= nnonopts;
- else
- pos += nopts;
- swap = nargv[pos];
-
- ((char **) nargv)[pos] = nargv[cstart];
-
- ((char **)nargv)[cstart] = swap;
- }
- }
- }
- static void
- warnx(const char *fmt, ...)
- {
- extern char *program_name;
- va_list ap;
- va_start(ap, fmt);
- fprintf(stderr, "%s: ", program_name);
- vfprintf(stderr, fmt, ap);
- fprintf(stderr, "\n");
- va_end(ap);
- }
- static int
- parse_long_options(char * const *nargv, const char *options,
- const struct option *long_options, int *idx, int short_too, int flags)
- {
- const char *current_argv, *has_equal;
- #ifdef GNU_COMPATIBLE
- const char *current_dash;
- #endif
- size_t current_argv_len;
- int i, match, exact_match, second_partial_match;
- current_argv = place;
- #ifdef GNU_COMPATIBLE
- switch (dash_prefix) {
- case D_PREFIX:
- current_dash = "-";
- break;
- case DD_PREFIX:
- current_dash = "--";
- break;
- case W_PREFIX:
- current_dash = "-W ";
- break;
- default:
- current_dash = "";
- break;
- }
- #endif
- match = -1;
- exact_match = 0;
- second_partial_match = 0;
- optind++;
- if ((has_equal = strchr(current_argv, '=')) != NULL) {
-
- current_argv_len = has_equal - current_argv;
- has_equal++;
- } else
- current_argv_len = strlen(current_argv);
- for (i = 0; long_options[i].name; i++) {
-
- if (strncmp(current_argv, long_options[i].name,
- current_argv_len))
- continue;
- if (strlen(long_options[i].name) == current_argv_len) {
-
- match = i;
- exact_match = 1;
- break;
- }
-
- if (short_too && current_argv_len == 1)
- continue;
- if (match == -1)
- match = i;
- else if ((flags & FLAG_LONGONLY) ||
- long_options[i].has_arg !=
- long_options[match].has_arg ||
- long_options[i].flag != long_options[match].flag ||
- long_options[i].val != long_options[match].val)
- second_partial_match = 1;
- }
- if (!exact_match && second_partial_match) {
-
- if (PRINT_ERROR)
- warnx(ambig,
- #ifdef GNU_COMPATIBLE
- current_dash,
- #endif
- (int)current_argv_len,
- current_argv);
- optopt = 0;
- return (BADCH);
- }
- if (match != -1) {
- if (long_options[match].has_arg == no_argument
- && has_equal) {
- if (PRINT_ERROR)
- warnx(noarg,
- #ifdef GNU_COMPATIBLE
- current_dash,
- #endif
- (int)current_argv_len,
- current_argv);
-
- if (long_options[match].flag == NULL)
- optopt = long_options[match].val;
- else
- optopt = 0;
- #ifdef GNU_COMPATIBLE
- return (BADCH);
- #else
- return (BADARG);
- #endif
- }
- if (long_options[match].has_arg == required_argument ||
- long_options[match].has_arg == optional_argument) {
- if (has_equal)
- optarg = (char *)has_equal;
- else if (long_options[match].has_arg ==
- required_argument) {
-
- optarg = nargv[optind++];
- }
- }
- if ((long_options[match].has_arg == required_argument)
- && (optarg == NULL)) {
-
- if (PRINT_ERROR)
- warnx(recargstring,
- #ifdef GNU_COMPATIBLE
- current_dash,
- #endif
- current_argv);
-
- if (long_options[match].flag == NULL)
- optopt = long_options[match].val;
- else
- optopt = 0;
- --optind;
- return (BADARG);
- }
- } else {
- if (short_too) {
- --optind;
- return (-1);
- }
- if (PRINT_ERROR)
- warnx(illoptstring,
- #ifdef GNU_COMPATIBLE
- current_dash,
- #endif
- current_argv);
- optopt = 0;
- return (BADCH);
- }
- if (idx)
- *idx = match;
- if (long_options[match].flag) {
- *long_options[match].flag = long_options[match].val;
- return (0);
- } else
- return (long_options[match].val);
- }
- static int
- getopt_internal(int nargc, char * const *nargv, const char *options,
- const struct option *long_options, int *idx, int flags)
- {
- char *oli;
- int optchar, short_too;
- int posixly_correct;
- if (options == NULL)
- return (-1);
-
- posixly_correct = (getenv("POSIXLY_CORRECT") != NULL);
- #ifdef GNU_COMPATIBLE
- if (*options == '-')
- flags |= FLAG_ALLARGS;
- else if (posixly_correct || *options == '+')
- flags &= ~FLAG_PERMUTE;
- #else
- if (posixly_correct || *options == '+')
- flags &= ~FLAG_PERMUTE;
- else if (*options == '-')
- flags |= FLAG_ALLARGS;
- #endif
- if (*options == '+' || *options == '-')
- options++;
-
- if (optind == 0)
- optind = 1;
- optarg = NULL;
- start:
- if (!*place) {
- if (optind >= nargc) {
- place = EMSG;
- if (nonopt_end != -1) {
-
- permute_args(nonopt_start, nonopt_end,
- optind, nargv);
- optind -= nonopt_end - nonopt_start;
- }
- else if (nonopt_start != -1) {
-
- optind = nonopt_start;
- }
- nonopt_start = nonopt_end = -1;
- return (-1);
- }
- if (*(place = nargv[optind]) != '-' ||
- #ifdef GNU_COMPATIBLE
- place[1] == '\0') {
- #else
- (place[1] == '\0' && strchr(options, '-') == NULL)) {
- #endif
- place = EMSG;
- if (flags & FLAG_ALLARGS) {
-
- optarg = nargv[optind++];
- return (INORDER);
- }
- if (!(flags & FLAG_PERMUTE)) {
-
- return (-1);
- }
-
- if (nonopt_start == -1)
- nonopt_start = optind;
- else if (nonopt_end != -1) {
- permute_args(nonopt_start, nonopt_end,
- optind, nargv);
- nonopt_start = optind -
- (nonopt_end - nonopt_start);
- nonopt_end = -1;
- }
- optind++;
-
- goto start;
- }
- if (nonopt_start != -1 && nonopt_end == -1)
- nonopt_end = optind;
-
- if (place[1] != '\0' && *++place == '-' && place[1] == '\0') {
- optind++;
- place = EMSG;
-
- if (nonopt_end != -1) {
- permute_args(nonopt_start, nonopt_end,
- optind, nargv);
- optind -= nonopt_end - nonopt_start;
- }
- nonopt_start = nonopt_end = -1;
- return (-1);
- }
- }
-
- if (long_options != NULL && place != nargv[optind] &&
- (*place == '-' || (flags & FLAG_LONGONLY))) {
- short_too = 0;
- #ifdef GNU_COMPATIBLE
- dash_prefix = D_PREFIX;
- #endif
- if (*place == '-') {
- place++;
- #ifdef GNU_COMPATIBLE
- dash_prefix = DD_PREFIX;
- #endif
- } else if (*place != ':' && strchr(options, *place) != NULL)
- short_too = 1;
- optchar = parse_long_options(nargv, options, long_options,
- idx, short_too, flags);
- if (optchar != -1) {
- place = EMSG;
- return (optchar);
- }
- }
- if ((optchar = (int)*place++) == (int)':' ||
- (optchar == (int)'-' && *place != '\0') ||
- (oli = strchr(options, optchar)) == NULL) {
-
- if (optchar == (int)'-' && *place == '\0')
- return (-1);
- if (!*place)
- ++optind;
- #ifdef GNU_COMPATIBLE
- if (PRINT_ERROR)
- warnx(posixly_correct ? illoptchar : gnuoptchar,
- optchar);
- #else
- if (PRINT_ERROR)
- warnx(illoptchar, optchar);
- #endif
- optopt = optchar;
- return (BADCH);
- }
- if (long_options != NULL && optchar == 'W' && oli[1] == ';') {
-
- if (*place)
- ;
- else if (++optind >= nargc) {
- place = EMSG;
- if (PRINT_ERROR)
- warnx(recargchar, optchar);
- optopt = optchar;
- return (BADARG);
- } else
- place = nargv[optind];
- #ifdef GNU_COMPATIBLE
- dash_prefix = W_PREFIX;
- #endif
- optchar = parse_long_options(nargv, options, long_options,
- idx, 0, flags);
- place = EMSG;
- return (optchar);
- }
- if (*++oli != ':') {
- if (!*place)
- ++optind;
- } else {
- optarg = NULL;
- if (*place)
- optarg = (char *)place;
- else if (oli[1] != ':') {
- if (++optind >= nargc) {
- place = EMSG;
- if (PRINT_ERROR)
- warnx(recargchar, optchar);
- optopt = optchar;
- return (BADARG);
- } else
- optarg = nargv[optind];
- }
- place = EMSG;
- ++optind;
- }
-
- return (optchar);
- }
- #ifdef REPLACE_GETOPT
- int
- getopt(int nargc, char * const *nargv, const char *options)
- {
-
- return (getopt_internal(nargc, nargv, options, NULL, NULL, 0));
- }
- #endif
- int
- getopt_long(int nargc, char * const *nargv, const char *options,
- const struct option *long_options, int *idx)
- {
- return (getopt_internal(nargc, nargv, options, long_options, idx,
- FLAG_PERMUTE));
- }
- int
- getopt_long_only(int nargc, char * const *nargv, const char *options,
- const struct option *long_options, int *idx)
- {
- return (getopt_internal(nargc, nargv, options, long_options, idx,
- FLAG_PERMUTE|FLAG_LONGONLY));
- }
|