1*b725ae77Skettenis /* MI Command Set - MI parser.
2*b725ae77Skettenis
3*b725ae77Skettenis Copyright 2000, 2001, 2002 Free Software Foundation, Inc.
4*b725ae77Skettenis
5*b725ae77Skettenis Contributed by Cygnus Solutions (a Red Hat company).
6*b725ae77Skettenis
7*b725ae77Skettenis This file is part of GDB.
8*b725ae77Skettenis
9*b725ae77Skettenis This program is free software; you can redistribute it and/or modify
10*b725ae77Skettenis it under the terms of the GNU General Public License as published by
11*b725ae77Skettenis the Free Software Foundation; either version 2 of the License, or
12*b725ae77Skettenis (at your option) any later version.
13*b725ae77Skettenis
14*b725ae77Skettenis This program is distributed in the hope that it will be useful,
15*b725ae77Skettenis but WITHOUT ANY WARRANTY; without even the implied warranty of
16*b725ae77Skettenis MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17*b725ae77Skettenis GNU General Public License for more details.
18*b725ae77Skettenis
19*b725ae77Skettenis You should have received a copy of the GNU General Public License
20*b725ae77Skettenis along with this program; if not, write to the Free Software
21*b725ae77Skettenis Foundation, Inc., 59 Temple Place - Suite 330,
22*b725ae77Skettenis Boston, MA 02111-1307, USA. */
23*b725ae77Skettenis
24*b725ae77Skettenis #include "defs.h"
25*b725ae77Skettenis #include "mi-cmds.h"
26*b725ae77Skettenis #include "mi-parse.h"
27*b725ae77Skettenis
28*b725ae77Skettenis #include <ctype.h>
29*b725ae77Skettenis #include "gdb_string.h"
30*b725ae77Skettenis
31*b725ae77Skettenis static void
mi_parse_argv(char * args,struct mi_parse * parse)32*b725ae77Skettenis mi_parse_argv (char *args, struct mi_parse *parse)
33*b725ae77Skettenis {
34*b725ae77Skettenis char *chp = args;
35*b725ae77Skettenis int argc = 0;
36*b725ae77Skettenis char **argv = xmalloc ((argc + 1) * sizeof (char *));
37*b725ae77Skettenis argv[argc] = NULL;
38*b725ae77Skettenis while (1)
39*b725ae77Skettenis {
40*b725ae77Skettenis char *arg;
41*b725ae77Skettenis /* skip leading white space */
42*b725ae77Skettenis while (isspace (*chp))
43*b725ae77Skettenis chp++;
44*b725ae77Skettenis /* Three possibilities: EOF, quoted string, or other text. */
45*b725ae77Skettenis switch (*chp)
46*b725ae77Skettenis {
47*b725ae77Skettenis case '\0':
48*b725ae77Skettenis parse->argv = argv;
49*b725ae77Skettenis parse->argc = argc;
50*b725ae77Skettenis return;
51*b725ae77Skettenis case '"':
52*b725ae77Skettenis {
53*b725ae77Skettenis /* A quoted string. */
54*b725ae77Skettenis int len;
55*b725ae77Skettenis char *start = chp + 1;
56*b725ae77Skettenis /* Determine the buffer size. */
57*b725ae77Skettenis chp = start;
58*b725ae77Skettenis len = 0;
59*b725ae77Skettenis while (*chp != '\0' && *chp != '"')
60*b725ae77Skettenis {
61*b725ae77Skettenis if (*chp == '\\')
62*b725ae77Skettenis {
63*b725ae77Skettenis chp++;
64*b725ae77Skettenis if (parse_escape (&chp) <= 0)
65*b725ae77Skettenis {
66*b725ae77Skettenis /* Do not allow split lines or "\000" */
67*b725ae77Skettenis freeargv (argv);
68*b725ae77Skettenis return;
69*b725ae77Skettenis }
70*b725ae77Skettenis }
71*b725ae77Skettenis else
72*b725ae77Skettenis chp++;
73*b725ae77Skettenis len++;
74*b725ae77Skettenis }
75*b725ae77Skettenis /* Insist on a closing quote. */
76*b725ae77Skettenis if (*chp != '"')
77*b725ae77Skettenis {
78*b725ae77Skettenis freeargv (argv);
79*b725ae77Skettenis return;
80*b725ae77Skettenis }
81*b725ae77Skettenis /* Insist on trailing white space. */
82*b725ae77Skettenis if (chp[1] != '\0' && !isspace (chp[1]))
83*b725ae77Skettenis {
84*b725ae77Skettenis freeargv (argv);
85*b725ae77Skettenis return;
86*b725ae77Skettenis }
87*b725ae77Skettenis /* create the buffer. */
88*b725ae77Skettenis arg = xmalloc ((len + 1) * sizeof (char));
89*b725ae77Skettenis /* And copy the characters in. */
90*b725ae77Skettenis chp = start;
91*b725ae77Skettenis len = 0;
92*b725ae77Skettenis while (*chp != '\0' && *chp != '"')
93*b725ae77Skettenis {
94*b725ae77Skettenis if (*chp == '\\')
95*b725ae77Skettenis {
96*b725ae77Skettenis chp++;
97*b725ae77Skettenis arg[len] = parse_escape (&chp);
98*b725ae77Skettenis }
99*b725ae77Skettenis else
100*b725ae77Skettenis arg[len] = *chp++;
101*b725ae77Skettenis len++;
102*b725ae77Skettenis }
103*b725ae77Skettenis arg[len] = '\0';
104*b725ae77Skettenis chp++; /* that closing quote. */
105*b725ae77Skettenis break;
106*b725ae77Skettenis }
107*b725ae77Skettenis default:
108*b725ae77Skettenis {
109*b725ae77Skettenis /* An unquoted string. Accumulate all non blank
110*b725ae77Skettenis characters into a buffer. */
111*b725ae77Skettenis int len;
112*b725ae77Skettenis char *start = chp;
113*b725ae77Skettenis while (*chp != '\0' && !isspace (*chp))
114*b725ae77Skettenis {
115*b725ae77Skettenis chp++;
116*b725ae77Skettenis }
117*b725ae77Skettenis len = chp - start;
118*b725ae77Skettenis arg = xmalloc ((len + 1) * sizeof (char));
119*b725ae77Skettenis strncpy (arg, start, len);
120*b725ae77Skettenis arg[len] = '\0';
121*b725ae77Skettenis break;
122*b725ae77Skettenis }
123*b725ae77Skettenis }
124*b725ae77Skettenis /* Append arg to argv. */
125*b725ae77Skettenis argv = xrealloc (argv, (argc + 2) * sizeof (char *));
126*b725ae77Skettenis argv[argc++] = arg;
127*b725ae77Skettenis argv[argc] = NULL;
128*b725ae77Skettenis }
129*b725ae77Skettenis }
130*b725ae77Skettenis
131*b725ae77Skettenis
132*b725ae77Skettenis void
mi_parse_free(struct mi_parse * parse)133*b725ae77Skettenis mi_parse_free (struct mi_parse *parse)
134*b725ae77Skettenis {
135*b725ae77Skettenis if (parse == NULL)
136*b725ae77Skettenis return;
137*b725ae77Skettenis if (parse->command != NULL)
138*b725ae77Skettenis xfree (parse->command);
139*b725ae77Skettenis if (parse->token != NULL)
140*b725ae77Skettenis xfree (parse->token);
141*b725ae77Skettenis if (parse->args != NULL)
142*b725ae77Skettenis xfree (parse->args);
143*b725ae77Skettenis if (parse->argv != NULL)
144*b725ae77Skettenis freeargv (parse->argv);
145*b725ae77Skettenis xfree (parse);
146*b725ae77Skettenis }
147*b725ae77Skettenis
148*b725ae77Skettenis
149*b725ae77Skettenis struct mi_parse *
mi_parse(char * cmd)150*b725ae77Skettenis mi_parse (char *cmd)
151*b725ae77Skettenis {
152*b725ae77Skettenis char *chp;
153*b725ae77Skettenis struct mi_parse *parse = XMALLOC (struct mi_parse);
154*b725ae77Skettenis memset (parse, 0, sizeof (*parse));
155*b725ae77Skettenis
156*b725ae77Skettenis /* Before starting, skip leading white space. */
157*b725ae77Skettenis while (isspace (*cmd))
158*b725ae77Skettenis cmd++;
159*b725ae77Skettenis
160*b725ae77Skettenis /* Find/skip any token and then extract it. */
161*b725ae77Skettenis for (chp = cmd; *chp >= '0' && *chp <= '9'; chp++)
162*b725ae77Skettenis ;
163*b725ae77Skettenis parse->token = xmalloc ((chp - cmd + 1) * sizeof (char *));
164*b725ae77Skettenis memcpy (parse->token, cmd, (chp - cmd));
165*b725ae77Skettenis parse->token[chp - cmd] = '\0';
166*b725ae77Skettenis
167*b725ae77Skettenis /* This wasn't a real MI command. Return it as a CLI_COMMAND. */
168*b725ae77Skettenis if (*chp != '-')
169*b725ae77Skettenis {
170*b725ae77Skettenis while (isspace (*chp))
171*b725ae77Skettenis chp++;
172*b725ae77Skettenis parse->command = xstrdup (chp);
173*b725ae77Skettenis parse->op = CLI_COMMAND;
174*b725ae77Skettenis return parse;
175*b725ae77Skettenis }
176*b725ae77Skettenis
177*b725ae77Skettenis /* Extract the command. */
178*b725ae77Skettenis {
179*b725ae77Skettenis char *tmp = chp + 1; /* discard ``-'' */
180*b725ae77Skettenis for (; *chp && !isspace (*chp); chp++)
181*b725ae77Skettenis ;
182*b725ae77Skettenis parse->command = xmalloc ((chp - tmp + 1) * sizeof (char *));
183*b725ae77Skettenis memcpy (parse->command, tmp, chp - tmp);
184*b725ae77Skettenis parse->command[chp - tmp] = '\0';
185*b725ae77Skettenis }
186*b725ae77Skettenis
187*b725ae77Skettenis /* Find the command in the MI table. */
188*b725ae77Skettenis parse->cmd = mi_lookup (parse->command);
189*b725ae77Skettenis if (parse->cmd == NULL)
190*b725ae77Skettenis {
191*b725ae77Skettenis /* FIXME: This should be a function call. */
192*b725ae77Skettenis fprintf_unfiltered
193*b725ae77Skettenis (raw_stdout,
194*b725ae77Skettenis "%s^error,msg=\"Undefined MI command: %s\"\n",
195*b725ae77Skettenis parse->token, parse->command);
196*b725ae77Skettenis mi_parse_free (parse);
197*b725ae77Skettenis return NULL;
198*b725ae77Skettenis }
199*b725ae77Skettenis
200*b725ae77Skettenis /* Skip white space following the command. */
201*b725ae77Skettenis while (isspace (*chp))
202*b725ae77Skettenis chp++;
203*b725ae77Skettenis
204*b725ae77Skettenis /* For new argv commands, attempt to return the parsed argument
205*b725ae77Skettenis list. */
206*b725ae77Skettenis if (parse->cmd->argv_func != NULL)
207*b725ae77Skettenis {
208*b725ae77Skettenis mi_parse_argv (chp, parse);
209*b725ae77Skettenis if (parse->argv == NULL)
210*b725ae77Skettenis {
211*b725ae77Skettenis /* FIXME: This should be a function call. */
212*b725ae77Skettenis fprintf_unfiltered
213*b725ae77Skettenis (raw_stdout,
214*b725ae77Skettenis "%s^error,msg=\"Problem parsing arguments: %s %s\"\n",
215*b725ae77Skettenis parse->token, parse->command, chp);
216*b725ae77Skettenis mi_parse_free (parse);
217*b725ae77Skettenis return NULL;
218*b725ae77Skettenis }
219*b725ae77Skettenis }
220*b725ae77Skettenis
221*b725ae77Skettenis /* FIXME: DELETE THIS */
222*b725ae77Skettenis /* For CLI and old ARGS commands, also return the remainder of the
223*b725ae77Skettenis command line as a single string. */
224*b725ae77Skettenis if (parse->cmd->args_func != NULL
225*b725ae77Skettenis || parse->cmd->cli.cmd != NULL)
226*b725ae77Skettenis {
227*b725ae77Skettenis parse->args = xstrdup (chp);
228*b725ae77Skettenis }
229*b725ae77Skettenis
230*b725ae77Skettenis /* Fully parsed. */
231*b725ae77Skettenis parse->op = MI_COMMAND;
232*b725ae77Skettenis return parse;
233*b725ae77Skettenis }
234