2000-02-23 00:25:43 +00:00
|
|
|
/* MI Command Set - MI parser.
|
2002-03-19 02:51:09 +00:00
|
|
|
|
2011-01-01 15:34:07 +00:00
|
|
|
Copyright (C) 2000, 2001, 2002, 2007, 2008, 2009, 2010, 2011
|
2009-01-03 05:58:08 +00:00
|
|
|
Free Software Foundation, Inc.
|
2002-03-19 02:51:09 +00:00
|
|
|
|
2000-02-23 06:19:22 +00:00
|
|
|
Contributed by Cygnus Solutions (a Red Hat company).
|
2000-02-23 00:25:43 +00:00
|
|
|
|
|
|
|
This file is part of GDB.
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
2007-08-23 18:08:50 +00:00
|
|
|
the Free Software Foundation; either version 3 of the License, or
|
2000-02-23 00:25:43 +00:00
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2007-08-23 18:08:50 +00:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
2000-02-23 00:25:43 +00:00
|
|
|
|
|
|
|
#include "defs.h"
|
|
|
|
#include "mi-cmds.h"
|
|
|
|
#include "mi-parse.h"
|
2010-03-05 20:18:19 +00:00
|
|
|
#include "charset.h"
|
2000-02-23 00:25:43 +00:00
|
|
|
|
|
|
|
#include <ctype.h>
|
2001-01-31 15:46:36 +00:00
|
|
|
#include "gdb_string.h"
|
2000-02-23 00:25:43 +00:00
|
|
|
|
2010-03-05 20:18:19 +00:00
|
|
|
/* Like parse_escape, but leave the results as a host char, not a
|
|
|
|
target char. */
|
|
|
|
|
|
|
|
static int
|
|
|
|
mi_parse_escape (char **string_ptr)
|
|
|
|
{
|
|
|
|
int c = *(*string_ptr)++;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2010-03-05 20:18:19 +00:00
|
|
|
switch (c)
|
|
|
|
{
|
|
|
|
case '\n':
|
|
|
|
return -2;
|
|
|
|
case 0:
|
|
|
|
(*string_ptr)--;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case '0':
|
|
|
|
case '1':
|
|
|
|
case '2':
|
|
|
|
case '3':
|
|
|
|
case '4':
|
|
|
|
case '5':
|
|
|
|
case '6':
|
|
|
|
case '7':
|
|
|
|
{
|
|
|
|
int i = host_hex_value (c);
|
|
|
|
int count = 0;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2010-03-05 20:18:19 +00:00
|
|
|
while (++count < 3)
|
|
|
|
{
|
|
|
|
c = (**string_ptr);
|
|
|
|
if (isdigit (c) && c != '8' && c != '9')
|
|
|
|
{
|
|
|
|
(*string_ptr)++;
|
|
|
|
i *= 8;
|
|
|
|
i += host_hex_value (c);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'a':
|
|
|
|
c = '\a';
|
|
|
|
break;
|
|
|
|
case 'b':
|
|
|
|
c = '\b';
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
c = '\f';
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
c = '\n';
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
c = '\r';
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
c = '\t';
|
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
c = '\v';
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
static void
|
|
|
|
mi_parse_argv (char *args, struct mi_parse *parse)
|
|
|
|
{
|
|
|
|
char *chp = args;
|
|
|
|
int argc = 0;
|
|
|
|
char **argv = xmalloc ((argc + 1) * sizeof (char *));
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
argv[argc] = NULL;
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
char *arg;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
/* skip leading white space */
|
|
|
|
while (isspace (*chp))
|
|
|
|
chp++;
|
|
|
|
/* Three possibilities: EOF, quoted string, or other text. */
|
|
|
|
switch (*chp)
|
|
|
|
{
|
|
|
|
case '\0':
|
|
|
|
parse->argv = argv;
|
|
|
|
parse->argc = argc;
|
|
|
|
return;
|
|
|
|
case '"':
|
|
|
|
{
|
|
|
|
/* A quoted string. */
|
|
|
|
int len;
|
|
|
|
char *start = chp + 1;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
/* Determine the buffer size. */
|
|
|
|
chp = start;
|
|
|
|
len = 0;
|
|
|
|
while (*chp != '\0' && *chp != '"')
|
|
|
|
{
|
|
|
|
if (*chp == '\\')
|
|
|
|
{
|
|
|
|
chp++;
|
2010-03-05 20:18:19 +00:00
|
|
|
if (mi_parse_escape (&chp) <= 0)
|
2000-02-23 00:25:43 +00:00
|
|
|
{
|
|
|
|
/* Do not allow split lines or "\000" */
|
|
|
|
freeargv (argv);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
chp++;
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
/* Insist on a closing quote. */
|
|
|
|
if (*chp != '"')
|
|
|
|
{
|
|
|
|
freeargv (argv);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* Insist on trailing white space. */
|
|
|
|
if (chp[1] != '\0' && !isspace (chp[1]))
|
|
|
|
{
|
|
|
|
freeargv (argv);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
/* create the buffer. */
|
|
|
|
arg = xmalloc ((len + 1) * sizeof (char));
|
|
|
|
/* And copy the characters in. */
|
|
|
|
chp = start;
|
|
|
|
len = 0;
|
|
|
|
while (*chp != '\0' && *chp != '"')
|
|
|
|
{
|
|
|
|
if (*chp == '\\')
|
|
|
|
{
|
|
|
|
chp++;
|
2010-03-05 20:18:19 +00:00
|
|
|
arg[len] = mi_parse_escape (&chp);
|
2000-02-23 00:25:43 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
arg[len] = *chp++;
|
|
|
|
len++;
|
|
|
|
}
|
|
|
|
arg[len] = '\0';
|
|
|
|
chp++; /* that closing quote. */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
/* An unquoted string. Accumulate all non blank
|
|
|
|
characters into a buffer. */
|
|
|
|
int len;
|
|
|
|
char *start = chp;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
while (*chp != '\0' && !isspace (*chp))
|
|
|
|
{
|
|
|
|
chp++;
|
|
|
|
}
|
|
|
|
len = chp - start;
|
|
|
|
arg = xmalloc ((len + 1) * sizeof (char));
|
|
|
|
strncpy (arg, start, len);
|
|
|
|
arg[len] = '\0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Append arg to argv. */
|
|
|
|
argv = xrealloc (argv, (argc + 2) * sizeof (char *));
|
|
|
|
argv[argc++] = arg;
|
|
|
|
argv[argc] = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
mi_parse_free (struct mi_parse *parse)
|
|
|
|
{
|
|
|
|
if (parse == NULL)
|
|
|
|
return;
|
|
|
|
if (parse->command != NULL)
|
2000-12-15 01:01:51 +00:00
|
|
|
xfree (parse->command);
|
2000-02-23 00:25:43 +00:00
|
|
|
if (parse->token != NULL)
|
2000-12-15 01:01:51 +00:00
|
|
|
xfree (parse->token);
|
2000-02-23 00:25:43 +00:00
|
|
|
if (parse->args != NULL)
|
2000-12-15 01:01:51 +00:00
|
|
|
xfree (parse->args);
|
2000-02-23 00:25:43 +00:00
|
|
|
if (parse->argv != NULL)
|
|
|
|
freeargv (parse->argv);
|
2000-12-15 01:01:51 +00:00
|
|
|
xfree (parse);
|
2000-02-23 00:25:43 +00:00
|
|
|
}
|
|
|
|
|
2010-12-09 19:23:49 +00:00
|
|
|
/* A cleanup that calls mi_parse_free. */
|
|
|
|
|
|
|
|
static void
|
|
|
|
mi_parse_cleanup (void *arg)
|
|
|
|
{
|
|
|
|
mi_parse_free (arg);
|
|
|
|
}
|
2000-02-23 00:25:43 +00:00
|
|
|
|
|
|
|
struct mi_parse *
|
2010-12-09 19:23:49 +00:00
|
|
|
mi_parse (char *cmd, char **token)
|
2000-02-23 00:25:43 +00:00
|
|
|
{
|
|
|
|
char *chp;
|
|
|
|
struct mi_parse *parse = XMALLOC (struct mi_parse);
|
2010-12-09 19:23:49 +00:00
|
|
|
struct cleanup *cleanup;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
memset (parse, 0, sizeof (*parse));
|
Multiexec MI
* breakpoint.c (clear_syscall_counts): Take struct inferior*.
* inferior.c (add_inferior_silent): Notify inferior_added
observer.
(delete_inferior_1): Notify inferior_removed observer.
(exit_inferior_1): Pass inferior, not pid, to observer.
(inferior_appeared): Likewise.
(add_inferior_with_spaces): New.
(add_inferior_command): Use the above.
* inferior.h (delete_inferior_1, add_inferior_with_spaces):
Declare.
* inflow.c (inflow_inferior_exit): Likewise.
* jit.c (jit_inferior_exit_hook): Likewise.
* mi/mi-cmds.c (mi_cmds): Register add-inferior and
remove-inferior.
* mi/mi-cmds.h (mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
* mi/mi-interp.c (mi_inferior_added, mi_inferior_removed): New.
(report_initial_inferior): New.
(mi_inferior_removed): Register the above. Make sure
inferior_added observer is called on the first inferior.
(mi_new_thread, mi_thread_exit): Thread group is now identified by
inferior number, not pid.
(mi_solib_loaded, mi_solib_unloaded): Report which inferiors are
affected.
* mi/mi-main.c (current_context): New.
(proceed_thread_callback): Use typed closure.
Proceed everything if pid is 0. Most implementation split into
(proceed_thread): ... this.
(run_one_inferior): New.
(mi_cmd_exec_continue, mi_cmd_exec_interrupt, mi_cmd_exec_run):
Adjust for multiexec behaviour.
(mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
(mi_cmd_execute): Handle the 'thread-group' option here.
Do some extra checks.
* mi-parse.c (mi_parse): Handle the --all and --thread-group
options.
* mi-parse.h (struct mi_parse): New fields all and thread_group.
2010-02-24 07:51:46 +00:00
|
|
|
parse->all = 0;
|
|
|
|
parse->thread_group = -1;
|
2008-07-12 16:37:57 +00:00
|
|
|
parse->thread = -1;
|
|
|
|
parse->frame = -1;
|
2000-02-23 00:25:43 +00:00
|
|
|
|
2010-12-09 19:23:49 +00:00
|
|
|
cleanup = make_cleanup (mi_parse_cleanup, parse);
|
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
/* Before starting, skip leading white space. */
|
|
|
|
while (isspace (*cmd))
|
|
|
|
cmd++;
|
|
|
|
|
|
|
|
/* Find/skip any token and then extract it. */
|
|
|
|
for (chp = cmd; *chp >= '0' && *chp <= '9'; chp++)
|
|
|
|
;
|
2011-04-29 18:44:15 +00:00
|
|
|
*token = xmalloc (chp - cmd + 1);
|
2010-12-09 19:23:49 +00:00
|
|
|
memcpy (*token, cmd, (chp - cmd));
|
|
|
|
(*token)[chp - cmd] = '\0';
|
2000-02-23 00:25:43 +00:00
|
|
|
|
|
|
|
/* This wasn't a real MI command. Return it as a CLI_COMMAND. */
|
|
|
|
if (*chp != '-')
|
|
|
|
{
|
|
|
|
while (isspace (*chp))
|
|
|
|
chp++;
|
|
|
|
parse->command = xstrdup (chp);
|
|
|
|
parse->op = CLI_COMMAND;
|
2010-12-09 19:23:49 +00:00
|
|
|
|
|
|
|
discard_cleanups (cleanup);
|
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
return parse;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extract the command. */
|
|
|
|
{
|
|
|
|
char *tmp = chp + 1; /* discard ``-'' */
|
2010-05-17 20:49:39 +00:00
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
for (; *chp && !isspace (*chp); chp++)
|
|
|
|
;
|
2011-04-29 18:44:15 +00:00
|
|
|
parse->command = xmalloc (chp - tmp + 1);
|
2000-02-23 00:25:43 +00:00
|
|
|
memcpy (parse->command, tmp, chp - tmp);
|
|
|
|
parse->command[chp - tmp] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Find the command in the MI table. */
|
|
|
|
parse->cmd = mi_lookup (parse->command);
|
|
|
|
if (parse->cmd == NULL)
|
2010-12-09 19:23:49 +00:00
|
|
|
error (_("Undefined MI command: %s"), parse->command);
|
2000-02-23 00:25:43 +00:00
|
|
|
|
|
|
|
/* Skip white space following the command. */
|
|
|
|
while (isspace (*chp))
|
|
|
|
chp++;
|
|
|
|
|
2008-07-12 16:37:57 +00:00
|
|
|
/* Parse the --thread and --frame options, if present. At present,
|
|
|
|
some important commands, like '-break-*' are implemented by forwarding
|
|
|
|
to the CLI layer directly. We want to parse --thread and --frame
|
|
|
|
here, so as not to leave those option in the string that will be passed
|
|
|
|
to CLI. */
|
|
|
|
for (;;)
|
|
|
|
{
|
2010-12-09 20:41:54 +00:00
|
|
|
const char *option;
|
Multiexec MI
* breakpoint.c (clear_syscall_counts): Take struct inferior*.
* inferior.c (add_inferior_silent): Notify inferior_added
observer.
(delete_inferior_1): Notify inferior_removed observer.
(exit_inferior_1): Pass inferior, not pid, to observer.
(inferior_appeared): Likewise.
(add_inferior_with_spaces): New.
(add_inferior_command): Use the above.
* inferior.h (delete_inferior_1, add_inferior_with_spaces):
Declare.
* inflow.c (inflow_inferior_exit): Likewise.
* jit.c (jit_inferior_exit_hook): Likewise.
* mi/mi-cmds.c (mi_cmds): Register add-inferior and
remove-inferior.
* mi/mi-cmds.h (mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
* mi/mi-interp.c (mi_inferior_added, mi_inferior_removed): New.
(report_initial_inferior): New.
(mi_inferior_removed): Register the above. Make sure
inferior_added observer is called on the first inferior.
(mi_new_thread, mi_thread_exit): Thread group is now identified by
inferior number, not pid.
(mi_solib_loaded, mi_solib_unloaded): Report which inferiors are
affected.
* mi/mi-main.c (current_context): New.
(proceed_thread_callback): Use typed closure.
Proceed everything if pid is 0. Most implementation split into
(proceed_thread): ... this.
(run_one_inferior): New.
(mi_cmd_exec_continue, mi_cmd_exec_interrupt, mi_cmd_exec_run):
Adjust for multiexec behaviour.
(mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
(mi_cmd_execute): Handle the 'thread-group' option here.
Do some extra checks.
* mi-parse.c (mi_parse): Handle the --all and --thread-group
options.
* mi-parse.h (struct mi_parse): New fields all and thread_group.
2010-02-24 07:51:46 +00:00
|
|
|
size_t as = sizeof ("--all ") - 1;
|
|
|
|
size_t tgs = sizeof ("--thread-group ") - 1;
|
2008-07-12 16:37:57 +00:00
|
|
|
size_t ts = sizeof ("--thread ") - 1;
|
|
|
|
size_t fs = sizeof ("--frame ") - 1;
|
2010-05-17 20:49:39 +00:00
|
|
|
|
Multiexec MI
* breakpoint.c (clear_syscall_counts): Take struct inferior*.
* inferior.c (add_inferior_silent): Notify inferior_added
observer.
(delete_inferior_1): Notify inferior_removed observer.
(exit_inferior_1): Pass inferior, not pid, to observer.
(inferior_appeared): Likewise.
(add_inferior_with_spaces): New.
(add_inferior_command): Use the above.
* inferior.h (delete_inferior_1, add_inferior_with_spaces):
Declare.
* inflow.c (inflow_inferior_exit): Likewise.
* jit.c (jit_inferior_exit_hook): Likewise.
* mi/mi-cmds.c (mi_cmds): Register add-inferior and
remove-inferior.
* mi/mi-cmds.h (mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
* mi/mi-interp.c (mi_inferior_added, mi_inferior_removed): New.
(report_initial_inferior): New.
(mi_inferior_removed): Register the above. Make sure
inferior_added observer is called on the first inferior.
(mi_new_thread, mi_thread_exit): Thread group is now identified by
inferior number, not pid.
(mi_solib_loaded, mi_solib_unloaded): Report which inferiors are
affected.
* mi/mi-main.c (current_context): New.
(proceed_thread_callback): Use typed closure.
Proceed everything if pid is 0. Most implementation split into
(proceed_thread): ... this.
(run_one_inferior): New.
(mi_cmd_exec_continue, mi_cmd_exec_interrupt, mi_cmd_exec_run):
Adjust for multiexec behaviour.
(mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
(mi_cmd_execute): Handle the 'thread-group' option here.
Do some extra checks.
* mi-parse.c (mi_parse): Handle the --all and --thread-group
options.
* mi-parse.h (struct mi_parse): New fields all and thread_group.
2010-02-24 07:51:46 +00:00
|
|
|
if (strncmp (chp, "--all ", as) == 0)
|
|
|
|
{
|
|
|
|
parse->all = 1;
|
|
|
|
chp += as;
|
|
|
|
}
|
|
|
|
/* See if --all is the last token in the input. */
|
|
|
|
if (strcmp (chp, "--all") == 0)
|
|
|
|
{
|
|
|
|
parse->all = 1;
|
|
|
|
chp += strlen (chp);
|
|
|
|
}
|
|
|
|
if (strncmp (chp, "--thread-group ", tgs) == 0)
|
|
|
|
{
|
2010-12-09 20:41:54 +00:00
|
|
|
option = "--thread-group";
|
Multiexec MI
* breakpoint.c (clear_syscall_counts): Take struct inferior*.
* inferior.c (add_inferior_silent): Notify inferior_added
observer.
(delete_inferior_1): Notify inferior_removed observer.
(exit_inferior_1): Pass inferior, not pid, to observer.
(inferior_appeared): Likewise.
(add_inferior_with_spaces): New.
(add_inferior_command): Use the above.
* inferior.h (delete_inferior_1, add_inferior_with_spaces):
Declare.
* inflow.c (inflow_inferior_exit): Likewise.
* jit.c (jit_inferior_exit_hook): Likewise.
* mi/mi-cmds.c (mi_cmds): Register add-inferior and
remove-inferior.
* mi/mi-cmds.h (mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
* mi/mi-interp.c (mi_inferior_added, mi_inferior_removed): New.
(report_initial_inferior): New.
(mi_inferior_removed): Register the above. Make sure
inferior_added observer is called on the first inferior.
(mi_new_thread, mi_thread_exit): Thread group is now identified by
inferior number, not pid.
(mi_solib_loaded, mi_solib_unloaded): Report which inferiors are
affected.
* mi/mi-main.c (current_context): New.
(proceed_thread_callback): Use typed closure.
Proceed everything if pid is 0. Most implementation split into
(proceed_thread): ... this.
(run_one_inferior): New.
(mi_cmd_exec_continue, mi_cmd_exec_interrupt, mi_cmd_exec_run):
Adjust for multiexec behaviour.
(mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
(mi_cmd_execute): Handle the 'thread-group' option here.
Do some extra checks.
* mi-parse.c (mi_parse): Handle the --all and --thread-group
options.
* mi-parse.h (struct mi_parse): New fields all and thread_group.
2010-02-24 07:51:46 +00:00
|
|
|
if (parse->thread_group != -1)
|
|
|
|
error (_("Duplicate '--thread-group' option"));
|
|
|
|
chp += tgs;
|
|
|
|
if (*chp != 'i')
|
|
|
|
error (_("Invalid thread group id"));
|
|
|
|
chp += 1;
|
|
|
|
parse->thread_group = strtol (chp, &chp, 10);
|
|
|
|
}
|
2010-12-06 14:16:43 +00:00
|
|
|
else if (strncmp (chp, "--thread ", ts) == 0)
|
2008-07-12 16:37:57 +00:00
|
|
|
{
|
2010-12-09 20:41:54 +00:00
|
|
|
option = "--thread";
|
2008-07-12 16:37:57 +00:00
|
|
|
if (parse->thread != -1)
|
Multiexec MI
* breakpoint.c (clear_syscall_counts): Take struct inferior*.
* inferior.c (add_inferior_silent): Notify inferior_added
observer.
(delete_inferior_1): Notify inferior_removed observer.
(exit_inferior_1): Pass inferior, not pid, to observer.
(inferior_appeared): Likewise.
(add_inferior_with_spaces): New.
(add_inferior_command): Use the above.
* inferior.h (delete_inferior_1, add_inferior_with_spaces):
Declare.
* inflow.c (inflow_inferior_exit): Likewise.
* jit.c (jit_inferior_exit_hook): Likewise.
* mi/mi-cmds.c (mi_cmds): Register add-inferior and
remove-inferior.
* mi/mi-cmds.h (mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
* mi/mi-interp.c (mi_inferior_added, mi_inferior_removed): New.
(report_initial_inferior): New.
(mi_inferior_removed): Register the above. Make sure
inferior_added observer is called on the first inferior.
(mi_new_thread, mi_thread_exit): Thread group is now identified by
inferior number, not pid.
(mi_solib_loaded, mi_solib_unloaded): Report which inferiors are
affected.
* mi/mi-main.c (current_context): New.
(proceed_thread_callback): Use typed closure.
Proceed everything if pid is 0. Most implementation split into
(proceed_thread): ... this.
(run_one_inferior): New.
(mi_cmd_exec_continue, mi_cmd_exec_interrupt, mi_cmd_exec_run):
Adjust for multiexec behaviour.
(mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
(mi_cmd_execute): Handle the 'thread-group' option here.
Do some extra checks.
* mi-parse.c (mi_parse): Handle the --all and --thread-group
options.
* mi-parse.h (struct mi_parse): New fields all and thread_group.
2010-02-24 07:51:46 +00:00
|
|
|
error (_("Duplicate '--thread' option"));
|
2008-07-12 16:37:57 +00:00
|
|
|
chp += ts;
|
|
|
|
parse->thread = strtol (chp, &chp, 10);
|
|
|
|
}
|
|
|
|
else if (strncmp (chp, "--frame ", fs) == 0)
|
|
|
|
{
|
2010-12-09 20:41:54 +00:00
|
|
|
option = "--frame";
|
2008-07-12 16:37:57 +00:00
|
|
|
if (parse->frame != -1)
|
Multiexec MI
* breakpoint.c (clear_syscall_counts): Take struct inferior*.
* inferior.c (add_inferior_silent): Notify inferior_added
observer.
(delete_inferior_1): Notify inferior_removed observer.
(exit_inferior_1): Pass inferior, not pid, to observer.
(inferior_appeared): Likewise.
(add_inferior_with_spaces): New.
(add_inferior_command): Use the above.
* inferior.h (delete_inferior_1, add_inferior_with_spaces):
Declare.
* inflow.c (inflow_inferior_exit): Likewise.
* jit.c (jit_inferior_exit_hook): Likewise.
* mi/mi-cmds.c (mi_cmds): Register add-inferior and
remove-inferior.
* mi/mi-cmds.h (mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
* mi/mi-interp.c (mi_inferior_added, mi_inferior_removed): New.
(report_initial_inferior): New.
(mi_inferior_removed): Register the above. Make sure
inferior_added observer is called on the first inferior.
(mi_new_thread, mi_thread_exit): Thread group is now identified by
inferior number, not pid.
(mi_solib_loaded, mi_solib_unloaded): Report which inferiors are
affected.
* mi/mi-main.c (current_context): New.
(proceed_thread_callback): Use typed closure.
Proceed everything if pid is 0. Most implementation split into
(proceed_thread): ... this.
(run_one_inferior): New.
(mi_cmd_exec_continue, mi_cmd_exec_interrupt, mi_cmd_exec_run):
Adjust for multiexec behaviour.
(mi_cmd_add_inferior, mi_cmd_remove_inferior): New.
(mi_cmd_execute): Handle the 'thread-group' option here.
Do some extra checks.
* mi-parse.c (mi_parse): Handle the --all and --thread-group
options.
* mi-parse.h (struct mi_parse): New fields all and thread_group.
2010-02-24 07:51:46 +00:00
|
|
|
error (_("Duplicate '--frame' option"));
|
2008-07-12 16:37:57 +00:00
|
|
|
chp += fs;
|
|
|
|
parse->frame = strtol (chp, &chp, 10);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (*chp != '\0' && !isspace (*chp))
|
2010-12-09 20:41:54 +00:00
|
|
|
error (_("Invalid value for the '%s' option"), option);
|
2008-07-12 16:37:57 +00:00
|
|
|
while (isspace (*chp))
|
|
|
|
chp++;
|
|
|
|
}
|
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
/* For new argv commands, attempt to return the parsed argument
|
|
|
|
list. */
|
|
|
|
if (parse->cmd->argv_func != NULL)
|
|
|
|
{
|
|
|
|
mi_parse_argv (chp, parse);
|
|
|
|
if (parse->argv == NULL)
|
2010-12-09 19:23:49 +00:00
|
|
|
error (_("Problem parsing arguments: %s %s"), parse->command, chp);
|
2000-02-23 00:25:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME: DELETE THIS */
|
2008-04-24 13:28:06 +00:00
|
|
|
/* For CLI commands, also return the remainder of the
|
2000-02-23 00:25:43 +00:00
|
|
|
command line as a single string. */
|
2008-04-24 13:28:06 +00:00
|
|
|
if (parse->cmd->cli.cmd != NULL)
|
|
|
|
parse->args = xstrdup (chp);
|
2000-02-23 00:25:43 +00:00
|
|
|
|
2010-12-09 19:23:49 +00:00
|
|
|
discard_cleanups (cleanup);
|
|
|
|
|
2000-02-23 00:25:43 +00:00
|
|
|
/* Fully parsed. */
|
|
|
|
parse->op = MI_COMMAND;
|
|
|
|
return parse;
|
|
|
|
}
|