1
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
32
33
34
35
36
37
38
39
40
41
42
102
103
104
105
108
109
110
111
112
/* ... */
#include <stdio.h>
#include <ctype.h>
#include <string.h>
#define SS_FLAG_ESCAPE 0x8
typedef enum {
SS_SPACE = 0x0,
SS_ARG = 0x1,
SS_QUOTED_ARG = 0x2,
SS_ARG_ESCAPED = SS_ARG | SS_FLAG_ESCAPE,
SS_QUOTED_ARG_ESCAPED = SS_QUOTED_ARG | SS_FLAG_ESCAPE,
}{ ... } split_state_t;
#define END_ARG() do { \
char_out = 0; \
argv[argc++] = next_arg_start; \
state = SS_SPACE; \
}{...} while(0)...
size_t esp_console_split_argv(char *line, char **argv, size_t argv_size)
{
const int QUOTE = '"';
const int ESCAPE = '\\';
const int SPACE = ' ';
split_state_t state = SS_SPACE;
size_t argc = 0;
char *next_arg_start = line;
char *out_ptr = line;
for (char *in_ptr = line; argc < argv_size - 1; ++in_ptr) {
int char_in = (unsigned char) * in_ptr;
if (char_in == 0) {
break;
}{...}
int char_out = -1;
switch (state) {
case SS_SPACE:
if (char_in == SPACE) {
}{...} else if (char_in == QUOTE) {
next_arg_start = out_ptr;
state = SS_QUOTED_ARG;
}{...} else if (char_in == ESCAPE) {
next_arg_start = out_ptr;
state = SS_ARG_ESCAPED;
}{...} else {
next_arg_start = out_ptr;
state = SS_ARG;
char_out = char_in;
}{...}
break;
...
case SS_QUOTED_ARG:
if (char_in == QUOTE) {
END_ARG();
}{...} else if (char_in == ESCAPE) {
state = SS_QUOTED_ARG_ESCAPED;
}{...} else {
char_out = char_in;
}{...}
break;
...
case SS_ARG_ESCAPED:
case SS_QUOTED_ARG_ESCAPED:
if (char_in == ESCAPE || char_in == QUOTE || char_in == SPACE) {
char_out = char_in;
}{...} else {
}{...}
state = (split_state_t)(state & (~SS_FLAG_ESCAPE));
break;
...
case SS_ARG:
if (char_in == SPACE) {
END_ARG();
}{...} else if (char_in == ESCAPE) {
state = SS_ARG_ESCAPED;
}{...} else {
char_out = char_in;
}{...}
break;...
}{...}
if (char_out >= 0) {
*out_ptr = char_out;
++out_ptr;
}{...}
}{...}
*out_ptr = 0;
if (state != SS_SPACE && argc < argv_size - 1) {
argv[argc++] = next_arg_start;
}{...}
argv[argc] = NULL;
return argc;
}{ ... }