xref: /aosp_15_r20/external/musl/src/misc/getopt.c (revision c9945492fdd68bbe62686c5b452b4dc1be3f8453)
1*c9945492SAndroid Build Coastguard Worker #define _BSD_SOURCE
2*c9945492SAndroid Build Coastguard Worker #include <unistd.h>
3*c9945492SAndroid Build Coastguard Worker #include <wchar.h>
4*c9945492SAndroid Build Coastguard Worker #include <string.h>
5*c9945492SAndroid Build Coastguard Worker #include <limits.h>
6*c9945492SAndroid Build Coastguard Worker #include <stdlib.h>
7*c9945492SAndroid Build Coastguard Worker #include "locale_impl.h"
8*c9945492SAndroid Build Coastguard Worker #include "stdio_impl.h"
9*c9945492SAndroid Build Coastguard Worker 
10*c9945492SAndroid Build Coastguard Worker char *optarg;
11*c9945492SAndroid Build Coastguard Worker int optind=1, opterr=1, optopt, __optpos, __optreset=0;
12*c9945492SAndroid Build Coastguard Worker 
13*c9945492SAndroid Build Coastguard Worker #define optpos __optpos
14*c9945492SAndroid Build Coastguard Worker weak_alias(__optreset, optreset);
15*c9945492SAndroid Build Coastguard Worker 
__getopt_msg(const char * a,const char * b,const char * c,size_t l)16*c9945492SAndroid Build Coastguard Worker void __getopt_msg(const char *a, const char *b, const char *c, size_t l)
17*c9945492SAndroid Build Coastguard Worker {
18*c9945492SAndroid Build Coastguard Worker 	FILE *f = stderr;
19*c9945492SAndroid Build Coastguard Worker 	b = __lctrans_cur(b);
20*c9945492SAndroid Build Coastguard Worker 	FLOCK(f);
21*c9945492SAndroid Build Coastguard Worker 	fputs(a, f)>=0
22*c9945492SAndroid Build Coastguard Worker 	&& fwrite(b, strlen(b), 1, f)
23*c9945492SAndroid Build Coastguard Worker 	&& fwrite(c, 1, l, f)==l
24*c9945492SAndroid Build Coastguard Worker 	&& putc('\n', f);
25*c9945492SAndroid Build Coastguard Worker 	FUNLOCK(f);
26*c9945492SAndroid Build Coastguard Worker }
27*c9945492SAndroid Build Coastguard Worker 
__posix_getopt(int argc,char * const argv[],const char * optstring)28*c9945492SAndroid Build Coastguard Worker int __posix_getopt(int argc, char * const argv[], const char *optstring)
29*c9945492SAndroid Build Coastguard Worker {
30*c9945492SAndroid Build Coastguard Worker 	int i;
31*c9945492SAndroid Build Coastguard Worker 	wchar_t c, d;
32*c9945492SAndroid Build Coastguard Worker 	int k, l;
33*c9945492SAndroid Build Coastguard Worker 	char *optchar;
34*c9945492SAndroid Build Coastguard Worker 
35*c9945492SAndroid Build Coastguard Worker 	if (!optind || __optreset) {
36*c9945492SAndroid Build Coastguard Worker 		__optreset = 0;
37*c9945492SAndroid Build Coastguard Worker 		__optpos = 0;
38*c9945492SAndroid Build Coastguard Worker 		optind = 1;
39*c9945492SAndroid Build Coastguard Worker 	}
40*c9945492SAndroid Build Coastguard Worker 
41*c9945492SAndroid Build Coastguard Worker 	if (optind >= argc || !argv[optind])
42*c9945492SAndroid Build Coastguard Worker 		return -1;
43*c9945492SAndroid Build Coastguard Worker 
44*c9945492SAndroid Build Coastguard Worker 	if (argv[optind][0] != '-') {
45*c9945492SAndroid Build Coastguard Worker 		if (optstring[0] == '-') {
46*c9945492SAndroid Build Coastguard Worker 			optarg = argv[optind++];
47*c9945492SAndroid Build Coastguard Worker 			return 1;
48*c9945492SAndroid Build Coastguard Worker 		}
49*c9945492SAndroid Build Coastguard Worker 		return -1;
50*c9945492SAndroid Build Coastguard Worker 	}
51*c9945492SAndroid Build Coastguard Worker 
52*c9945492SAndroid Build Coastguard Worker 	if (!argv[optind][1])
53*c9945492SAndroid Build Coastguard Worker 		return -1;
54*c9945492SAndroid Build Coastguard Worker 
55*c9945492SAndroid Build Coastguard Worker 	if (argv[optind][1] == '-' && !argv[optind][2])
56*c9945492SAndroid Build Coastguard Worker 		return optind++, -1;
57*c9945492SAndroid Build Coastguard Worker 
58*c9945492SAndroid Build Coastguard Worker 	if (!optpos) optpos++;
59*c9945492SAndroid Build Coastguard Worker 	if ((k = mbtowc(&c, argv[optind]+optpos, MB_LEN_MAX)) < 0) {
60*c9945492SAndroid Build Coastguard Worker 		k = 1;
61*c9945492SAndroid Build Coastguard Worker 		c = 0xfffd; /* replacement char */
62*c9945492SAndroid Build Coastguard Worker 	}
63*c9945492SAndroid Build Coastguard Worker 	optchar = argv[optind]+optpos;
64*c9945492SAndroid Build Coastguard Worker 	optpos += k;
65*c9945492SAndroid Build Coastguard Worker 
66*c9945492SAndroid Build Coastguard Worker 	if (!argv[optind][optpos]) {
67*c9945492SAndroid Build Coastguard Worker 		optind++;
68*c9945492SAndroid Build Coastguard Worker 		optpos = 0;
69*c9945492SAndroid Build Coastguard Worker 	}
70*c9945492SAndroid Build Coastguard Worker 
71*c9945492SAndroid Build Coastguard Worker 	if (optstring[0] == '-' || optstring[0] == '+')
72*c9945492SAndroid Build Coastguard Worker 		optstring++;
73*c9945492SAndroid Build Coastguard Worker 
74*c9945492SAndroid Build Coastguard Worker 	i = 0;
75*c9945492SAndroid Build Coastguard Worker 	d = 0;
76*c9945492SAndroid Build Coastguard Worker 	do {
77*c9945492SAndroid Build Coastguard Worker 		l = mbtowc(&d, optstring+i, MB_LEN_MAX);
78*c9945492SAndroid Build Coastguard Worker 		if (l>0) i+=l; else i++;
79*c9945492SAndroid Build Coastguard Worker 	} while (l && d != c);
80*c9945492SAndroid Build Coastguard Worker 
81*c9945492SAndroid Build Coastguard Worker 	if (d != c || c == ':') {
82*c9945492SAndroid Build Coastguard Worker 		optopt = c;
83*c9945492SAndroid Build Coastguard Worker 		if (optstring[0] != ':' && opterr)
84*c9945492SAndroid Build Coastguard Worker 			__getopt_msg(argv[0], ": unrecognized option: ", optchar, k);
85*c9945492SAndroid Build Coastguard Worker 		return '?';
86*c9945492SAndroid Build Coastguard Worker 	}
87*c9945492SAndroid Build Coastguard Worker 	if (optstring[i] == ':') {
88*c9945492SAndroid Build Coastguard Worker 		optarg = 0;
89*c9945492SAndroid Build Coastguard Worker 		if (optstring[i+1] != ':' || optpos) {
90*c9945492SAndroid Build Coastguard Worker 			optarg = argv[optind++];
91*c9945492SAndroid Build Coastguard Worker 			if (optpos) optarg += optpos;
92*c9945492SAndroid Build Coastguard Worker 			optpos = 0;
93*c9945492SAndroid Build Coastguard Worker 		}
94*c9945492SAndroid Build Coastguard Worker 		if (optind > argc) {
95*c9945492SAndroid Build Coastguard Worker 			optopt = c;
96*c9945492SAndroid Build Coastguard Worker 			if (optstring[0] == ':') return ':';
97*c9945492SAndroid Build Coastguard Worker 			if (opterr) __getopt_msg(argv[0],
98*c9945492SAndroid Build Coastguard Worker 				": option requires an argument: ",
99*c9945492SAndroid Build Coastguard Worker 				optchar, k);
100*c9945492SAndroid Build Coastguard Worker 			return '?';
101*c9945492SAndroid Build Coastguard Worker 		}
102*c9945492SAndroid Build Coastguard Worker 	}
103*c9945492SAndroid Build Coastguard Worker 	return c;
104*c9945492SAndroid Build Coastguard Worker }
105