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