xref: /aosp_15_r20/external/bc/src/opt.c (revision 5a6e848804d15c18a0125914844ee4eb0bda4fcf)
1*5a6e8488SAndroid Build Coastguard Worker /*
2*5a6e8488SAndroid Build Coastguard Worker  * *****************************************************************************
3*5a6e8488SAndroid Build Coastguard Worker  *
4*5a6e8488SAndroid Build Coastguard Worker  * SPDX-License-Identifier: BSD-2-Clause
5*5a6e8488SAndroid Build Coastguard Worker  *
6*5a6e8488SAndroid Build Coastguard Worker  * Copyright (c) 2018-2024 Gavin D. Howard and contributors.
7*5a6e8488SAndroid Build Coastguard Worker  *
8*5a6e8488SAndroid Build Coastguard Worker  * Redistribution and use in source and binary forms, with or without
9*5a6e8488SAndroid Build Coastguard Worker  * modification, are permitted provided that the following conditions are met:
10*5a6e8488SAndroid Build Coastguard Worker  *
11*5a6e8488SAndroid Build Coastguard Worker  * * Redistributions of source code must retain the above copyright notice, this
12*5a6e8488SAndroid Build Coastguard Worker  *   list of conditions and the following disclaimer.
13*5a6e8488SAndroid Build Coastguard Worker  *
14*5a6e8488SAndroid Build Coastguard Worker  * * Redistributions in binary form must reproduce the above copyright notice,
15*5a6e8488SAndroid Build Coastguard Worker  *   this list of conditions and the following disclaimer in the documentation
16*5a6e8488SAndroid Build Coastguard Worker  *   and/or other materials provided with the distribution.
17*5a6e8488SAndroid Build Coastguard Worker  *
18*5a6e8488SAndroid Build Coastguard Worker  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
19*5a6e8488SAndroid Build Coastguard Worker  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20*5a6e8488SAndroid Build Coastguard Worker  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21*5a6e8488SAndroid Build Coastguard Worker  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
22*5a6e8488SAndroid Build Coastguard Worker  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
23*5a6e8488SAndroid Build Coastguard Worker  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
24*5a6e8488SAndroid Build Coastguard Worker  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
25*5a6e8488SAndroid Build Coastguard Worker  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
26*5a6e8488SAndroid Build Coastguard Worker  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
27*5a6e8488SAndroid Build Coastguard Worker  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28*5a6e8488SAndroid Build Coastguard Worker  * POSSIBILITY OF SUCH DAMAGE.
29*5a6e8488SAndroid Build Coastguard Worker  *
30*5a6e8488SAndroid Build Coastguard Worker  * *****************************************************************************
31*5a6e8488SAndroid Build Coastguard Worker  *
32*5a6e8488SAndroid Build Coastguard Worker  * Adapted from https://github.com/skeeto/optparse
33*5a6e8488SAndroid Build Coastguard Worker  *
34*5a6e8488SAndroid Build Coastguard Worker  * *****************************************************************************
35*5a6e8488SAndroid Build Coastguard Worker  *
36*5a6e8488SAndroid Build Coastguard Worker  * Code for getopt_long() replacement. It turns out that getopt_long() has
37*5a6e8488SAndroid Build Coastguard Worker  * different behavior on different platforms.
38*5a6e8488SAndroid Build Coastguard Worker  *
39*5a6e8488SAndroid Build Coastguard Worker  */
40*5a6e8488SAndroid Build Coastguard Worker 
41*5a6e8488SAndroid Build Coastguard Worker #include <assert.h>
42*5a6e8488SAndroid Build Coastguard Worker #include <stdbool.h>
43*5a6e8488SAndroid Build Coastguard Worker #include <stdlib.h>
44*5a6e8488SAndroid Build Coastguard Worker #include <string.h>
45*5a6e8488SAndroid Build Coastguard Worker 
46*5a6e8488SAndroid Build Coastguard Worker #include <status.h>
47*5a6e8488SAndroid Build Coastguard Worker #include <opt.h>
48*5a6e8488SAndroid Build Coastguard Worker #include <vm.h>
49*5a6e8488SAndroid Build Coastguard Worker 
50*5a6e8488SAndroid Build Coastguard Worker /**
51*5a6e8488SAndroid Build Coastguard Worker  * Returns true if index @a i is the end of the longopts array.
52*5a6e8488SAndroid Build Coastguard Worker  * @param longopts  The long options array.
53*5a6e8488SAndroid Build Coastguard Worker  * @param i         The index to test.
54*5a6e8488SAndroid Build Coastguard Worker  * @return          True if @a i is the last index, false otherwise.
55*5a6e8488SAndroid Build Coastguard Worker  */
56*5a6e8488SAndroid Build Coastguard Worker static inline bool
bc_opt_longoptsEnd(const BcOptLong * longopts,size_t i)57*5a6e8488SAndroid Build Coastguard Worker bc_opt_longoptsEnd(const BcOptLong* longopts, size_t i)
58*5a6e8488SAndroid Build Coastguard Worker {
59*5a6e8488SAndroid Build Coastguard Worker 	return !longopts[i].name && !longopts[i].val;
60*5a6e8488SAndroid Build Coastguard Worker }
61*5a6e8488SAndroid Build Coastguard Worker 
62*5a6e8488SAndroid Build Coastguard Worker /**
63*5a6e8488SAndroid Build Coastguard Worker  * Returns the name of the long option that matches the character @a c.
64*5a6e8488SAndroid Build Coastguard Worker  * @param longopts  The long options array.
65*5a6e8488SAndroid Build Coastguard Worker  * @param c         The character to match against.
66*5a6e8488SAndroid Build Coastguard Worker  * @return          The name of the long option that matches @a c, or "NULL".
67*5a6e8488SAndroid Build Coastguard Worker  */
68*5a6e8488SAndroid Build Coastguard Worker static const char*
bc_opt_longopt(const BcOptLong * longopts,int c)69*5a6e8488SAndroid Build Coastguard Worker bc_opt_longopt(const BcOptLong* longopts, int c)
70*5a6e8488SAndroid Build Coastguard Worker {
71*5a6e8488SAndroid Build Coastguard Worker 	size_t i;
72*5a6e8488SAndroid Build Coastguard Worker 
73*5a6e8488SAndroid Build Coastguard Worker 	for (i = 0; !bc_opt_longoptsEnd(longopts, i); ++i)
74*5a6e8488SAndroid Build Coastguard Worker 	{
75*5a6e8488SAndroid Build Coastguard Worker 		if (longopts[i].val == c) return longopts[i].name;
76*5a6e8488SAndroid Build Coastguard Worker 	}
77*5a6e8488SAndroid Build Coastguard Worker 
78*5a6e8488SAndroid Build Coastguard Worker 	BC_UNREACHABLE
79*5a6e8488SAndroid Build Coastguard Worker 
80*5a6e8488SAndroid Build Coastguard Worker #if !BC_CLANG
81*5a6e8488SAndroid Build Coastguard Worker 	return "NULL";
82*5a6e8488SAndroid Build Coastguard Worker #endif // !BC_CLANG
83*5a6e8488SAndroid Build Coastguard Worker }
84*5a6e8488SAndroid Build Coastguard Worker 
85*5a6e8488SAndroid Build Coastguard Worker /**
86*5a6e8488SAndroid Build Coastguard Worker  * Issues a fatal error for an option parsing failure.
87*5a6e8488SAndroid Build Coastguard Worker  * @param err        The error.
88*5a6e8488SAndroid Build Coastguard Worker  * @param c          The character for the failing option.
89*5a6e8488SAndroid Build Coastguard Worker  * @param str        Either the string for the failing option, or the invalid
90*5a6e8488SAndroid Build Coastguard Worker  *                   option.
91*5a6e8488SAndroid Build Coastguard Worker  * @param use_short  True if the short option should be used for error printing,
92*5a6e8488SAndroid Build Coastguard Worker  *                   false otherwise.
93*5a6e8488SAndroid Build Coastguard Worker  */
94*5a6e8488SAndroid Build Coastguard Worker static void
bc_opt_error(BcErr err,int c,const char * str,bool use_short)95*5a6e8488SAndroid Build Coastguard Worker bc_opt_error(BcErr err, int c, const char* str, bool use_short)
96*5a6e8488SAndroid Build Coastguard Worker {
97*5a6e8488SAndroid Build Coastguard Worker 	if (err == BC_ERR_FATAL_OPTION)
98*5a6e8488SAndroid Build Coastguard Worker 	{
99*5a6e8488SAndroid Build Coastguard Worker 		if (use_short)
100*5a6e8488SAndroid Build Coastguard Worker 		{
101*5a6e8488SAndroid Build Coastguard Worker 			char short_str[2];
102*5a6e8488SAndroid Build Coastguard Worker 
103*5a6e8488SAndroid Build Coastguard Worker 			short_str[0] = (char) c;
104*5a6e8488SAndroid Build Coastguard Worker 			short_str[1] = '\0';
105*5a6e8488SAndroid Build Coastguard Worker 
106*5a6e8488SAndroid Build Coastguard Worker 			bc_error(err, 0, short_str);
107*5a6e8488SAndroid Build Coastguard Worker 		}
108*5a6e8488SAndroid Build Coastguard Worker 		else bc_error(err, 0, str);
109*5a6e8488SAndroid Build Coastguard Worker 	}
110*5a6e8488SAndroid Build Coastguard Worker 	else bc_error(err, 0, (int) c, str);
111*5a6e8488SAndroid Build Coastguard Worker }
112*5a6e8488SAndroid Build Coastguard Worker 
113*5a6e8488SAndroid Build Coastguard Worker /**
114*5a6e8488SAndroid Build Coastguard Worker  * Returns the type of the long option that matches @a c.
115*5a6e8488SAndroid Build Coastguard Worker  * @param longopts  The long options array.
116*5a6e8488SAndroid Build Coastguard Worker  * @param c         The character to match against.
117*5a6e8488SAndroid Build Coastguard Worker  * @return          The type of the long option as an integer, or -1 if none.
118*5a6e8488SAndroid Build Coastguard Worker  */
119*5a6e8488SAndroid Build Coastguard Worker static int
bc_opt_type(const BcOptLong * longopts,char c)120*5a6e8488SAndroid Build Coastguard Worker bc_opt_type(const BcOptLong* longopts, char c)
121*5a6e8488SAndroid Build Coastguard Worker {
122*5a6e8488SAndroid Build Coastguard Worker 	size_t i;
123*5a6e8488SAndroid Build Coastguard Worker 
124*5a6e8488SAndroid Build Coastguard Worker 	if (c == ':') return -1;
125*5a6e8488SAndroid Build Coastguard Worker 
126*5a6e8488SAndroid Build Coastguard Worker 	for (i = 0; !bc_opt_longoptsEnd(longopts, i) && longopts[i].val != c; ++i)
127*5a6e8488SAndroid Build Coastguard Worker 	{
128*5a6e8488SAndroid Build Coastguard Worker 		continue;
129*5a6e8488SAndroid Build Coastguard Worker 	}
130*5a6e8488SAndroid Build Coastguard Worker 
131*5a6e8488SAndroid Build Coastguard Worker 	if (bc_opt_longoptsEnd(longopts, i)) return -1;
132*5a6e8488SAndroid Build Coastguard Worker 
133*5a6e8488SAndroid Build Coastguard Worker 	return (int) longopts[i].type;
134*5a6e8488SAndroid Build Coastguard Worker }
135*5a6e8488SAndroid Build Coastguard Worker 
136*5a6e8488SAndroid Build Coastguard Worker /**
137*5a6e8488SAndroid Build Coastguard Worker  * Parses a short option.
138*5a6e8488SAndroid Build Coastguard Worker  * @param o         The option parser.
139*5a6e8488SAndroid Build Coastguard Worker  * @param longopts  The long options array.
140*5a6e8488SAndroid Build Coastguard Worker  * @return          The character for the short option, or -1 if none left.
141*5a6e8488SAndroid Build Coastguard Worker  */
142*5a6e8488SAndroid Build Coastguard Worker static int
bc_opt_parseShort(BcOpt * o,const BcOptLong * longopts)143*5a6e8488SAndroid Build Coastguard Worker bc_opt_parseShort(BcOpt* o, const BcOptLong* longopts)
144*5a6e8488SAndroid Build Coastguard Worker {
145*5a6e8488SAndroid Build Coastguard Worker 	int type;
146*5a6e8488SAndroid Build Coastguard Worker 	const char* next;
147*5a6e8488SAndroid Build Coastguard Worker 	const char* option = o->argv[o->optind];
148*5a6e8488SAndroid Build Coastguard Worker 	int ret = -1;
149*5a6e8488SAndroid Build Coastguard Worker 
150*5a6e8488SAndroid Build Coastguard Worker 	// Make sure to clear these.
151*5a6e8488SAndroid Build Coastguard Worker 	o->optopt = 0;
152*5a6e8488SAndroid Build Coastguard Worker 	o->optarg = NULL;
153*5a6e8488SAndroid Build Coastguard Worker 
154*5a6e8488SAndroid Build Coastguard Worker 	// Get the next option.
155*5a6e8488SAndroid Build Coastguard Worker 	option += o->subopt + 1;
156*5a6e8488SAndroid Build Coastguard Worker 	o->optopt = option[0];
157*5a6e8488SAndroid Build Coastguard Worker 
158*5a6e8488SAndroid Build Coastguard Worker 	// Get the type and the next data.
159*5a6e8488SAndroid Build Coastguard Worker 	type = bc_opt_type(longopts, option[0]);
160*5a6e8488SAndroid Build Coastguard Worker 	next = o->argv[o->optind + 1];
161*5a6e8488SAndroid Build Coastguard Worker 
162*5a6e8488SAndroid Build Coastguard Worker 	switch (type)
163*5a6e8488SAndroid Build Coastguard Worker 	{
164*5a6e8488SAndroid Build Coastguard Worker 		case -1:
165*5a6e8488SAndroid Build Coastguard Worker 		case BC_OPT_BC_ONLY:
166*5a6e8488SAndroid Build Coastguard Worker 		case BC_OPT_DC_ONLY:
167*5a6e8488SAndroid Build Coastguard Worker 		{
168*5a6e8488SAndroid Build Coastguard Worker 			// Check for invalid option and barf if so.
169*5a6e8488SAndroid Build Coastguard Worker 			if (type == -1 || (type == BC_OPT_BC_ONLY && BC_IS_DC) ||
170*5a6e8488SAndroid Build Coastguard Worker 			    (type == BC_OPT_DC_ONLY && BC_IS_BC))
171*5a6e8488SAndroid Build Coastguard Worker 			{
172*5a6e8488SAndroid Build Coastguard Worker 				char str[2] = { 0, 0 };
173*5a6e8488SAndroid Build Coastguard Worker 
174*5a6e8488SAndroid Build Coastguard Worker 				str[0] = option[0];
175*5a6e8488SAndroid Build Coastguard Worker 				o->optind += 1;
176*5a6e8488SAndroid Build Coastguard Worker 
177*5a6e8488SAndroid Build Coastguard Worker 				bc_opt_error(BC_ERR_FATAL_OPTION, option[0], str, true);
178*5a6e8488SAndroid Build Coastguard Worker 			}
179*5a6e8488SAndroid Build Coastguard Worker 
180*5a6e8488SAndroid Build Coastguard Worker 			// Fallthrough.
181*5a6e8488SAndroid Build Coastguard Worker 			BC_FALLTHROUGH
182*5a6e8488SAndroid Build Coastguard Worker 		}
183*5a6e8488SAndroid Build Coastguard Worker 
184*5a6e8488SAndroid Build Coastguard Worker 		case BC_OPT_NONE:
185*5a6e8488SAndroid Build Coastguard Worker 		{
186*5a6e8488SAndroid Build Coastguard Worker 			// If there is something else, update the suboption.
187*5a6e8488SAndroid Build Coastguard Worker 			if (option[1]) o->subopt += 1;
188*5a6e8488SAndroid Build Coastguard Worker 			else
189*5a6e8488SAndroid Build Coastguard Worker 			{
190*5a6e8488SAndroid Build Coastguard Worker 				// Go to the next argument.
191*5a6e8488SAndroid Build Coastguard Worker 				o->subopt = 0;
192*5a6e8488SAndroid Build Coastguard Worker 				o->optind += 1;
193*5a6e8488SAndroid Build Coastguard Worker 			}
194*5a6e8488SAndroid Build Coastguard Worker 
195*5a6e8488SAndroid Build Coastguard Worker 			ret = (int) option[0];
196*5a6e8488SAndroid Build Coastguard Worker 
197*5a6e8488SAndroid Build Coastguard Worker 			break;
198*5a6e8488SAndroid Build Coastguard Worker 		}
199*5a6e8488SAndroid Build Coastguard Worker 
200*5a6e8488SAndroid Build Coastguard Worker 		case BC_OPT_REQUIRED_BC_ONLY:
201*5a6e8488SAndroid Build Coastguard Worker 		{
202*5a6e8488SAndroid Build Coastguard Worker #if DC_ENABLED
203*5a6e8488SAndroid Build Coastguard Worker 			if (BC_IS_DC)
204*5a6e8488SAndroid Build Coastguard Worker 			{
205*5a6e8488SAndroid Build Coastguard Worker 				bc_opt_error(BC_ERR_FATAL_OPTION, option[0],
206*5a6e8488SAndroid Build Coastguard Worker 				             bc_opt_longopt(longopts, option[0]), true);
207*5a6e8488SAndroid Build Coastguard Worker 			}
208*5a6e8488SAndroid Build Coastguard Worker #endif // DC_ENABLED
209*5a6e8488SAndroid Build Coastguard Worker 
210*5a6e8488SAndroid Build Coastguard Worker 			// Fallthrough
211*5a6e8488SAndroid Build Coastguard Worker 			BC_FALLTHROUGH
212*5a6e8488SAndroid Build Coastguard Worker 		}
213*5a6e8488SAndroid Build Coastguard Worker 
214*5a6e8488SAndroid Build Coastguard Worker 		case BC_OPT_REQUIRED:
215*5a6e8488SAndroid Build Coastguard Worker 		{
216*5a6e8488SAndroid Build Coastguard Worker 			// Always go to the next argument.
217*5a6e8488SAndroid Build Coastguard Worker 			o->subopt = 0;
218*5a6e8488SAndroid Build Coastguard Worker 			o->optind += 1;
219*5a6e8488SAndroid Build Coastguard Worker 
220*5a6e8488SAndroid Build Coastguard Worker 			// Use the next characters, if they exist.
221*5a6e8488SAndroid Build Coastguard Worker 			if (option[1]) o->optarg = option + 1;
222*5a6e8488SAndroid Build Coastguard Worker 			else if (next != NULL)
223*5a6e8488SAndroid Build Coastguard Worker 			{
224*5a6e8488SAndroid Build Coastguard Worker 				// USe the next.
225*5a6e8488SAndroid Build Coastguard Worker 				o->optarg = next;
226*5a6e8488SAndroid Build Coastguard Worker 				o->optind += 1;
227*5a6e8488SAndroid Build Coastguard Worker 			}
228*5a6e8488SAndroid Build Coastguard Worker 			// No argument, barf.
229*5a6e8488SAndroid Build Coastguard Worker 			else
230*5a6e8488SAndroid Build Coastguard Worker 			{
231*5a6e8488SAndroid Build Coastguard Worker 				bc_opt_error(BC_ERR_FATAL_OPTION_NO_ARG, option[0],
232*5a6e8488SAndroid Build Coastguard Worker 				             bc_opt_longopt(longopts, option[0]), true);
233*5a6e8488SAndroid Build Coastguard Worker 			}
234*5a6e8488SAndroid Build Coastguard Worker 
235*5a6e8488SAndroid Build Coastguard Worker 			ret = (int) option[0];
236*5a6e8488SAndroid Build Coastguard Worker 
237*5a6e8488SAndroid Build Coastguard Worker 			break;
238*5a6e8488SAndroid Build Coastguard Worker 		}
239*5a6e8488SAndroid Build Coastguard Worker 	}
240*5a6e8488SAndroid Build Coastguard Worker 
241*5a6e8488SAndroid Build Coastguard Worker 	return ret;
242*5a6e8488SAndroid Build Coastguard Worker }
243*5a6e8488SAndroid Build Coastguard Worker 
244*5a6e8488SAndroid Build Coastguard Worker /**
245*5a6e8488SAndroid Build Coastguard Worker  * Ensures that a long option argument matches a long option name, regardless of
246*5a6e8488SAndroid Build Coastguard Worker  * "=<data>" at the end.
247*5a6e8488SAndroid Build Coastguard Worker  * @param name    The name to match.
248*5a6e8488SAndroid Build Coastguard Worker  * @param option  The command-line argument.
249*5a6e8488SAndroid Build Coastguard Worker  * @return        True if @a option matches @a name, false otherwise.
250*5a6e8488SAndroid Build Coastguard Worker  */
251*5a6e8488SAndroid Build Coastguard Worker static bool
bc_opt_longoptsMatch(const char * name,const char * option)252*5a6e8488SAndroid Build Coastguard Worker bc_opt_longoptsMatch(const char* name, const char* option)
253*5a6e8488SAndroid Build Coastguard Worker {
254*5a6e8488SAndroid Build Coastguard Worker 	const char* a = option;
255*5a6e8488SAndroid Build Coastguard Worker 	const char* n = name;
256*5a6e8488SAndroid Build Coastguard Worker 
257*5a6e8488SAndroid Build Coastguard Worker 	// Can never match a NULL name.
258*5a6e8488SAndroid Build Coastguard Worker 	if (name == NULL) return false;
259*5a6e8488SAndroid Build Coastguard Worker 
260*5a6e8488SAndroid Build Coastguard Worker 	// Loop through.
261*5a6e8488SAndroid Build Coastguard Worker 	for (; *a && *n && *a != '='; ++a, ++n)
262*5a6e8488SAndroid Build Coastguard Worker 	{
263*5a6e8488SAndroid Build Coastguard Worker 		if (*a != *n) return false;
264*5a6e8488SAndroid Build Coastguard Worker 	}
265*5a6e8488SAndroid Build Coastguard Worker 
266*5a6e8488SAndroid Build Coastguard Worker 	// Ensure they both end at the same place.
267*5a6e8488SAndroid Build Coastguard Worker 	return (*n == '\0' && (*a == '\0' || *a == '='));
268*5a6e8488SAndroid Build Coastguard Worker }
269*5a6e8488SAndroid Build Coastguard Worker 
270*5a6e8488SAndroid Build Coastguard Worker /**
271*5a6e8488SAndroid Build Coastguard Worker  * Returns a pointer to the argument of a long option, or NULL if it not in the
272*5a6e8488SAndroid Build Coastguard Worker  * same argument.
273*5a6e8488SAndroid Build Coastguard Worker  * @param option  The option to find the argument of.
274*5a6e8488SAndroid Build Coastguard Worker  * @return        A pointer to the argument of the option, or NULL if none.
275*5a6e8488SAndroid Build Coastguard Worker  */
276*5a6e8488SAndroid Build Coastguard Worker static const char*
bc_opt_longoptsArg(const char * option)277*5a6e8488SAndroid Build Coastguard Worker bc_opt_longoptsArg(const char* option)
278*5a6e8488SAndroid Build Coastguard Worker {
279*5a6e8488SAndroid Build Coastguard Worker 	// Find the end or equals sign.
280*5a6e8488SAndroid Build Coastguard Worker 	for (; *option && *option != '='; ++option)
281*5a6e8488SAndroid Build Coastguard Worker 	{
282*5a6e8488SAndroid Build Coastguard Worker 		continue;
283*5a6e8488SAndroid Build Coastguard Worker 	}
284*5a6e8488SAndroid Build Coastguard Worker 
285*5a6e8488SAndroid Build Coastguard Worker 	if (*option == '=') return option + 1;
286*5a6e8488SAndroid Build Coastguard Worker 	else return NULL;
287*5a6e8488SAndroid Build Coastguard Worker }
288*5a6e8488SAndroid Build Coastguard Worker 
289*5a6e8488SAndroid Build Coastguard Worker int
bc_opt_parse(BcOpt * o,const BcOptLong * longopts)290*5a6e8488SAndroid Build Coastguard Worker bc_opt_parse(BcOpt* o, const BcOptLong* longopts)
291*5a6e8488SAndroid Build Coastguard Worker {
292*5a6e8488SAndroid Build Coastguard Worker 	size_t i;
293*5a6e8488SAndroid Build Coastguard Worker 	const char* option;
294*5a6e8488SAndroid Build Coastguard Worker 	bool empty;
295*5a6e8488SAndroid Build Coastguard Worker 
296*5a6e8488SAndroid Build Coastguard Worker 	// This just eats empty options.
297*5a6e8488SAndroid Build Coastguard Worker 	do
298*5a6e8488SAndroid Build Coastguard Worker 	{
299*5a6e8488SAndroid Build Coastguard Worker 		option = o->argv[o->optind];
300*5a6e8488SAndroid Build Coastguard Worker 		if (option == NULL) return -1;
301*5a6e8488SAndroid Build Coastguard Worker 
302*5a6e8488SAndroid Build Coastguard Worker 		empty = !strcmp(option, "");
303*5a6e8488SAndroid Build Coastguard Worker 		o->optind += empty;
304*5a6e8488SAndroid Build Coastguard Worker 	}
305*5a6e8488SAndroid Build Coastguard Worker 	while (empty);
306*5a6e8488SAndroid Build Coastguard Worker 
307*5a6e8488SAndroid Build Coastguard Worker 	// If the option is just a "--".
308*5a6e8488SAndroid Build Coastguard Worker 	if (BC_OPT_ISDASHDASH(option))
309*5a6e8488SAndroid Build Coastguard Worker 	{
310*5a6e8488SAndroid Build Coastguard Worker 		// Consume "--".
311*5a6e8488SAndroid Build Coastguard Worker 		o->optind += 1;
312*5a6e8488SAndroid Build Coastguard Worker 		return -1;
313*5a6e8488SAndroid Build Coastguard Worker 	}
314*5a6e8488SAndroid Build Coastguard Worker 	// Parse a short option.
315*5a6e8488SAndroid Build Coastguard Worker 	else if (BC_OPT_ISSHORTOPT(option)) return bc_opt_parseShort(o, longopts);
316*5a6e8488SAndroid Build Coastguard Worker 	// If the option is not long at this point, we are done.
317*5a6e8488SAndroid Build Coastguard Worker 	else if (!BC_OPT_ISLONGOPT(option)) return -1;
318*5a6e8488SAndroid Build Coastguard Worker 
319*5a6e8488SAndroid Build Coastguard Worker 	// Clear these.
320*5a6e8488SAndroid Build Coastguard Worker 	o->optopt = 0;
321*5a6e8488SAndroid Build Coastguard Worker 	o->optarg = NULL;
322*5a6e8488SAndroid Build Coastguard Worker 
323*5a6e8488SAndroid Build Coastguard Worker 	// Skip "--" at beginning of the option.
324*5a6e8488SAndroid Build Coastguard Worker 	option += 2;
325*5a6e8488SAndroid Build Coastguard Worker 	o->optind += 1;
326*5a6e8488SAndroid Build Coastguard Worker 
327*5a6e8488SAndroid Build Coastguard Worker 	// Loop through the valid long options.
328*5a6e8488SAndroid Build Coastguard Worker 	for (i = 0; !bc_opt_longoptsEnd(longopts, i); i++)
329*5a6e8488SAndroid Build Coastguard Worker 	{
330*5a6e8488SAndroid Build Coastguard Worker 		const char* name = longopts[i].name;
331*5a6e8488SAndroid Build Coastguard Worker 
332*5a6e8488SAndroid Build Coastguard Worker 		// If we have a match...
333*5a6e8488SAndroid Build Coastguard Worker 		if (bc_opt_longoptsMatch(name, option))
334*5a6e8488SAndroid Build Coastguard Worker 		{
335*5a6e8488SAndroid Build Coastguard Worker 			const char* arg;
336*5a6e8488SAndroid Build Coastguard Worker 
337*5a6e8488SAndroid Build Coastguard Worker 			// Get the option char and the argument.
338*5a6e8488SAndroid Build Coastguard Worker 			o->optopt = longopts[i].val;
339*5a6e8488SAndroid Build Coastguard Worker 			arg = bc_opt_longoptsArg(option);
340*5a6e8488SAndroid Build Coastguard Worker 
341*5a6e8488SAndroid Build Coastguard Worker 			// Error if the option is invalid..
342*5a6e8488SAndroid Build Coastguard Worker 			if ((longopts[i].type == BC_OPT_BC_ONLY && BC_IS_DC) ||
343*5a6e8488SAndroid Build Coastguard Worker 			    (longopts[i].type == BC_OPT_REQUIRED_BC_ONLY && BC_IS_DC) ||
344*5a6e8488SAndroid Build Coastguard Worker 			    (longopts[i].type == BC_OPT_DC_ONLY && BC_IS_BC))
345*5a6e8488SAndroid Build Coastguard Worker 			{
346*5a6e8488SAndroid Build Coastguard Worker 				bc_opt_error(BC_ERR_FATAL_OPTION, o->optopt, name, false);
347*5a6e8488SAndroid Build Coastguard Worker 			}
348*5a6e8488SAndroid Build Coastguard Worker 
349*5a6e8488SAndroid Build Coastguard Worker 			// Error if we have an argument and should not.
350*5a6e8488SAndroid Build Coastguard Worker 			if (longopts[i].type == BC_OPT_NONE && arg != NULL)
351*5a6e8488SAndroid Build Coastguard Worker 			{
352*5a6e8488SAndroid Build Coastguard Worker 				bc_opt_error(BC_ERR_FATAL_OPTION_ARG, o->optopt, name, false);
353*5a6e8488SAndroid Build Coastguard Worker 			}
354*5a6e8488SAndroid Build Coastguard Worker 
355*5a6e8488SAndroid Build Coastguard Worker 			// Set the argument, or check the next argument if we don't have
356*5a6e8488SAndroid Build Coastguard Worker 			// one.
357*5a6e8488SAndroid Build Coastguard Worker 			if (arg != NULL) o->optarg = arg;
358*5a6e8488SAndroid Build Coastguard Worker 			else if (longopts[i].type == BC_OPT_REQUIRED ||
359*5a6e8488SAndroid Build Coastguard Worker 			         longopts[i].type == BC_OPT_REQUIRED_BC_ONLY)
360*5a6e8488SAndroid Build Coastguard Worker 			{
361*5a6e8488SAndroid Build Coastguard Worker 				// Get the next argument.
362*5a6e8488SAndroid Build Coastguard Worker 				o->optarg = o->argv[o->optind];
363*5a6e8488SAndroid Build Coastguard Worker 
364*5a6e8488SAndroid Build Coastguard Worker 				// All's good if it exists; otherwise, barf.
365*5a6e8488SAndroid Build Coastguard Worker 				if (o->optarg != NULL) o->optind += 1;
366*5a6e8488SAndroid Build Coastguard Worker 				else
367*5a6e8488SAndroid Build Coastguard Worker 				{
368*5a6e8488SAndroid Build Coastguard Worker 					bc_opt_error(BC_ERR_FATAL_OPTION_NO_ARG, o->optopt, name,
369*5a6e8488SAndroid Build Coastguard Worker 					             false);
370*5a6e8488SAndroid Build Coastguard Worker 				}
371*5a6e8488SAndroid Build Coastguard Worker 			}
372*5a6e8488SAndroid Build Coastguard Worker 
373*5a6e8488SAndroid Build Coastguard Worker 			return o->optopt;
374*5a6e8488SAndroid Build Coastguard Worker 		}
375*5a6e8488SAndroid Build Coastguard Worker 	}
376*5a6e8488SAndroid Build Coastguard Worker 
377*5a6e8488SAndroid Build Coastguard Worker 	// If we reach this point, the option is invalid.
378*5a6e8488SAndroid Build Coastguard Worker 	bc_opt_error(BC_ERR_FATAL_OPTION, 0, option, false);
379*5a6e8488SAndroid Build Coastguard Worker 
380*5a6e8488SAndroid Build Coastguard Worker 	BC_UNREACHABLE
381*5a6e8488SAndroid Build Coastguard Worker 
382*5a6e8488SAndroid Build Coastguard Worker #if !BC_CLANG
383*5a6e8488SAndroid Build Coastguard Worker 	return -1;
384*5a6e8488SAndroid Build Coastguard Worker #endif // !BC_CLANG
385*5a6e8488SAndroid Build Coastguard Worker }
386*5a6e8488SAndroid Build Coastguard Worker 
387*5a6e8488SAndroid Build Coastguard Worker void
bc_opt_init(BcOpt * o,const char * argv[])388*5a6e8488SAndroid Build Coastguard Worker bc_opt_init(BcOpt* o, const char* argv[])
389*5a6e8488SAndroid Build Coastguard Worker {
390*5a6e8488SAndroid Build Coastguard Worker 	o->argv = argv;
391*5a6e8488SAndroid Build Coastguard Worker 	o->optind = 1;
392*5a6e8488SAndroid Build Coastguard Worker 	o->subopt = 0;
393*5a6e8488SAndroid Build Coastguard Worker 	o->optarg = NULL;
394*5a6e8488SAndroid Build Coastguard Worker }
395