xref: /aosp_15_r20/external/libdav1d/tools/compat/getopt.c (revision c09093415860a1c2373dacd84c4fde00c507cdfd)
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