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