xref: /openbsd-src/gnu/usr.bin/binutils/gdb/mi/mi-parse.c (revision b725ae7711052a2233e31a66fefb8a752c388d7a)
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