167 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * argv_parse.c --- utility function for parsing a string into a
 | |
|  * 	argc, argv array.
 | |
|  *
 | |
|  * This file defines a function argv_parse() which parsing a
 | |
|  * passed-in string, handling double quotes and backslashes, and
 | |
|  * creates an allocated argv vector which can be freed using the
 | |
|  * argv_free() function.
 | |
|  *
 | |
|  * See argv_parse.h for the formal definition of the functions.
 | |
|  *
 | |
|  * Copyright 1999 by Theodore Ts'o.
 | |
|  *
 | |
|  * Permission to use, copy, modify, and distribute this software for
 | |
|  * any purpose with or without fee is hereby granted, provided that
 | |
|  * the above copyright notice and this permission notice appear in all
 | |
|  * copies.  THE SOFTWARE IS PROVIDED "AS IS" AND THEODORE TS'O (THE
 | |
|  * AUTHOR) DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 | |
|  * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS.
 | |
|  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
 | |
|  * INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
 | |
|  * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
 | |
|  * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR
 | |
|  * IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.  (Isn't
 | |
|  * it sick that the U.S. culture of lawsuit-happy lawyers requires
 | |
|  * this kind of disclaimer?)
 | |
|  *
 | |
|  * Version 1.1, modified 2/27/1999
 | |
|  */
 | |
| 
 | |
| #include "config.h"
 | |
| #ifdef HAVE_STDLIB_H
 | |
| #include <stdlib.h>
 | |
| #endif
 | |
| #include <ctype.h>
 | |
| #include <string.h>
 | |
| #include "argv_parse.h"
 | |
| 
 | |
| #define STATE_WHITESPACE	1
 | |
| #define STATE_TOKEN		2
 | |
| #define STATE_QUOTED		3
 | |
| 
 | |
| /*
 | |
|  * Returns 0 on success, -1 on failure.
 | |
|  */
 | |
| int argv_parse(char *in_buf, int *ret_argc, char ***ret_argv)
 | |
| {
 | |
| 	int	argc = 0, max_argc = 0;
 | |
| 	char 	**argv, **new_argv, *buf, ch;
 | |
| 	char	*cp = 0, *outcp = 0;
 | |
| 	int	state = STATE_WHITESPACE;
 | |
| 
 | |
| 	buf = malloc(strlen(in_buf)+1);
 | |
| 	if (!buf)
 | |
| 		return -1;
 | |
| 
 | |
| 	max_argc = 0; argc = 0; argv = 0;
 | |
| 	outcp = buf;
 | |
| 	for (cp = in_buf; (ch = *cp); cp++) {
 | |
| 		if (state == STATE_WHITESPACE) {
 | |
| 			if (isspace((int) ch))
 | |
| 				continue;
 | |
| 			/* Not whitespace, so start a new token */
 | |
| 			state = STATE_TOKEN;
 | |
| 			if (argc >= max_argc) {
 | |
| 				max_argc += 3;
 | |
| 				new_argv = realloc(argv,
 | |
| 						  (max_argc+1)*sizeof(char *));
 | |
| 				if (!new_argv) {
 | |
| 					free(argv);
 | |
| 					free(buf);
 | |
| 					return -1;
 | |
| 				}
 | |
| 				argv = new_argv;
 | |
| 			}
 | |
| 			argv[argc++] = outcp;
 | |
| 		}
 | |
| 		if (state == STATE_QUOTED) {
 | |
| 			if (ch == '"')
 | |
| 				state = STATE_TOKEN;
 | |
| 			else
 | |
| 				*outcp++ = ch;
 | |
| 			continue;
 | |
| 		}
 | |
| 		/* Must be processing characters in a word */
 | |
| 		if (isspace((int) ch)) {
 | |
| 			/*
 | |
| 			 * Terminate the current word and start
 | |
| 			 * looking for the beginning of the next word.
 | |
| 			 */
 | |
| 			*outcp++ = 0;
 | |
| 			state = STATE_WHITESPACE;
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (ch == '"') {
 | |
| 			state = STATE_QUOTED;
 | |
| 			continue;
 | |
| 		}
 | |
| 		if (ch == '\\') {
 | |
| 			ch = *++cp;
 | |
| 			switch (ch) {
 | |
| 			case '\0':
 | |
| 				ch = '\\'; cp--; break;
 | |
| 			case 'n':
 | |
| 				ch = '\n'; break;
 | |
| 			case 't':
 | |
| 				ch = '\t'; break;
 | |
| 			case 'b':
 | |
| 				ch = '\b'; break;
 | |
| 			}
 | |
| 		}
 | |
| 		*outcp++ = ch;
 | |
| 	}
 | |
| 	if (state != STATE_WHITESPACE)
 | |
| 		*outcp++ = '\0';
 | |
| 	if (argv == 0) {
 | |
| 		argv = malloc(sizeof(char *));
 | |
| 		free(buf);
 | |
| 	}
 | |
| 	argv[argc] = 0;
 | |
| 	if (ret_argc)
 | |
| 		*ret_argc = argc;
 | |
| 	if (ret_argv)
 | |
| 		*ret_argv = argv;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void argv_free(char **argv)
 | |
| {
 | |
| 	free(*argv);
 | |
| 	free(argv);
 | |
| }
 | |
| 
 | |
| #ifdef DEBUG
 | |
| /*
 | |
|  * For debugging
 | |
|  */
 | |
| 
 | |
| #include <stdio.h>
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
| 	int	ac, ret;
 | |
| 	char	**av, **cpp;
 | |
| 	char	buf[256];
 | |
| 
 | |
| 	while (!feof(stdin)) {
 | |
| 		if (fgets(buf, sizeof(buf), stdin) == NULL)
 | |
| 			break;
 | |
| 		ret = argv_parse(buf, &ac, &av);
 | |
| 		if (ret != 0) {
 | |
| 			printf("Argv_parse returned %d!\n", ret);
 | |
| 			continue;
 | |
| 		}
 | |
| 		printf("Argv_parse returned %d arguments...\n", ac);
 | |
| 		for (cpp = av; *cpp; cpp++) {
 | |
| 			if (cpp != av)
 | |
| 				printf(", ");
 | |
| 			printf("'%s'", *cpp);
 | |
| 		}
 | |
| 		printf("\n");
 | |
| 		argv_free(av);
 | |
| 	}
 | |
| 	exit(0);
 | |
| }
 | |
| #endif /* DEBUG */
 |