xref: /aosp_15_r20/external/e2fsprogs/lib/support/argv_parse.c (revision 6a54128f25917bfc36a8a6e9d722c04a0b4641b6)
1*6a54128fSAndroid Build Coastguard Worker /*
2*6a54128fSAndroid Build Coastguard Worker  * argv_parse.c --- utility function for parsing a string into a
3*6a54128fSAndroid Build Coastguard Worker  * 	argc, argv array.
4*6a54128fSAndroid Build Coastguard Worker  *
5*6a54128fSAndroid Build Coastguard Worker  * This file defines a function argv_parse() which parsing a
6*6a54128fSAndroid Build Coastguard Worker  * passed-in string, handling double quotes and backslashes, and
7*6a54128fSAndroid Build Coastguard Worker  * creates an allocated argv vector which can be freed using the
8*6a54128fSAndroid Build Coastguard Worker  * argv_free() function.
9*6a54128fSAndroid Build Coastguard Worker  *
10*6a54128fSAndroid Build Coastguard Worker  * See argv_parse.h for the formal definition of the functions.
11*6a54128fSAndroid Build Coastguard Worker  *
12*6a54128fSAndroid Build Coastguard Worker  * Copyright 1999 by Theodore Ts'o.
13*6a54128fSAndroid Build Coastguard Worker  *
14*6a54128fSAndroid Build Coastguard Worker  * Permission to use, copy, modify, and distribute this software for
15*6a54128fSAndroid Build Coastguard Worker  * any purpose with or without fee is hereby granted, provided that
16*6a54128fSAndroid Build Coastguard Worker  * the above copyright notice and this permission notice appear in all
17*6a54128fSAndroid Build Coastguard Worker  * copies.  THE SOFTWARE IS PROVIDED "AS IS" AND THEODORE TS'O (THE
18*6a54128fSAndroid Build Coastguard Worker  * AUTHOR) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
19*6a54128fSAndroid Build Coastguard Worker  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.
20*6a54128fSAndroid Build Coastguard Worker  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
21*6a54128fSAndroid Build Coastguard Worker  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
22*6a54128fSAndroid Build Coastguard Worker  * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
23*6a54128fSAndroid Build Coastguard Worker  * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
24*6a54128fSAndroid Build Coastguard Worker  * IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.  (Isn't
25*6a54128fSAndroid Build Coastguard Worker  * it sick that the U.S. culture of lawsuit-happy lawyers requires
26*6a54128fSAndroid Build Coastguard Worker  * this kind of disclaimer?)
27*6a54128fSAndroid Build Coastguard Worker  *
28*6a54128fSAndroid Build Coastguard Worker  * Version 1.1, modified 2/27/1999
29*6a54128fSAndroid Build Coastguard Worker  */
30*6a54128fSAndroid Build Coastguard Worker 
31*6a54128fSAndroid Build Coastguard Worker #include "config.h"
32*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_STDLIB_H
33*6a54128fSAndroid Build Coastguard Worker #include <stdlib.h>
34*6a54128fSAndroid Build Coastguard Worker #endif
35*6a54128fSAndroid Build Coastguard Worker #include <ctype.h>
36*6a54128fSAndroid Build Coastguard Worker #include <string.h>
37*6a54128fSAndroid Build Coastguard Worker #include "argv_parse.h"
38*6a54128fSAndroid Build Coastguard Worker 
39*6a54128fSAndroid Build Coastguard Worker #define STATE_WHITESPACE	1
40*6a54128fSAndroid Build Coastguard Worker #define STATE_TOKEN		2
41*6a54128fSAndroid Build Coastguard Worker #define STATE_QUOTED		3
42*6a54128fSAndroid Build Coastguard Worker 
43*6a54128fSAndroid Build Coastguard Worker /*
44*6a54128fSAndroid Build Coastguard Worker  * Returns 0 on success, -1 on failure.
45*6a54128fSAndroid Build Coastguard Worker  */
argv_parse(char * in_buf,int * ret_argc,char *** ret_argv)46*6a54128fSAndroid Build Coastguard Worker int argv_parse(char *in_buf, int *ret_argc, char ***ret_argv)
47*6a54128fSAndroid Build Coastguard Worker {
48*6a54128fSAndroid Build Coastguard Worker 	int	argc = 0, max_argc = 0;
49*6a54128fSAndroid Build Coastguard Worker 	char 	**argv, **new_argv, *buf, ch;
50*6a54128fSAndroid Build Coastguard Worker 	char	*cp = 0, *outcp = 0;
51*6a54128fSAndroid Build Coastguard Worker 	int	state = STATE_WHITESPACE;
52*6a54128fSAndroid Build Coastguard Worker 
53*6a54128fSAndroid Build Coastguard Worker 	buf = malloc(strlen(in_buf)+1);
54*6a54128fSAndroid Build Coastguard Worker 	if (!buf)
55*6a54128fSAndroid Build Coastguard Worker 		return -1;
56*6a54128fSAndroid Build Coastguard Worker 
57*6a54128fSAndroid Build Coastguard Worker 	max_argc = 0; argc = 0; argv = 0;
58*6a54128fSAndroid Build Coastguard Worker 	outcp = buf;
59*6a54128fSAndroid Build Coastguard Worker 	for (cp = in_buf; (ch = *cp); cp++) {
60*6a54128fSAndroid Build Coastguard Worker 		if (state == STATE_WHITESPACE) {
61*6a54128fSAndroid Build Coastguard Worker 			if (isspace((int) ch))
62*6a54128fSAndroid Build Coastguard Worker 				continue;
63*6a54128fSAndroid Build Coastguard Worker 			/* Not whitespace, so start a new token */
64*6a54128fSAndroid Build Coastguard Worker 			state = STATE_TOKEN;
65*6a54128fSAndroid Build Coastguard Worker 			if (argc >= max_argc) {
66*6a54128fSAndroid Build Coastguard Worker 				max_argc += 3;
67*6a54128fSAndroid Build Coastguard Worker 				new_argv = realloc(argv,
68*6a54128fSAndroid Build Coastguard Worker 						  (max_argc+1)*sizeof(char *));
69*6a54128fSAndroid Build Coastguard Worker 				if (!new_argv) {
70*6a54128fSAndroid Build Coastguard Worker 					free(argv);
71*6a54128fSAndroid Build Coastguard Worker 					free(buf);
72*6a54128fSAndroid Build Coastguard Worker 					return -1;
73*6a54128fSAndroid Build Coastguard Worker 				}
74*6a54128fSAndroid Build Coastguard Worker 				argv = new_argv;
75*6a54128fSAndroid Build Coastguard Worker 			}
76*6a54128fSAndroid Build Coastguard Worker 			argv[argc++] = outcp;
77*6a54128fSAndroid Build Coastguard Worker 		}
78*6a54128fSAndroid Build Coastguard Worker 		if (state == STATE_QUOTED) {
79*6a54128fSAndroid Build Coastguard Worker 			if (ch == '"')
80*6a54128fSAndroid Build Coastguard Worker 				state = STATE_TOKEN;
81*6a54128fSAndroid Build Coastguard Worker 			else
82*6a54128fSAndroid Build Coastguard Worker 				*outcp++ = ch;
83*6a54128fSAndroid Build Coastguard Worker 			continue;
84*6a54128fSAndroid Build Coastguard Worker 		}
85*6a54128fSAndroid Build Coastguard Worker 		/* Must be processing characters in a word */
86*6a54128fSAndroid Build Coastguard Worker 		if (isspace((int) ch)) {
87*6a54128fSAndroid Build Coastguard Worker 			/*
88*6a54128fSAndroid Build Coastguard Worker 			 * Terminate the current word and start
89*6a54128fSAndroid Build Coastguard Worker 			 * looking for the beginning of the next word.
90*6a54128fSAndroid Build Coastguard Worker 			 */
91*6a54128fSAndroid Build Coastguard Worker 			*outcp++ = 0;
92*6a54128fSAndroid Build Coastguard Worker 			state = STATE_WHITESPACE;
93*6a54128fSAndroid Build Coastguard Worker 			continue;
94*6a54128fSAndroid Build Coastguard Worker 		}
95*6a54128fSAndroid Build Coastguard Worker 		if (ch == '"') {
96*6a54128fSAndroid Build Coastguard Worker 			state = STATE_QUOTED;
97*6a54128fSAndroid Build Coastguard Worker 			continue;
98*6a54128fSAndroid Build Coastguard Worker 		}
99*6a54128fSAndroid Build Coastguard Worker 		if (ch == '\\') {
100*6a54128fSAndroid Build Coastguard Worker 			ch = *++cp;
101*6a54128fSAndroid Build Coastguard Worker 			switch (ch) {
102*6a54128fSAndroid Build Coastguard Worker 			case '\0':
103*6a54128fSAndroid Build Coastguard Worker 				ch = '\\'; cp--; break;
104*6a54128fSAndroid Build Coastguard Worker 			case 'n':
105*6a54128fSAndroid Build Coastguard Worker 				ch = '\n'; break;
106*6a54128fSAndroid Build Coastguard Worker 			case 't':
107*6a54128fSAndroid Build Coastguard Worker 				ch = '\t'; break;
108*6a54128fSAndroid Build Coastguard Worker 			case 'b':
109*6a54128fSAndroid Build Coastguard Worker 				ch = '\b'; break;
110*6a54128fSAndroid Build Coastguard Worker 			}
111*6a54128fSAndroid Build Coastguard Worker 		}
112*6a54128fSAndroid Build Coastguard Worker 		*outcp++ = ch;
113*6a54128fSAndroid Build Coastguard Worker 	}
114*6a54128fSAndroid Build Coastguard Worker 	if (state != STATE_WHITESPACE)
115*6a54128fSAndroid Build Coastguard Worker 		*outcp++ = '\0';
116*6a54128fSAndroid Build Coastguard Worker 	if (argv == 0) {
117*6a54128fSAndroid Build Coastguard Worker 		argv = malloc(sizeof(char *));
118*6a54128fSAndroid Build Coastguard Worker 		free(buf);
119*6a54128fSAndroid Build Coastguard Worker 		if (!argv)
120*6a54128fSAndroid Build Coastguard Worker 			return -1;
121*6a54128fSAndroid Build Coastguard Worker 	}
122*6a54128fSAndroid Build Coastguard Worker 	argv[argc] = 0;
123*6a54128fSAndroid Build Coastguard Worker 	if (ret_argc)
124*6a54128fSAndroid Build Coastguard Worker 		*ret_argc = argc;
125*6a54128fSAndroid Build Coastguard Worker 	if (ret_argv)
126*6a54128fSAndroid Build Coastguard Worker 		*ret_argv = argv;
127*6a54128fSAndroid Build Coastguard Worker 	return 0;
128*6a54128fSAndroid Build Coastguard Worker }
129*6a54128fSAndroid Build Coastguard Worker 
argv_free(char ** argv)130*6a54128fSAndroid Build Coastguard Worker void argv_free(char **argv)
131*6a54128fSAndroid Build Coastguard Worker {
132*6a54128fSAndroid Build Coastguard Worker 	free(*argv);
133*6a54128fSAndroid Build Coastguard Worker 	free(argv);
134*6a54128fSAndroid Build Coastguard Worker }
135*6a54128fSAndroid Build Coastguard Worker 
136*6a54128fSAndroid Build Coastguard Worker #ifdef DEBUG
137*6a54128fSAndroid Build Coastguard Worker /*
138*6a54128fSAndroid Build Coastguard Worker  * For debugging
139*6a54128fSAndroid Build Coastguard Worker  */
140*6a54128fSAndroid Build Coastguard Worker 
141*6a54128fSAndroid Build Coastguard Worker #include <stdio.h>
142*6a54128fSAndroid Build Coastguard Worker 
main(int argc,char ** argv)143*6a54128fSAndroid Build Coastguard Worker int main(int argc, char **argv)
144*6a54128fSAndroid Build Coastguard Worker {
145*6a54128fSAndroid Build Coastguard Worker 	int	ac, ret;
146*6a54128fSAndroid Build Coastguard Worker 	char	**av, **cpp;
147*6a54128fSAndroid Build Coastguard Worker 	char	buf[256];
148*6a54128fSAndroid Build Coastguard Worker 
149*6a54128fSAndroid Build Coastguard Worker 	while (!feof(stdin)) {
150*6a54128fSAndroid Build Coastguard Worker 		if (fgets(buf, sizeof(buf), stdin) == NULL)
151*6a54128fSAndroid Build Coastguard Worker 			break;
152*6a54128fSAndroid Build Coastguard Worker 		ret = argv_parse(buf, &ac, &av);
153*6a54128fSAndroid Build Coastguard Worker 		if (ret != 0) {
154*6a54128fSAndroid Build Coastguard Worker 			printf("Argv_parse returned %d!\n", ret);
155*6a54128fSAndroid Build Coastguard Worker 			continue;
156*6a54128fSAndroid Build Coastguard Worker 		}
157*6a54128fSAndroid Build Coastguard Worker 		printf("Argv_parse returned %d arguments...\n", ac);
158*6a54128fSAndroid Build Coastguard Worker 		for (cpp = av; *cpp; cpp++) {
159*6a54128fSAndroid Build Coastguard Worker 			if (cpp != av)
160*6a54128fSAndroid Build Coastguard Worker 				printf(", ");
161*6a54128fSAndroid Build Coastguard Worker 			printf("'%s'", *cpp);
162*6a54128fSAndroid Build Coastguard Worker 		}
163*6a54128fSAndroid Build Coastguard Worker 		printf("\n");
164*6a54128fSAndroid Build Coastguard Worker 		argv_free(av);
165*6a54128fSAndroid Build Coastguard Worker 	}
166*6a54128fSAndroid Build Coastguard Worker 	exit(0);
167*6a54128fSAndroid Build Coastguard Worker }
168*6a54128fSAndroid Build Coastguard Worker #endif /* DEBUG */
169