1*c0909341SAndroid Build Coastguard Worker /* $OpenBSD: getopt_long.c,v 1.23 2007/10/31 12:34:57 chl Exp $ */
2*c0909341SAndroid Build Coastguard Worker /* $NetBSD: getopt_long.c,v 1.15 2002/01/31 22:43:40 tv Exp $ */
3*c0909341SAndroid Build Coastguard Worker
4*c0909341SAndroid Build Coastguard Worker /*
5*c0909341SAndroid Build Coastguard Worker * Copyright (c) 2002 Todd C. Miller <[email protected]>
6*c0909341SAndroid Build Coastguard Worker *
7*c0909341SAndroid Build Coastguard Worker * Permission to use, copy, modify, and distribute this software for any
8*c0909341SAndroid Build Coastguard Worker * purpose with or without fee is hereby granted, provided that the above
9*c0909341SAndroid Build Coastguard Worker * copyright notice and this permission notice appear in all copies.
10*c0909341SAndroid Build Coastguard Worker *
11*c0909341SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
12*c0909341SAndroid Build Coastguard Worker * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
13*c0909341SAndroid Build Coastguard Worker * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
14*c0909341SAndroid Build Coastguard Worker * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
15*c0909341SAndroid Build Coastguard Worker * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
16*c0909341SAndroid Build Coastguard Worker * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
17*c0909341SAndroid Build Coastguard Worker * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
18*c0909341SAndroid Build Coastguard Worker *
19*c0909341SAndroid Build Coastguard Worker * Sponsored in part by the Defense Advanced Research Projects
20*c0909341SAndroid Build Coastguard Worker * Agency (DARPA) and Air Force Research Laboratory, Air Force
21*c0909341SAndroid Build Coastguard Worker * Materiel Command, USAF, under agreement number F39502-99-1-0512.
22*c0909341SAndroid Build Coastguard Worker */
23*c0909341SAndroid Build Coastguard Worker /*-
24*c0909341SAndroid Build Coastguard Worker * Copyright (c) 2000 The NetBSD Foundation, Inc.
25*c0909341SAndroid Build Coastguard Worker * All rights reserved.
26*c0909341SAndroid Build Coastguard Worker *
27*c0909341SAndroid Build Coastguard Worker * This code is derived from software contributed to The NetBSD Foundation
28*c0909341SAndroid Build Coastguard Worker * by Dieter Baron and Thomas Klausner.
29*c0909341SAndroid Build Coastguard Worker *
30*c0909341SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
31*c0909341SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions
32*c0909341SAndroid Build Coastguard Worker * are met:
33*c0909341SAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright
34*c0909341SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer.
35*c0909341SAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright
36*c0909341SAndroid Build Coastguard Worker * notice, this list of conditions and the following disclaimer in the
37*c0909341SAndroid Build Coastguard Worker * documentation and/or other materials provided with the distribution.
38*c0909341SAndroid Build Coastguard Worker *
39*c0909341SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
40*c0909341SAndroid Build Coastguard Worker * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
41*c0909341SAndroid Build Coastguard Worker * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
42*c0909341SAndroid Build Coastguard Worker * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
43*c0909341SAndroid Build Coastguard Worker * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
44*c0909341SAndroid Build Coastguard Worker * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
45*c0909341SAndroid Build Coastguard Worker * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
46*c0909341SAndroid Build Coastguard Worker * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
47*c0909341SAndroid Build Coastguard Worker * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
48*c0909341SAndroid Build Coastguard Worker * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
49*c0909341SAndroid Build Coastguard Worker * POSSIBILITY OF SUCH DAMAGE.
50*c0909341SAndroid Build Coastguard Worker */
51*c0909341SAndroid Build Coastguard Worker
52*c0909341SAndroid Build Coastguard Worker #include <errno.h>
53*c0909341SAndroid Build Coastguard Worker #include <stdlib.h>
54*c0909341SAndroid Build Coastguard Worker #include <string.h>
55*c0909341SAndroid Build Coastguard Worker #include <getopt.h>
56*c0909341SAndroid Build Coastguard Worker #include <stdarg.h>
57*c0909341SAndroid Build Coastguard Worker #include <stdio.h>
58*c0909341SAndroid Build Coastguard Worker #ifdef _WIN32
59*c0909341SAndroid Build Coastguard Worker #include <windows.h>
60*c0909341SAndroid Build Coastguard Worker #else
61*c0909341SAndroid Build Coastguard Worker #include <err.h>
62*c0909341SAndroid Build Coastguard Worker #endif
63*c0909341SAndroid Build Coastguard Worker
64*c0909341SAndroid Build Coastguard Worker #define REPLACE_GETOPT /* use this getopt as the system getopt(3) */
65*c0909341SAndroid Build Coastguard Worker
66*c0909341SAndroid Build Coastguard Worker #ifdef REPLACE_GETOPT
67*c0909341SAndroid Build Coastguard Worker int opterr = 1; /* if error message should be printed */
68*c0909341SAndroid Build Coastguard Worker int optind = 1; /* index into parent argv vector */
69*c0909341SAndroid Build Coastguard Worker int optopt = '?'; /* character checked for validity */
70*c0909341SAndroid Build Coastguard Worker #undef optreset /* see getopt.h */
71*c0909341SAndroid Build Coastguard Worker #define optreset __mingw_optreset
72*c0909341SAndroid Build Coastguard Worker int optreset; /* reset getopt */
73*c0909341SAndroid Build Coastguard Worker char *optarg; /* argument associated with option */
74*c0909341SAndroid Build Coastguard Worker #endif
75*c0909341SAndroid Build Coastguard Worker
76*c0909341SAndroid Build Coastguard Worker #define PRINT_ERROR ((opterr) && (*options != ':'))
77*c0909341SAndroid Build Coastguard Worker
78*c0909341SAndroid Build Coastguard Worker #define FLAG_PERMUTE 0x01 /* permute non-options to the end of argv */
79*c0909341SAndroid Build Coastguard Worker #define FLAG_ALLARGS 0x02 /* treat non-options as args to option "-1" */
80*c0909341SAndroid Build Coastguard Worker #define FLAG_LONGONLY 0x04 /* operate as getopt_long_only */
81*c0909341SAndroid Build Coastguard Worker
82*c0909341SAndroid Build Coastguard Worker /* return values */
83*c0909341SAndroid Build Coastguard Worker #define BADCH (int)'?'
84*c0909341SAndroid Build Coastguard Worker #define BADARG ((*options == ':') ? (int)':' : (int)'?')
85*c0909341SAndroid Build Coastguard Worker #define INORDER (int)1
86*c0909341SAndroid Build Coastguard Worker
87*c0909341SAndroid Build Coastguard Worker #ifdef __CYGWIN__
88*c0909341SAndroid Build Coastguard Worker static char EMSG[] = "";
89*c0909341SAndroid Build Coastguard Worker #else
90*c0909341SAndroid Build Coastguard Worker #define EMSG ""
91*c0909341SAndroid Build Coastguard Worker #endif
92*c0909341SAndroid Build Coastguard Worker
93*c0909341SAndroid Build Coastguard Worker static int getopt_internal(int, char * const *, const char *,
94*c0909341SAndroid Build Coastguard Worker const struct option *, int *, int);
95*c0909341SAndroid Build Coastguard Worker static int parse_long_options(char * const *, const char *,
96*c0909341SAndroid Build Coastguard Worker const struct option *, int *, int);
97*c0909341SAndroid Build Coastguard Worker static int gcd(int, int);
98*c0909341SAndroid Build Coastguard Worker static void permute_args(int, int, int, char * const *);
99*c0909341SAndroid Build Coastguard Worker
100*c0909341SAndroid Build Coastguard Worker static char *place = EMSG; /* option letter processing */
101*c0909341SAndroid Build Coastguard Worker
102*c0909341SAndroid Build Coastguard Worker /* XXX: set optreset to 1 rather than these two */
103*c0909341SAndroid Build Coastguard Worker static int nonopt_start = -1; /* first non option argument (for permute) */
104*c0909341SAndroid Build Coastguard Worker static int nonopt_end = -1; /* first option after non options (for permute) */
105*c0909341SAndroid Build Coastguard Worker
106*c0909341SAndroid Build Coastguard Worker /* Error messages */
107*c0909341SAndroid Build Coastguard Worker static const char recargchar[] = "option requires an argument -- %c";
108*c0909341SAndroid Build Coastguard Worker static const char recargstring[] = "option requires an argument -- %s";
109*c0909341SAndroid Build Coastguard Worker static const char ambig[] = "ambiguous option -- %.*s";
110*c0909341SAndroid Build Coastguard Worker static const char noarg[] = "option doesn't take an argument -- %.*s";
111*c0909341SAndroid Build Coastguard Worker static const char illoptchar[] = "unknown option -- %c";
112*c0909341SAndroid Build Coastguard Worker static const char illoptstring[] = "unknown option -- %s";
113*c0909341SAndroid Build Coastguard Worker
114*c0909341SAndroid Build Coastguard Worker #ifdef _WIN32
115*c0909341SAndroid Build Coastguard Worker #ifndef __CYGWIN__
116*c0909341SAndroid Build Coastguard Worker #define __progname __argv[0]
117*c0909341SAndroid Build Coastguard Worker #else
118*c0909341SAndroid Build Coastguard Worker extern char __declspec(dllimport) *__progname;
119*c0909341SAndroid Build Coastguard Worker #endif
120*c0909341SAndroid Build Coastguard Worker
121*c0909341SAndroid Build Coastguard Worker static void
_vwarnx(const char * fmt,va_list ap)122*c0909341SAndroid Build Coastguard Worker _vwarnx(const char *fmt,va_list ap)
123*c0909341SAndroid Build Coastguard Worker {
124*c0909341SAndroid Build Coastguard Worker (void)fprintf(stderr,"%s: ",__progname);
125*c0909341SAndroid Build Coastguard Worker if (fmt != NULL)
126*c0909341SAndroid Build Coastguard Worker (void)vfprintf(stderr,fmt,ap);
127*c0909341SAndroid Build Coastguard Worker (void)fprintf(stderr,"\n");
128*c0909341SAndroid Build Coastguard Worker }
129*c0909341SAndroid Build Coastguard Worker
130*c0909341SAndroid Build Coastguard Worker static void
warnx(const char * fmt,...)131*c0909341SAndroid Build Coastguard Worker warnx(const char *fmt,...)
132*c0909341SAndroid Build Coastguard Worker {
133*c0909341SAndroid Build Coastguard Worker va_list ap;
134*c0909341SAndroid Build Coastguard Worker va_start(ap,fmt);
135*c0909341SAndroid Build Coastguard Worker _vwarnx(fmt,ap);
136*c0909341SAndroid Build Coastguard Worker va_end(ap);
137*c0909341SAndroid Build Coastguard Worker }
138*c0909341SAndroid Build Coastguard Worker #endif
139*c0909341SAndroid Build Coastguard Worker
140*c0909341SAndroid Build Coastguard Worker /*
141*c0909341SAndroid Build Coastguard Worker * Compute the greatest common divisor of a and b.
142*c0909341SAndroid Build Coastguard Worker */
143*c0909341SAndroid Build Coastguard Worker static int
gcd(int a,int b)144*c0909341SAndroid Build Coastguard Worker gcd(int a, int b)
145*c0909341SAndroid Build Coastguard Worker {
146*c0909341SAndroid Build Coastguard Worker int c;
147*c0909341SAndroid Build Coastguard Worker
148*c0909341SAndroid Build Coastguard Worker c = a % b;
149*c0909341SAndroid Build Coastguard Worker while (c != 0) {
150*c0909341SAndroid Build Coastguard Worker a = b;
151*c0909341SAndroid Build Coastguard Worker b = c;
152*c0909341SAndroid Build Coastguard Worker c = a % b;
153*c0909341SAndroid Build Coastguard Worker }
154*c0909341SAndroid Build Coastguard Worker
155*c0909341SAndroid Build Coastguard Worker return (b);
156*c0909341SAndroid Build Coastguard Worker }
157*c0909341SAndroid Build Coastguard Worker
158*c0909341SAndroid Build Coastguard Worker /*
159*c0909341SAndroid Build Coastguard Worker * Exchange the block from nonopt_start to nonopt_end with the block
160*c0909341SAndroid Build Coastguard Worker * from nonopt_end to opt_end (keeping the same order of arguments
161*c0909341SAndroid Build Coastguard Worker * in each block).
162*c0909341SAndroid Build Coastguard Worker */
163*c0909341SAndroid Build Coastguard Worker static void
permute_args(int panonopt_start,int panonopt_end,int opt_end,char * const * nargv)164*c0909341SAndroid Build Coastguard Worker permute_args(int panonopt_start, int panonopt_end, int opt_end,
165*c0909341SAndroid Build Coastguard Worker char * const *nargv)
166*c0909341SAndroid Build Coastguard Worker {
167*c0909341SAndroid Build Coastguard Worker int cstart, cyclelen, i, j, ncycle, nnonopts, nopts, pos;
168*c0909341SAndroid Build Coastguard Worker char *swap;
169*c0909341SAndroid Build Coastguard Worker
170*c0909341SAndroid Build Coastguard Worker /*
171*c0909341SAndroid Build Coastguard Worker * compute lengths of blocks and number and size of cycles
172*c0909341SAndroid Build Coastguard Worker */
173*c0909341SAndroid Build Coastguard Worker nnonopts = panonopt_end - panonopt_start;
174*c0909341SAndroid Build Coastguard Worker nopts = opt_end - panonopt_end;
175*c0909341SAndroid Build Coastguard Worker ncycle = gcd(nnonopts, nopts);
176*c0909341SAndroid Build Coastguard Worker cyclelen = (opt_end - panonopt_start) / ncycle;
177*c0909341SAndroid Build Coastguard Worker
178*c0909341SAndroid Build Coastguard Worker for (i = 0; i < ncycle; i++) {
179*c0909341SAndroid Build Coastguard Worker cstart = panonopt_end+i;
180*c0909341SAndroid Build Coastguard Worker pos = cstart;
181*c0909341SAndroid Build Coastguard Worker for (j = 0; j < cyclelen; j++) {
182*c0909341SAndroid Build Coastguard Worker if (pos >= panonopt_end)
183*c0909341SAndroid Build Coastguard Worker pos -= nnonopts;
184*c0909341SAndroid Build Coastguard Worker else
185*c0909341SAndroid Build Coastguard Worker pos += nopts;
186*c0909341SAndroid Build Coastguard Worker swap = nargv[pos];
187*c0909341SAndroid Build Coastguard Worker /* LINTED const cast */
188*c0909341SAndroid Build Coastguard Worker ((char **) nargv)[pos] = nargv[cstart];
189*c0909341SAndroid Build Coastguard Worker /* LINTED const cast */
190*c0909341SAndroid Build Coastguard Worker ((char **)nargv)[cstart] = swap;
191*c0909341SAndroid Build Coastguard Worker }
192*c0909341SAndroid Build Coastguard Worker }
193*c0909341SAndroid Build Coastguard Worker }
194*c0909341SAndroid Build Coastguard Worker
195*c0909341SAndroid Build Coastguard Worker /*
196*c0909341SAndroid Build Coastguard Worker * parse_long_options --
197*c0909341SAndroid Build Coastguard Worker * Parse long options in argc/argv argument vector.
198*c0909341SAndroid Build Coastguard Worker * Returns -1 if short_too is set and the option does not match long_options.
199*c0909341SAndroid Build Coastguard Worker */
200*c0909341SAndroid Build Coastguard Worker static int
parse_long_options(char * const * nargv,const char * options,const struct option * long_options,int * idx,int short_too)201*c0909341SAndroid Build Coastguard Worker parse_long_options(char * const *nargv, const char *options,
202*c0909341SAndroid Build Coastguard Worker const struct option *long_options, int *idx, int short_too)
203*c0909341SAndroid Build Coastguard Worker {
204*c0909341SAndroid Build Coastguard Worker char *current_argv, *has_equal;
205*c0909341SAndroid Build Coastguard Worker size_t current_argv_len;
206*c0909341SAndroid Build Coastguard Worker int i, ambiguous, match;
207*c0909341SAndroid Build Coastguard Worker
208*c0909341SAndroid Build Coastguard Worker #define IDENTICAL_INTERPRETATION(_x, _y) \
209*c0909341SAndroid Build Coastguard Worker (long_options[(_x)].has_arg == long_options[(_y)].has_arg && \
210*c0909341SAndroid Build Coastguard Worker long_options[(_x)].flag == long_options[(_y)].flag && \
211*c0909341SAndroid Build Coastguard Worker long_options[(_x)].val == long_options[(_y)].val)
212*c0909341SAndroid Build Coastguard Worker
213*c0909341SAndroid Build Coastguard Worker current_argv = place;
214*c0909341SAndroid Build Coastguard Worker match = -1;
215*c0909341SAndroid Build Coastguard Worker ambiguous = 0;
216*c0909341SAndroid Build Coastguard Worker
217*c0909341SAndroid Build Coastguard Worker optind++;
218*c0909341SAndroid Build Coastguard Worker
219*c0909341SAndroid Build Coastguard Worker if ((has_equal = strchr(current_argv, '=')) != NULL) {
220*c0909341SAndroid Build Coastguard Worker /* argument found (--option=arg) */
221*c0909341SAndroid Build Coastguard Worker current_argv_len = has_equal - current_argv;
222*c0909341SAndroid Build Coastguard Worker has_equal++;
223*c0909341SAndroid Build Coastguard Worker } else
224*c0909341SAndroid Build Coastguard Worker current_argv_len = strlen(current_argv);
225*c0909341SAndroid Build Coastguard Worker
226*c0909341SAndroid Build Coastguard Worker for (i = 0; long_options[i].name; i++) {
227*c0909341SAndroid Build Coastguard Worker /* find matching long option */
228*c0909341SAndroid Build Coastguard Worker if (strncmp(current_argv, long_options[i].name,
229*c0909341SAndroid Build Coastguard Worker current_argv_len))
230*c0909341SAndroid Build Coastguard Worker continue;
231*c0909341SAndroid Build Coastguard Worker
232*c0909341SAndroid Build Coastguard Worker if (strlen(long_options[i].name) == current_argv_len) {
233*c0909341SAndroid Build Coastguard Worker /* exact match */
234*c0909341SAndroid Build Coastguard Worker match = i;
235*c0909341SAndroid Build Coastguard Worker ambiguous = 0;
236*c0909341SAndroid Build Coastguard Worker break;
237*c0909341SAndroid Build Coastguard Worker }
238*c0909341SAndroid Build Coastguard Worker /*
239*c0909341SAndroid Build Coastguard Worker * If this is a known short option, don't allow
240*c0909341SAndroid Build Coastguard Worker * a partial match of a single character.
241*c0909341SAndroid Build Coastguard Worker */
242*c0909341SAndroid Build Coastguard Worker if (short_too && current_argv_len == 1)
243*c0909341SAndroid Build Coastguard Worker continue;
244*c0909341SAndroid Build Coastguard Worker
245*c0909341SAndroid Build Coastguard Worker if (match == -1) /* partial match */
246*c0909341SAndroid Build Coastguard Worker match = i;
247*c0909341SAndroid Build Coastguard Worker else if (!IDENTICAL_INTERPRETATION(i, match))
248*c0909341SAndroid Build Coastguard Worker ambiguous = 1;
249*c0909341SAndroid Build Coastguard Worker }
250*c0909341SAndroid Build Coastguard Worker if (ambiguous) {
251*c0909341SAndroid Build Coastguard Worker /* ambiguous abbreviation */
252*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
253*c0909341SAndroid Build Coastguard Worker warnx(ambig, (int)current_argv_len,
254*c0909341SAndroid Build Coastguard Worker current_argv);
255*c0909341SAndroid Build Coastguard Worker optopt = 0;
256*c0909341SAndroid Build Coastguard Worker return (BADCH);
257*c0909341SAndroid Build Coastguard Worker }
258*c0909341SAndroid Build Coastguard Worker if (match != -1) { /* option found */
259*c0909341SAndroid Build Coastguard Worker if (long_options[match].has_arg == no_argument
260*c0909341SAndroid Build Coastguard Worker && has_equal) {
261*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
262*c0909341SAndroid Build Coastguard Worker warnx(noarg, (int)current_argv_len,
263*c0909341SAndroid Build Coastguard Worker current_argv);
264*c0909341SAndroid Build Coastguard Worker /*
265*c0909341SAndroid Build Coastguard Worker * XXX: GNU sets optopt to val regardless of flag
266*c0909341SAndroid Build Coastguard Worker */
267*c0909341SAndroid Build Coastguard Worker if (long_options[match].flag == NULL)
268*c0909341SAndroid Build Coastguard Worker optopt = long_options[match].val;
269*c0909341SAndroid Build Coastguard Worker else
270*c0909341SAndroid Build Coastguard Worker optopt = 0;
271*c0909341SAndroid Build Coastguard Worker return (BADARG);
272*c0909341SAndroid Build Coastguard Worker }
273*c0909341SAndroid Build Coastguard Worker if (long_options[match].has_arg == required_argument ||
274*c0909341SAndroid Build Coastguard Worker long_options[match].has_arg == optional_argument) {
275*c0909341SAndroid Build Coastguard Worker if (has_equal)
276*c0909341SAndroid Build Coastguard Worker optarg = has_equal;
277*c0909341SAndroid Build Coastguard Worker else if (long_options[match].has_arg ==
278*c0909341SAndroid Build Coastguard Worker required_argument) {
279*c0909341SAndroid Build Coastguard Worker /*
280*c0909341SAndroid Build Coastguard Worker * optional argument doesn't use next nargv
281*c0909341SAndroid Build Coastguard Worker */
282*c0909341SAndroid Build Coastguard Worker optarg = nargv[optind++];
283*c0909341SAndroid Build Coastguard Worker }
284*c0909341SAndroid Build Coastguard Worker }
285*c0909341SAndroid Build Coastguard Worker if ((long_options[match].has_arg == required_argument)
286*c0909341SAndroid Build Coastguard Worker && (optarg == NULL)) {
287*c0909341SAndroid Build Coastguard Worker /*
288*c0909341SAndroid Build Coastguard Worker * Missing argument; leading ':' indicates no error
289*c0909341SAndroid Build Coastguard Worker * should be generated.
290*c0909341SAndroid Build Coastguard Worker */
291*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
292*c0909341SAndroid Build Coastguard Worker warnx(recargstring,
293*c0909341SAndroid Build Coastguard Worker current_argv);
294*c0909341SAndroid Build Coastguard Worker /*
295*c0909341SAndroid Build Coastguard Worker * XXX: GNU sets optopt to val regardless of flag
296*c0909341SAndroid Build Coastguard Worker */
297*c0909341SAndroid Build Coastguard Worker if (long_options[match].flag == NULL)
298*c0909341SAndroid Build Coastguard Worker optopt = long_options[match].val;
299*c0909341SAndroid Build Coastguard Worker else
300*c0909341SAndroid Build Coastguard Worker optopt = 0;
301*c0909341SAndroid Build Coastguard Worker --optind;
302*c0909341SAndroid Build Coastguard Worker return (BADARG);
303*c0909341SAndroid Build Coastguard Worker }
304*c0909341SAndroid Build Coastguard Worker } else { /* unknown option */
305*c0909341SAndroid Build Coastguard Worker if (short_too) {
306*c0909341SAndroid Build Coastguard Worker --optind;
307*c0909341SAndroid Build Coastguard Worker return (-1);
308*c0909341SAndroid Build Coastguard Worker }
309*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
310*c0909341SAndroid Build Coastguard Worker warnx(illoptstring, current_argv);
311*c0909341SAndroid Build Coastguard Worker optopt = 0;
312*c0909341SAndroid Build Coastguard Worker return (BADCH);
313*c0909341SAndroid Build Coastguard Worker }
314*c0909341SAndroid Build Coastguard Worker if (idx)
315*c0909341SAndroid Build Coastguard Worker *idx = match;
316*c0909341SAndroid Build Coastguard Worker if (long_options[match].flag) {
317*c0909341SAndroid Build Coastguard Worker *long_options[match].flag = long_options[match].val;
318*c0909341SAndroid Build Coastguard Worker return (0);
319*c0909341SAndroid Build Coastguard Worker } else
320*c0909341SAndroid Build Coastguard Worker return (long_options[match].val);
321*c0909341SAndroid Build Coastguard Worker #undef IDENTICAL_INTERPRETATION
322*c0909341SAndroid Build Coastguard Worker }
323*c0909341SAndroid Build Coastguard Worker
324*c0909341SAndroid Build Coastguard Worker /*
325*c0909341SAndroid Build Coastguard Worker * getopt_internal --
326*c0909341SAndroid Build Coastguard Worker * Parse argc/argv argument vector. Called by user level routines.
327*c0909341SAndroid Build Coastguard Worker */
328*c0909341SAndroid Build Coastguard Worker static int
getopt_internal(int nargc,char * const * nargv,const char * options,const struct option * long_options,int * idx,int flags)329*c0909341SAndroid Build Coastguard Worker getopt_internal(int nargc, char * const *nargv, const char *options,
330*c0909341SAndroid Build Coastguard Worker const struct option *long_options, int *idx, int flags)
331*c0909341SAndroid Build Coastguard Worker {
332*c0909341SAndroid Build Coastguard Worker char *oli; /* option letter list index */
333*c0909341SAndroid Build Coastguard Worker int optchar, short_too;
334*c0909341SAndroid Build Coastguard Worker static int posixly_correct = -1;
335*c0909341SAndroid Build Coastguard Worker
336*c0909341SAndroid Build Coastguard Worker if (options == NULL)
337*c0909341SAndroid Build Coastguard Worker return (-1);
338*c0909341SAndroid Build Coastguard Worker
339*c0909341SAndroid Build Coastguard Worker /*
340*c0909341SAndroid Build Coastguard Worker * XXX Some GNU programs (like cvs) set optind to 0 instead of
341*c0909341SAndroid Build Coastguard Worker * XXX using optreset. Work around this braindamage.
342*c0909341SAndroid Build Coastguard Worker */
343*c0909341SAndroid Build Coastguard Worker if (optind == 0)
344*c0909341SAndroid Build Coastguard Worker optind = optreset = 1;
345*c0909341SAndroid Build Coastguard Worker
346*c0909341SAndroid Build Coastguard Worker /*
347*c0909341SAndroid Build Coastguard Worker * Disable GNU extensions if POSIXLY_CORRECT is set or options
348*c0909341SAndroid Build Coastguard Worker * string begins with a '+'.
349*c0909341SAndroid Build Coastguard Worker *
350*c0909341SAndroid Build Coastguard Worker * CV, 2009-12-14: Check POSIXLY_CORRECT anew if optind == 0 or
351*c0909341SAndroid Build Coastguard Worker * optreset != 0 for GNU compatibility.
352*c0909341SAndroid Build Coastguard Worker */
353*c0909341SAndroid Build Coastguard Worker if (posixly_correct == -1 || optreset != 0)
354*c0909341SAndroid Build Coastguard Worker posixly_correct = (getenv("POSIXLY_CORRECT") != NULL);
355*c0909341SAndroid Build Coastguard Worker if (*options == '-')
356*c0909341SAndroid Build Coastguard Worker flags |= FLAG_ALLARGS;
357*c0909341SAndroid Build Coastguard Worker else if (posixly_correct || *options == '+')
358*c0909341SAndroid Build Coastguard Worker flags &= ~FLAG_PERMUTE;
359*c0909341SAndroid Build Coastguard Worker if (*options == '+' || *options == '-')
360*c0909341SAndroid Build Coastguard Worker options++;
361*c0909341SAndroid Build Coastguard Worker
362*c0909341SAndroid Build Coastguard Worker optarg = NULL;
363*c0909341SAndroid Build Coastguard Worker if (optreset)
364*c0909341SAndroid Build Coastguard Worker nonopt_start = nonopt_end = -1;
365*c0909341SAndroid Build Coastguard Worker start:
366*c0909341SAndroid Build Coastguard Worker if (optreset || !*place) { /* update scanning pointer */
367*c0909341SAndroid Build Coastguard Worker optreset = 0;
368*c0909341SAndroid Build Coastguard Worker if (optind >= nargc) { /* end of argument vector */
369*c0909341SAndroid Build Coastguard Worker place = EMSG;
370*c0909341SAndroid Build Coastguard Worker if (nonopt_end != -1) {
371*c0909341SAndroid Build Coastguard Worker /* do permutation, if we have to */
372*c0909341SAndroid Build Coastguard Worker permute_args(nonopt_start, nonopt_end,
373*c0909341SAndroid Build Coastguard Worker optind, nargv);
374*c0909341SAndroid Build Coastguard Worker optind -= nonopt_end - nonopt_start;
375*c0909341SAndroid Build Coastguard Worker }
376*c0909341SAndroid Build Coastguard Worker else if (nonopt_start != -1) {
377*c0909341SAndroid Build Coastguard Worker /*
378*c0909341SAndroid Build Coastguard Worker * If we skipped non-options, set optind
379*c0909341SAndroid Build Coastguard Worker * to the first of them.
380*c0909341SAndroid Build Coastguard Worker */
381*c0909341SAndroid Build Coastguard Worker optind = nonopt_start;
382*c0909341SAndroid Build Coastguard Worker }
383*c0909341SAndroid Build Coastguard Worker nonopt_start = nonopt_end = -1;
384*c0909341SAndroid Build Coastguard Worker return (-1);
385*c0909341SAndroid Build Coastguard Worker }
386*c0909341SAndroid Build Coastguard Worker if (*(place = nargv[optind]) != '-' ||
387*c0909341SAndroid Build Coastguard Worker (place[1] == '\0' && strchr(options, '-') == NULL)) {
388*c0909341SAndroid Build Coastguard Worker place = EMSG; /* found non-option */
389*c0909341SAndroid Build Coastguard Worker if (flags & FLAG_ALLARGS) {
390*c0909341SAndroid Build Coastguard Worker /*
391*c0909341SAndroid Build Coastguard Worker * GNU extension:
392*c0909341SAndroid Build Coastguard Worker * return non-option as argument to option 1
393*c0909341SAndroid Build Coastguard Worker */
394*c0909341SAndroid Build Coastguard Worker optarg = nargv[optind++];
395*c0909341SAndroid Build Coastguard Worker return (INORDER);
396*c0909341SAndroid Build Coastguard Worker }
397*c0909341SAndroid Build Coastguard Worker if (!(flags & FLAG_PERMUTE)) {
398*c0909341SAndroid Build Coastguard Worker /*
399*c0909341SAndroid Build Coastguard Worker * If no permutation wanted, stop parsing
400*c0909341SAndroid Build Coastguard Worker * at first non-option.
401*c0909341SAndroid Build Coastguard Worker */
402*c0909341SAndroid Build Coastguard Worker return (-1);
403*c0909341SAndroid Build Coastguard Worker }
404*c0909341SAndroid Build Coastguard Worker /* do permutation */
405*c0909341SAndroid Build Coastguard Worker if (nonopt_start == -1)
406*c0909341SAndroid Build Coastguard Worker nonopt_start = optind;
407*c0909341SAndroid Build Coastguard Worker else if (nonopt_end != -1) {
408*c0909341SAndroid Build Coastguard Worker permute_args(nonopt_start, nonopt_end,
409*c0909341SAndroid Build Coastguard Worker optind, nargv);
410*c0909341SAndroid Build Coastguard Worker nonopt_start = optind -
411*c0909341SAndroid Build Coastguard Worker (nonopt_end - nonopt_start);
412*c0909341SAndroid Build Coastguard Worker nonopt_end = -1;
413*c0909341SAndroid Build Coastguard Worker }
414*c0909341SAndroid Build Coastguard Worker optind++;
415*c0909341SAndroid Build Coastguard Worker /* process next argument */
416*c0909341SAndroid Build Coastguard Worker goto start;
417*c0909341SAndroid Build Coastguard Worker }
418*c0909341SAndroid Build Coastguard Worker if (nonopt_start != -1 && nonopt_end == -1)
419*c0909341SAndroid Build Coastguard Worker nonopt_end = optind;
420*c0909341SAndroid Build Coastguard Worker
421*c0909341SAndroid Build Coastguard Worker /*
422*c0909341SAndroid Build Coastguard Worker * If we have "-" do nothing, if "--" we are done.
423*c0909341SAndroid Build Coastguard Worker */
424*c0909341SAndroid Build Coastguard Worker if (place[1] != '\0' && *++place == '-' && place[1] == '\0') {
425*c0909341SAndroid Build Coastguard Worker optind++;
426*c0909341SAndroid Build Coastguard Worker place = EMSG;
427*c0909341SAndroid Build Coastguard Worker /*
428*c0909341SAndroid Build Coastguard Worker * We found an option (--), so if we skipped
429*c0909341SAndroid Build Coastguard Worker * non-options, we have to permute.
430*c0909341SAndroid Build Coastguard Worker */
431*c0909341SAndroid Build Coastguard Worker if (nonopt_end != -1) {
432*c0909341SAndroid Build Coastguard Worker permute_args(nonopt_start, nonopt_end,
433*c0909341SAndroid Build Coastguard Worker optind, nargv);
434*c0909341SAndroid Build Coastguard Worker optind -= nonopt_end - nonopt_start;
435*c0909341SAndroid Build Coastguard Worker }
436*c0909341SAndroid Build Coastguard Worker nonopt_start = nonopt_end = -1;
437*c0909341SAndroid Build Coastguard Worker return (-1);
438*c0909341SAndroid Build Coastguard Worker }
439*c0909341SAndroid Build Coastguard Worker }
440*c0909341SAndroid Build Coastguard Worker
441*c0909341SAndroid Build Coastguard Worker /*
442*c0909341SAndroid Build Coastguard Worker * Check long options if:
443*c0909341SAndroid Build Coastguard Worker * 1) we were passed some
444*c0909341SAndroid Build Coastguard Worker * 2) the arg is not just "-"
445*c0909341SAndroid Build Coastguard Worker * 3) either the arg starts with -- we are getopt_long_only()
446*c0909341SAndroid Build Coastguard Worker */
447*c0909341SAndroid Build Coastguard Worker if (long_options != NULL && place != nargv[optind] &&
448*c0909341SAndroid Build Coastguard Worker (*place == '-' || (flags & FLAG_LONGONLY))) {
449*c0909341SAndroid Build Coastguard Worker short_too = 0;
450*c0909341SAndroid Build Coastguard Worker if (*place == '-')
451*c0909341SAndroid Build Coastguard Worker place++; /* --foo long option */
452*c0909341SAndroid Build Coastguard Worker else if (*place != ':' && strchr(options, *place) != NULL)
453*c0909341SAndroid Build Coastguard Worker short_too = 1; /* could be short option too */
454*c0909341SAndroid Build Coastguard Worker
455*c0909341SAndroid Build Coastguard Worker optchar = parse_long_options(nargv, options, long_options,
456*c0909341SAndroid Build Coastguard Worker idx, short_too);
457*c0909341SAndroid Build Coastguard Worker if (optchar != -1) {
458*c0909341SAndroid Build Coastguard Worker place = EMSG;
459*c0909341SAndroid Build Coastguard Worker return (optchar);
460*c0909341SAndroid Build Coastguard Worker }
461*c0909341SAndroid Build Coastguard Worker }
462*c0909341SAndroid Build Coastguard Worker
463*c0909341SAndroid Build Coastguard Worker if ((optchar = (int)*place++) == (int)':' ||
464*c0909341SAndroid Build Coastguard Worker (optchar == (int)'-' && *place != '\0') ||
465*c0909341SAndroid Build Coastguard Worker (oli = strchr(options, optchar)) == NULL) {
466*c0909341SAndroid Build Coastguard Worker /*
467*c0909341SAndroid Build Coastguard Worker * If the user specified "-" and '-' isn't listed in
468*c0909341SAndroid Build Coastguard Worker * options, return -1 (non-option) as per POSIX.
469*c0909341SAndroid Build Coastguard Worker * Otherwise, it is an unknown option character (or ':').
470*c0909341SAndroid Build Coastguard Worker */
471*c0909341SAndroid Build Coastguard Worker if (optchar == (int)'-' && *place == '\0')
472*c0909341SAndroid Build Coastguard Worker return (-1);
473*c0909341SAndroid Build Coastguard Worker if (!*place)
474*c0909341SAndroid Build Coastguard Worker ++optind;
475*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
476*c0909341SAndroid Build Coastguard Worker warnx(illoptchar, optchar);
477*c0909341SAndroid Build Coastguard Worker optopt = optchar;
478*c0909341SAndroid Build Coastguard Worker return (BADCH);
479*c0909341SAndroid Build Coastguard Worker }
480*c0909341SAndroid Build Coastguard Worker if (long_options != NULL && optchar == 'W' && oli[1] == ';') {
481*c0909341SAndroid Build Coastguard Worker /* -W long-option */
482*c0909341SAndroid Build Coastguard Worker if (*place) /* no space */
483*c0909341SAndroid Build Coastguard Worker /* NOTHING */;
484*c0909341SAndroid Build Coastguard Worker else if (++optind >= nargc) { /* no arg */
485*c0909341SAndroid Build Coastguard Worker place = EMSG;
486*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
487*c0909341SAndroid Build Coastguard Worker warnx(recargchar, optchar);
488*c0909341SAndroid Build Coastguard Worker optopt = optchar;
489*c0909341SAndroid Build Coastguard Worker return (BADARG);
490*c0909341SAndroid Build Coastguard Worker } else /* white space */
491*c0909341SAndroid Build Coastguard Worker place = nargv[optind];
492*c0909341SAndroid Build Coastguard Worker optchar = parse_long_options(nargv, options, long_options,
493*c0909341SAndroid Build Coastguard Worker idx, 0);
494*c0909341SAndroid Build Coastguard Worker place = EMSG;
495*c0909341SAndroid Build Coastguard Worker return (optchar);
496*c0909341SAndroid Build Coastguard Worker }
497*c0909341SAndroid Build Coastguard Worker if (*++oli != ':') { /* doesn't take argument */
498*c0909341SAndroid Build Coastguard Worker if (!*place)
499*c0909341SAndroid Build Coastguard Worker ++optind;
500*c0909341SAndroid Build Coastguard Worker } else { /* takes (optional) argument */
501*c0909341SAndroid Build Coastguard Worker optarg = NULL;
502*c0909341SAndroid Build Coastguard Worker if (*place) /* no white space */
503*c0909341SAndroid Build Coastguard Worker optarg = place;
504*c0909341SAndroid Build Coastguard Worker else if (oli[1] != ':') { /* arg not optional */
505*c0909341SAndroid Build Coastguard Worker if (++optind >= nargc) { /* no arg */
506*c0909341SAndroid Build Coastguard Worker place = EMSG;
507*c0909341SAndroid Build Coastguard Worker if (PRINT_ERROR)
508*c0909341SAndroid Build Coastguard Worker warnx(recargchar, optchar);
509*c0909341SAndroid Build Coastguard Worker optopt = optchar;
510*c0909341SAndroid Build Coastguard Worker return (BADARG);
511*c0909341SAndroid Build Coastguard Worker } else
512*c0909341SAndroid Build Coastguard Worker optarg = nargv[optind];
513*c0909341SAndroid Build Coastguard Worker }
514*c0909341SAndroid Build Coastguard Worker place = EMSG;
515*c0909341SAndroid Build Coastguard Worker ++optind;
516*c0909341SAndroid Build Coastguard Worker }
517*c0909341SAndroid Build Coastguard Worker /* dump back option letter */
518*c0909341SAndroid Build Coastguard Worker return (optchar);
519*c0909341SAndroid Build Coastguard Worker }
520*c0909341SAndroid Build Coastguard Worker
521*c0909341SAndroid Build Coastguard Worker #ifdef REPLACE_GETOPT
522*c0909341SAndroid Build Coastguard Worker /*
523*c0909341SAndroid Build Coastguard Worker * getopt --
524*c0909341SAndroid Build Coastguard Worker * Parse argc/argv argument vector.
525*c0909341SAndroid Build Coastguard Worker *
526*c0909341SAndroid Build Coastguard Worker * [eventually this will replace the BSD getopt]
527*c0909341SAndroid Build Coastguard Worker */
528*c0909341SAndroid Build Coastguard Worker int
getopt(int nargc,char * const * nargv,const char * options)529*c0909341SAndroid Build Coastguard Worker getopt(int nargc, char * const *nargv, const char *options)
530*c0909341SAndroid Build Coastguard Worker {
531*c0909341SAndroid Build Coastguard Worker
532*c0909341SAndroid Build Coastguard Worker /*
533*c0909341SAndroid Build Coastguard Worker * We don't pass FLAG_PERMUTE to getopt_internal() since
534*c0909341SAndroid Build Coastguard Worker * the BSD getopt(3) (unlike GNU) has never done this.
535*c0909341SAndroid Build Coastguard Worker *
536*c0909341SAndroid Build Coastguard Worker * Furthermore, since many privileged programs call getopt()
537*c0909341SAndroid Build Coastguard Worker * before dropping privileges it makes sense to keep things
538*c0909341SAndroid Build Coastguard Worker * as simple (and bug-free) as possible.
539*c0909341SAndroid Build Coastguard Worker */
540*c0909341SAndroid Build Coastguard Worker return (getopt_internal(nargc, nargv, options, NULL, NULL, 0));
541*c0909341SAndroid Build Coastguard Worker }
542*c0909341SAndroid Build Coastguard Worker #endif /* REPLACE_GETOPT */
543*c0909341SAndroid Build Coastguard Worker
544*c0909341SAndroid Build Coastguard Worker /*
545*c0909341SAndroid Build Coastguard Worker * getopt_long --
546*c0909341SAndroid Build Coastguard Worker * Parse argc/argv argument vector.
547*c0909341SAndroid Build Coastguard Worker */
548*c0909341SAndroid Build Coastguard Worker int
getopt_long(int nargc,char * const * nargv,const char * options,const struct option * long_options,int * idx)549*c0909341SAndroid Build Coastguard Worker getopt_long(int nargc, char * const *nargv, const char *options,
550*c0909341SAndroid Build Coastguard Worker const struct option *long_options, int *idx)
551*c0909341SAndroid Build Coastguard Worker {
552*c0909341SAndroid Build Coastguard Worker
553*c0909341SAndroid Build Coastguard Worker return (getopt_internal(nargc, nargv, options, long_options, idx,
554*c0909341SAndroid Build Coastguard Worker FLAG_PERMUTE));
555*c0909341SAndroid Build Coastguard Worker }
556*c0909341SAndroid Build Coastguard Worker
557*c0909341SAndroid Build Coastguard Worker /*
558*c0909341SAndroid Build Coastguard Worker * getopt_long_only --
559*c0909341SAndroid Build Coastguard Worker * Parse argc/argv argument vector.
560*c0909341SAndroid Build Coastguard Worker */
561*c0909341SAndroid Build Coastguard Worker int
getopt_long_only(int nargc,char * const * nargv,const char * options,const struct option * long_options,int * idx)562*c0909341SAndroid Build Coastguard Worker getopt_long_only(int nargc, char * const *nargv, const char *options,
563*c0909341SAndroid Build Coastguard Worker const struct option *long_options, int *idx)
564*c0909341SAndroid Build Coastguard Worker {
565*c0909341SAndroid Build Coastguard Worker
566*c0909341SAndroid Build Coastguard Worker return (getopt_internal(nargc, nargv, options, long_options, idx,
567*c0909341SAndroid Build Coastguard Worker FLAG_PERMUTE|FLAG_LONGONLY));
568*c0909341SAndroid Build Coastguard Worker }
569