From: Simon Marchi <simon.marchi@ericsson.com>
To: Pedro Alves <palves@redhat.com>, <gdb-patches@sourceware.org>
Subject: Re: [PATCH v3 07/34] Make the intepreters output to all UIs
Date: Thu, 19 May 2016 15:16:00 -0000 [thread overview]
Message-ID: <573DD8A7.60806@ericsson.com> (raw)
In-Reply-To: <1462538104-19109-8-git-send-email-palves@redhat.com>
On 16-05-06 08:34 AM, Pedro Alves wrote:
> When we have multiple consoles, MI channels, etc., then we need to
> broadcast breakpoint hits, etc. to all UIs. In the past, I've
> adjusted most of the run control to communicate events to the
> interpreters through observer notifications, so events would be
> properly sent to console and MI streams, in sync and async modes.
>
> This patch does the next logical step -- have each interpreter's
> observers output interpreter-specific info to _all_ UIs.
>
> Note that when we have multiple instances of active cli/tui
> interpreters, then the cli_interp and tui_interp globals no longer
> work. This is addressed by this patch.
>
> Also, the interpreters currently register some observers when resumed
> and remove them when suspended. If we have multiple instances of the
> interpreters, and they can be suspended/resumed at different,
> independent times, that no longer works. What we instead do is always
> install the observers, and then have the observers themselves know
> when to do nothing.
>
> An earlier prototype of this series did the looping over struct UIs in
> common code, and then dispatched events to the interpreters through a
> matching interp_on_foo method for each observer. That turned out a
> lot more complicated than the present solution, as we'd end up with
> having to create a new interp method every time some interpreter
> wanted to listen to some observer notification, resulting in a lot of
> duplicated make-work and more coupling than desirable.
>
> gdb/ChangeLog:
> yyyy-mm-dd Pedro Alves <palves@redhat.com>
>
> * cli/cli-interp.c (cli_interp): Delete.
> (as_cli_interp): New function.
> (cli_on_normal_stop, cli_on_signal_received)
> (cli_on_end_stepping_range, cli_on_signal_exited, cli_on_exited)
> (cli_on_no_history): Send output to all CLI UIs.
> (cli_on_sync_execution_done, cli_on_command_error): Skip output if
> the top level interpreter is not a CLI.
> (cli_interpreter_init): Don't set cli_interp or install observers
> here.
> (_initialize_cli_interp): Install observers here.
> * event-top.c (main_ui_, ui_list): New globals.
> (current_ui): Point to main_ui_.
> (restore_ui_cleanup, switch_thru_all_uis_init)
> (switch_thru_all_uis_cond, switch_thru_all_uis_next): New
> functions.
> * mi/mi-interp.c (as_mi_interp): New function.
> (mi_interpreter_init): Don't install observers here.
> (mi_on_sync_execution_done): Skip output if the top level
> interpreter is not a MI.
> (mi_new_thread, mi_thread_exit, mi_record_changed)
> (mi_inferior_added, mi_inferior_appeared, mi_inferior_exit)
> (mi_inferior_removed): Send output to all MI UIs.
> (find_mi_interpreter, mi_interp_data): Delete.
> (find_mi_interp): New function.
> (mi_on_signal_received, mi_on_end_stepping_range)
> (mi_on_signal_exited, mi_on_exited, mi_on_no_history): Send output
> to all MI UIs.
> (mi_on_normal_stop): Rename to ...
> (mi_on_normal_stop_1): ... this.
> (mi_on_normal_stop): Reimplement, sending output to all MI UIs.
> (mi_traceframe_changed, mi_tsv_created, mi_tsv_deleted)
> (mi_tsv_modified, mi_breakpoint_created, mi_breakpoint_deleted)
> (mi_breakpoint_modified, mi_output_running_pid): Send output to
> all MI UIs.
> (mi_on_resume): Rename to ...
> (mi_on_resume_1): ... this. Don't handle infcalls here.
> (mi_on_resume): Reimplement, sending output to all MI UIs.
> (mi_solib_loaded, mi_solib_unloaded, mi_command_param_changed)
> (mi_memory_changed): Send output to all MI UIs.
> (report_initial_inferior): Install observers here.
> * top.h (struct ui) <next>: New field.
> (ui_list): Declare.
> (struct switch_thru_all_uis): New.
> (switch_thru_all_uis_init, switch_thru_all_uis_cond)
> (switch_thru_all_uis_next): Declare.
> (SWITCH_THRU_ALL_UIS): New macro.
> * tui/tui-interp.c (tui_interp): Delete global.
> (as_tui_interp): New function.
> (tui_on_normal_stop, tui_on_signal_received)
> (tui_on_end_stepping_range, tui_on_signal_exited, tui_on_exited)
> (tui_on_no_history): Send output to all TUI UIs.
> (tui_on_sync_execution_done, tui_on_command_error): Skip output if
> the top level interpreter is not a TUI.
> (tui_init): Don't set tui_interp or install observers here.
> (_initialize_tui_interp): Install observers here.
> ---
> gdb/cli/cli-interp.c | 127 +++++--
> gdb/event-top.c | 51 ++-
> gdb/mi/mi-interp.c | 919 ++++++++++++++++++++++++++++++++-------------------
> gdb/top.h | 29 ++
> gdb/tui/tui-interp.c | 127 +++++--
> 5 files changed, 841 insertions(+), 412 deletions(-)
>
> diff --git a/gdb/cli/cli-interp.c b/gdb/cli/cli-interp.c
> index ac12a4c..cd33dd2 100644
> --- a/gdb/cli/cli-interp.c
> +++ b/gdb/cli/cli-interp.c
> @@ -33,8 +33,16 @@ struct cli_interp
> struct ui_out *cli_uiout;
> };
>
> -/* The interpreter for the console interpreter. */
> -static struct interp *cli_interp;
> +/* Returns the INTERP's data cast as cli_interp if INTERP is a CLI,
> + and returns NULL otherwise. */
> +
> +static struct cli_interp *
> +as_cli_interp (struct interp *interp)
> +{
> + if (strcmp (interp_name (interp), INTERP_CONSOLE) == 0)
> + return (struct cli_interp *) interp_data (interp);
> + return NULL;
> +}
>
> /* Longjmp-safe wrapper for "execute_command". */
> static struct gdb_exception safe_execute_command (struct ui_out *uiout,
> @@ -50,10 +58,17 @@ static struct gdb_exception safe_execute_command (struct ui_out *uiout,
> static void
> cli_on_normal_stop (struct bpstats *bs, int print_frame)
> {
> - if (!interp_quiet_p (cli_interp))
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> {
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + continue;
> +
> if (print_frame)
> - print_stop_event (interp_ui_out (cli_interp));
> + print_stop_event (cli->cli_uiout);
> }
> }
>
> @@ -62,8 +77,17 @@ cli_on_normal_stop (struct bpstats *bs, int print_frame)
> static void
> cli_on_signal_received (enum gdb_signal siggnal)
> {
> - if (!interp_quiet_p (cli_interp))
> - print_signal_received_reason (interp_ui_out (cli_interp), siggnal);
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + continue;
> +
> + print_signal_received_reason (cli->cli_uiout, siggnal);
> + }
> }
>
> /* Observer for the end_stepping_range notification. */
> @@ -71,8 +95,17 @@ cli_on_signal_received (enum gdb_signal siggnal)
> static void
> cli_on_end_stepping_range (void)
> {
> - if (!interp_quiet_p (cli_interp))
> - print_end_stepping_range_reason (interp_ui_out (cli_interp));
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + continue;
> +
> + print_end_stepping_range_reason (cli->cli_uiout);
> + }
> }
>
> /* Observer for the signalled notification. */
> @@ -80,8 +113,17 @@ cli_on_end_stepping_range (void)
> static void
> cli_on_signal_exited (enum gdb_signal siggnal)
> {
> - if (!interp_quiet_p (cli_interp))
> - print_signal_exited_reason (interp_ui_out (cli_interp), siggnal);
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + continue;
> +
> + print_signal_exited_reason (cli->cli_uiout, siggnal);
> + }
> }
>
> /* Observer for the exited notification. */
> @@ -89,8 +131,17 @@ cli_on_signal_exited (enum gdb_signal siggnal)
> static void
> cli_on_exited (int exitstatus)
> {
> - if (!interp_quiet_p (cli_interp))
> - print_exited_reason (interp_ui_out (cli_interp), exitstatus);
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + continue;
> +
> + print_exited_reason (cli->cli_uiout, exitstatus);
> + }
> }
>
> /* Observer for the no_history notification. */
> @@ -98,8 +149,17 @@ cli_on_exited (int exitstatus)
> static void
> cli_on_no_history (void)
> {
> - if (!interp_quiet_p (cli_interp))
> - print_no_history_reason (interp_ui_out (cli_interp));
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + continue;
> +
> + print_no_history_reason (cli->cli_uiout);
> + }
> }
>
> /* Observer for the sync_execution_done notification. */
> @@ -107,8 +167,12 @@ cli_on_no_history (void)
> static void
> cli_on_sync_execution_done (void)
> {
> - if (!interp_quiet_p (cli_interp))
> - display_gdb_prompt (NULL);
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + return;
> +
> + display_gdb_prompt (NULL);
> }
>
> /* Observer for the command_error notification. */
> @@ -116,8 +180,12 @@ cli_on_sync_execution_done (void)
> static void
> cli_on_command_error (void)
> {
> - if (!interp_quiet_p (cli_interp))
> - display_gdb_prompt (NULL);
> + struct cli_interp *cli = as_cli_interp (top_level_interpreter ());
> +
> + if (cli == NULL)
> + return;
> +
> + display_gdb_prompt (NULL);
> }
>
> /* These implement the cli out interpreter: */
> @@ -125,19 +193,6 @@ cli_on_command_error (void)
> static void *
> cli_interpreter_init (struct interp *self, int top_level)
> {
> - if (top_level)
> - cli_interp = self;
> -
> - /* If changing this, remember to update tui-interp.c as well. */
> - observer_attach_normal_stop (cli_on_normal_stop);
> - observer_attach_end_stepping_range (cli_on_end_stepping_range);
> - observer_attach_signal_received (cli_on_signal_received);
> - observer_attach_signal_exited (cli_on_signal_exited);
> - observer_attach_exited (cli_on_exited);
> - observer_attach_no_history (cli_on_no_history);
> - observer_attach_sync_execution_done (cli_on_sync_execution_done);
> - observer_attach_command_error (cli_on_command_error);
> -
> return interp_data (self);
> }
>
> @@ -269,4 +324,14 @@ void
> _initialize_cli_interp (void)
> {
> interp_factory_register (INTERP_CONSOLE, cli_interp_factory);
> +
> + /* If changing this, remember to update tui-interp.c as well. */
> + observer_attach_normal_stop (cli_on_normal_stop);
> + observer_attach_end_stepping_range (cli_on_end_stepping_range);
> + observer_attach_signal_received (cli_on_signal_received);
> + observer_attach_signal_exited (cli_on_signal_exited);
> + observer_attach_exited (cli_on_exited);
> + observer_attach_no_history (cli_on_no_history);
> + observer_attach_sync_execution_done (cli_on_sync_execution_done);
> + observer_attach_command_error (cli_on_command_error);
> }
> diff --git a/gdb/event-top.c b/gdb/event-top.c
> index 664543c..c6e3b7e 100644
> --- a/gdb/event-top.c
> +++ b/gdb/event-top.c
> @@ -437,8 +437,55 @@ top_level_prompt (void)
> return xstrdup (prompt);
> }
>
> -static struct ui current_ui_;
> -struct ui *current_ui = ¤t_ui_;
> +/* The main UI. This is the UI that is bound to stdin/stdout/stderr.
> + It always exists and is created automatically when GDB starts
> + up. */
> +static struct ui main_ui_;
> +
> +struct ui *current_ui = &main_ui_;
> +struct ui *ui_list = &main_ui_;
> +
> +/* Cleanup that restores the current UI. */
> +
> +static void
> +restore_ui_cleanup (void *data)
> +{
> + current_ui = (struct ui *) data;
> +}
> +
> +/* See top.h. */
> +
> +void
> +switch_thru_all_uis_init (struct switch_thru_all_uis *state)
> +{
> + state->iter = ui_list;
> + state->old_chain = make_cleanup (restore_ui_cleanup, current_ui);
> +}
> +
> +/* See top.h. */
> +
> +int
> +switch_thru_all_uis_cond (struct switch_thru_all_uis *state)
> +{
> + if (state->iter != NULL)
> + {
> + current_ui = state->iter;
> + return 1;
> + }
> + else
> + {
> + do_cleanups (state->old_chain);
> + return 0;
> + }
> +}
> +
> +/* See top.h. */
> +
> +void
> +switch_thru_all_uis_next (struct switch_thru_all_uis *state)
> +{
> + state->iter = state->iter->next;
> +}
>
> /* Get a pointer to the current UI's line buffer. This is used to
> construct a whole line of input from partial input. */
> diff --git a/gdb/mi/mi-interp.c b/gdb/mi/mi-interp.c
> index 8ba6110..f9820cb 100644
> --- a/gdb/mi/mi-interp.c
> +++ b/gdb/mi/mi-interp.c
> @@ -88,6 +88,17 @@ static void mi_on_sync_execution_done (void);
>
> static int report_initial_inferior (struct inferior *inf, void *closure);
>
> +/* Returns the INTERP's data cast as mi_interp if INTERP is an MI, and
> + returns NULL otherwise. */
> +
> +static struct mi_interp *
> +as_mi_interp (struct interp *interp)
> +{
> + if (ui_out_is_mi_like_p (interp_ui_out (interp)))
> + return (struct mi_interp *) interp_data (interp);
> + return NULL;
> +}
> +
> static void *
> mi_interpreter_init (struct interp *interp, int top_level)
> {
> @@ -127,39 +138,8 @@ mi_interpreter_init (struct interp *interp, int top_level)
> mi->mi_uiout = mi_out_new (mi_version);
> mi->cli_uiout = cli_out_new (mi->out);
>
> - /* There are installed even if MI is not the top level interpreter.
> - The callbacks themselves decide whether to be skipped. */
> - observer_attach_signal_received (mi_on_signal_received);
> - observer_attach_end_stepping_range (mi_on_end_stepping_range);
> - observer_attach_signal_exited (mi_on_signal_exited);
> - observer_attach_exited (mi_on_exited);
> - observer_attach_no_history (mi_on_no_history);
> -
> if (top_level)
> {
> - observer_attach_new_thread (mi_new_thread);
> - observer_attach_thread_exit (mi_thread_exit);
> - observer_attach_inferior_added (mi_inferior_added);
> - observer_attach_inferior_appeared (mi_inferior_appeared);
> - observer_attach_inferior_exit (mi_inferior_exit);
> - observer_attach_inferior_removed (mi_inferior_removed);
> - observer_attach_record_changed (mi_record_changed);
> - observer_attach_normal_stop (mi_on_normal_stop);
> - observer_attach_target_resumed (mi_on_resume);
> - observer_attach_solib_loaded (mi_solib_loaded);
> - observer_attach_solib_unloaded (mi_solib_unloaded);
> - observer_attach_about_to_proceed (mi_about_to_proceed);
> - observer_attach_traceframe_changed (mi_traceframe_changed);
> - observer_attach_tsv_created (mi_tsv_created);
> - observer_attach_tsv_deleted (mi_tsv_deleted);
> - observer_attach_tsv_modified (mi_tsv_modified);
> - observer_attach_breakpoint_created (mi_breakpoint_created);
> - observer_attach_breakpoint_deleted (mi_breakpoint_deleted);
> - observer_attach_breakpoint_modified (mi_breakpoint_modified);
> - observer_attach_command_param_changed (mi_command_param_changed);
> - observer_attach_memory_changed (mi_memory_changed);
> - observer_attach_sync_execution_done (mi_on_sync_execution_done);
> -
> /* The initial inferior is created before this function is
> called, so we need to report it explicitly. Use iteration in
> case future version of GDB creates more than one inferior
> @@ -311,6 +291,12 @@ mi_execute_command_wrapper (const char *cmd)
> static void
> mi_on_sync_execution_done (void)
> {
> + struct ui *ui = current_ui;
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> +
> + if (mi == NULL)
> + return;
> +
> /* If MI is sync, then output the MI prompt now, indicating we're
> ready for further input. */
> if (!mi_async_p ())
> @@ -356,45 +342,56 @@ mi_command_loop (void *data)
> static void
> mi_new_thread (struct thread_info *t)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> struct inferior *inf = find_inferior_ptid (t->ptid);
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> gdb_assert (inf);
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "thread-created,id=\"%d\",group-id=\"i%d\"",
> - t->global_num, inf->num);
> - gdb_flush (mi->event_channel);
> + if (mi == NULL)
> + continue;
>
> - do_cleanups (old_chain);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + fprintf_unfiltered (mi->event_channel,
> + "thread-created,id=\"%d\",group-id=\"i%d\"",
> + t->global_num, inf->num);
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> static void
> mi_thread_exit (struct thread_info *t, int silent)
> {
> - struct mi_interp *mi;
> - struct inferior *inf;
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (silent)
> return;
>
> - inf = find_inferior_ptid (t->ptid);
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - mi = (struct mi_interp *) top_level_interpreter_data ();
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + if (mi == NULL)
> + continue;
>
> - fprintf_unfiltered (mi->event_channel,
> - "thread-exited,id=\"%d\",group-id=\"i%d\"",
> - t->global_num, inf->num);
> - gdb_flush (mi->event_channel);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> + fprintf_unfiltered (mi->event_channel,
> + "thread-exited,id=\"%d\",group-id=\"i%d\"",
> + t->global_num, t->inf->num);
> + gdb_flush (mi->event_channel);
>
> - do_cleanups (old_chain);
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification on changing the state of record. */
> @@ -402,122 +399,155 @@ mi_thread_exit (struct thread_info *t, int silent)
> static void
> mi_record_changed (struct inferior *inferior, int started)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel, "record-%s,thread-group=\"i%d\"",
> - started ? "started" : "stopped", inferior->num);
> + if (mi == NULL)
> + continue;
>
> - gdb_flush (mi->event_channel);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - do_cleanups (old_chain);
> + fprintf_unfiltered (mi->event_channel, "record-%s,thread-group=\"i%d\"",
> + started ? "started" : "stopped", inferior->num);
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> static void
> mi_inferior_added (struct inferior *inf)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct interp *interp;
> + struct mi_interp *mi;
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "thread-group-added,id=\"i%d\"",
> - inf->num);
> - gdb_flush (mi->event_channel);
> + /* We'll be called once for the initial inferior, before the top
> + level interpreter is set. */
> + interp = top_level_interpreter ();
> + if (interp == NULL)
> + continue;
>
> - do_cleanups (old_chain);
> + mi = as_mi_interp (interp);
> + if (mi == NULL)
> + continue;
> +
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + fprintf_unfiltered (mi->event_channel,
> + "thread-group-added,id=\"i%d\"",
> + inf->num);
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> static void
> mi_inferior_appeared (struct inferior *inf)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "thread-group-started,id=\"i%d\",pid=\"%d\"",
> - inf->num, inf->pid);
> - gdb_flush (mi->event_channel);
> + if (mi == NULL)
> + continue;
>
> - do_cleanups (old_chain);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + fprintf_unfiltered (mi->event_channel,
> + "thread-group-started,id=\"i%d\",pid=\"%d\"",
> + inf->num, inf->pid);
> + gdb_flush (mi->event_channel);
> + do_cleanups (old_chain);
> + }
> }
>
> static void
> mi_inferior_exit (struct inferior *inf)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - if (inf->has_exit_code)
> - fprintf_unfiltered (mi->event_channel,
> - "thread-group-exited,id=\"i%d\",exit-code=\"%s\"",
> - inf->num, int_string (inf->exit_code, 8, 0, 0, 1));
> - else
> - fprintf_unfiltered (mi->event_channel,
> - "thread-group-exited,id=\"i%d\"", inf->num);
> - gdb_flush (mi->event_channel);
> + if (mi == NULL)
> + continue;
>
> - do_cleanups (old_chain);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + if (inf->has_exit_code)
> + fprintf_unfiltered (mi->event_channel,
> + "thread-group-exited,id=\"i%d\",exit-code=\"%s\"",
> + inf->num, int_string (inf->exit_code, 8, 0, 0, 1));
> + else
> + fprintf_unfiltered (mi->event_channel,
> + "thread-group-exited,id=\"i%d\"", inf->num);
> +
> + gdb_flush (mi->event_channel);
> + do_cleanups (old_chain);
> + }
> }
>
> static void
> mi_inferior_removed (struct inferior *inf)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "thread-group-removed,id=\"i%d\"",
> - inf->num);
> - gdb_flush (mi->event_channel);
> + if (mi == NULL)
> + continue;
>
> - do_cleanups (old_chain);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + fprintf_unfiltered (mi->event_channel,
> + "thread-group-removed,id=\"i%d\"",
> + inf->num);
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Return the MI interpreter, if it is active -- either because it's
> the top-level interpreter or the interpreter executing the current
> command. Returns NULL if the MI interpreter is not being used. */
>
> -static struct interp *
> -find_mi_interpreter (void)
> +static struct mi_interp *
> +find_mi_interp (void)
> {
> - struct interp *interp;
> -
> - interp = top_level_interpreter ();
> - if (ui_out_is_mi_like_p (interp_ui_out (interp)))
> - return interp;
> -
> - interp = command_interp ();
> - if (ui_out_is_mi_like_p (interp_ui_out (interp)))
> - return interp;
> -
> - return NULL;
> -}
> + struct mi_interp *mi;
>
> -/* Return the MI_INTERP structure of the active MI interpreter.
> - Returns NULL if MI is not active. */
> + mi = as_mi_interp (top_level_interpreter ());
> + if (mi != NULL)
> + return mi;
>
> -static struct mi_interp *
> -mi_interp_data (void)
> -{
> - struct interp *interp = find_mi_interpreter ();
> + mi = as_mi_interp (command_interp ());
> + if (mi != NULL)
> + return mi;
>
> - if (interp != NULL)
> - return (struct mi_interp *) interp_data (interp);
> return NULL;
> }
>
> @@ -530,13 +560,18 @@ mi_interp_data (void)
> static void
> mi_on_signal_received (enum gdb_signal siggnal)
> {
> - struct mi_interp *mi = mi_interp_data ();
> + struct switch_thru_all_uis state;
>
> - if (mi == NULL)
> - return;
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = find_mi_interp ();
> +
> + if (mi == NULL)
> + continue;
>
> - print_signal_received_reason (mi->mi_uiout, siggnal);
> - print_signal_received_reason (mi->cli_uiout, siggnal);
> + print_signal_received_reason (mi->mi_uiout, siggnal);
> + print_signal_received_reason (mi->cli_uiout, siggnal);
> + }
> }
>
> /* Observer for the end_stepping_range notification. */
> @@ -544,13 +579,18 @@ mi_on_signal_received (enum gdb_signal siggnal)
> static void
> mi_on_end_stepping_range (void)
> {
> - struct mi_interp *mi = mi_interp_data ();
> + struct switch_thru_all_uis state;
>
> - if (mi == NULL)
> - return;
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = find_mi_interp ();
> +
> + if (mi == NULL)
> + continue;
>
> - print_end_stepping_range_reason (mi->mi_uiout);
> - print_end_stepping_range_reason (mi->cli_uiout);
> + print_end_stepping_range_reason (mi->mi_uiout);
> + print_end_stepping_range_reason (mi->cli_uiout);
> + }
> }
>
> /* Observer for the signal_exited notification. */
> @@ -558,13 +598,18 @@ mi_on_end_stepping_range (void)
> static void
> mi_on_signal_exited (enum gdb_signal siggnal)
> {
> - struct mi_interp *mi = mi_interp_data ();
> + struct switch_thru_all_uis state;
>
> - if (mi == NULL)
> - return;
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = find_mi_interp ();
> +
> + if (mi == NULL)
> + continue;
>
> - print_signal_exited_reason (mi->mi_uiout, siggnal);
> - print_signal_exited_reason (mi->cli_uiout, siggnal);
> + print_signal_exited_reason (mi->mi_uiout, siggnal);
> + print_signal_exited_reason (mi->cli_uiout, siggnal);
> + }
> }
>
> /* Observer for the exited notification. */
> @@ -572,13 +617,18 @@ mi_on_signal_exited (enum gdb_signal siggnal)
> static void
> mi_on_exited (int exitstatus)
> {
> - struct mi_interp *mi = mi_interp_data ();
> + struct switch_thru_all_uis state;
>
> - if (mi == NULL)
> - return;
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = find_mi_interp ();
> +
> + if (mi == NULL)
> + continue;
>
> - print_exited_reason (mi->mi_uiout, exitstatus);
> - print_exited_reason (mi->cli_uiout, exitstatus);
> + print_exited_reason (mi->mi_uiout, exitstatus);
> + print_exited_reason (mi->cli_uiout, exitstatus);
> + }
> }
>
> /* Observer for the no_history notification. */
> @@ -586,17 +636,22 @@ mi_on_exited (int exitstatus)
> static void
> mi_on_no_history (void)
> {
> - struct mi_interp *mi = mi_interp_data ();
> + struct switch_thru_all_uis state;
>
> - if (mi == NULL)
> - return;
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = find_mi_interp ();
> +
> + if (mi == NULL)
> + continue;
>
> - print_no_history_reason (mi->mi_uiout);
> - print_no_history_reason (mi->cli_uiout);
> + print_no_history_reason (mi->mi_uiout);
> + print_no_history_reason (mi->cli_uiout);
> + }
> }
>
> static void
> -mi_on_normal_stop (struct bpstats *bs, int print_frame)
> +mi_on_normal_stop_1 (struct bpstats *bs, int print_frame)
> {
> /* Since this can be called when CLI command is executing,
> using cli interpreter, be sure to use MI uiout for output,
> @@ -677,6 +732,20 @@ mi_on_normal_stop (struct bpstats *bs, int print_frame)
> }
>
> static void
> +mi_on_normal_stop (struct bpstats *bs, int print_frame)
> +{
> + struct switch_thru_all_uis state;
> +
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + if (as_mi_interp (top_level_interpreter ()) == NULL)
> + continue;
> +
> + mi_on_normal_stop_1 (bs, print_frame);
> + }
> +}
> +
> +static void
> mi_about_to_proceed (void)
> {
> /* Suppress output while calling an inferior function. */
> @@ -707,25 +776,33 @@ struct mi_suppress_notification mi_suppress_notification =
> static void
> mi_traceframe_changed (int tfnum, int tpnum)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (mi_suppress_notification.traceframe)
> return;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - if (tfnum >= 0)
> - fprintf_unfiltered (mi->event_channel, "traceframe-changed,"
> - "num=\"%d\",tracepoint=\"%d\"\n",
> - tfnum, tpnum);
> - else
> - fprintf_unfiltered (mi->event_channel, "traceframe-changed,end");
> + if (mi == NULL)
> + continue;
>
> - gdb_flush (mi->event_channel);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - do_cleanups (old_chain);
> + if (tfnum >= 0)
> + fprintf_unfiltered (mi->event_channel, "traceframe-changed,"
> + "num=\"%d\",tracepoint=\"%d\"\n",
> + tfnum, tpnum);
> + else
> + fprintf_unfiltered (mi->event_channel, "traceframe-changed,end");
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification on creating a trace state variable. */
> @@ -733,19 +810,27 @@ mi_traceframe_changed (int tfnum, int tpnum)
> static void
> mi_tsv_created (const struct trace_state_variable *tsv)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel, "tsv-created,"
> - "name=\"%s\",initial=\"%s\"\n",
> - tsv->name, plongest (tsv->initial_value));
> + if (mi == NULL)
> + continue;
>
> - gdb_flush (mi->event_channel);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - do_cleanups (old_chain);
> + fprintf_unfiltered (mi->event_channel, "tsv-created,"
> + "name=\"%s\",initial=\"%s\"\n",
> + tsv->name, plongest (tsv->initial_value));
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification on deleting a trace state variable. */
> @@ -753,21 +838,29 @@ mi_tsv_created (const struct trace_state_variable *tsv)
> static void
> mi_tsv_deleted (const struct trace_state_variable *tsv)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - if (tsv != NULL)
> - fprintf_unfiltered (mi->event_channel, "tsv-deleted,"
> - "name=\"%s\"\n", tsv->name);
> - else
> - fprintf_unfiltered (mi->event_channel, "tsv-deleted\n");
> + if (mi == NULL)
> + continue;
>
> - gdb_flush (mi->event_channel);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - do_cleanups (old_chain);
> + if (tsv != NULL)
> + fprintf_unfiltered (mi->event_channel, "tsv-deleted,"
> + "name=\"%s\"\n", tsv->name);
> + else
> + fprintf_unfiltered (mi->event_channel, "tsv-deleted\n");
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification on modifying a trace state variable. */
> @@ -775,29 +868,39 @@ mi_tsv_deleted (const struct trace_state_variable *tsv)
> static void
> mi_tsv_modified (const struct trace_state_variable *tsv)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *mi_uiout = interp_ui_out (top_level_interpreter ());
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct ui_out *mi_uiout;
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "tsv-modified");
> + if (mi == NULL)
> + continue;
>
> - ui_out_redirect (mi_uiout, mi->event_channel);
> + mi_uiout = interp_ui_out (top_level_interpreter ());
>
> - ui_out_field_string (mi_uiout, "name", tsv->name);
> - ui_out_field_string (mi_uiout, "initial",
> - plongest (tsv->initial_value));
> - if (tsv->value_known)
> - ui_out_field_string (mi_uiout, "current", plongest (tsv->value));
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - ui_out_redirect (mi_uiout, NULL);
> + fprintf_unfiltered (mi->event_channel,
> + "tsv-modified");
>
> - gdb_flush (mi->event_channel);
> + ui_out_redirect (mi_uiout, mi->event_channel);
>
> - do_cleanups (old_chain);
> + ui_out_field_string (mi_uiout, "name", tsv->name);
> + ui_out_field_string (mi_uiout, "initial",
> + plongest (tsv->initial_value));
> + if (tsv->value_known)
> + ui_out_field_string (mi_uiout, "current", plongest (tsv->value));
> +
> + ui_out_redirect (mi_uiout, NULL);
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification about a created breakpoint. */
> @@ -805,9 +908,7 @@ mi_tsv_modified (const struct trace_state_variable *tsv)
> static void
> mi_breakpoint_created (struct breakpoint *b)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *mi_uiout = interp_ui_out (top_level_interpreter ());
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (mi_suppress_notification.breakpoint)
> return;
> @@ -815,33 +916,45 @@ mi_breakpoint_created (struct breakpoint *b)
> if (b->number <= 0)
> return;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> -
> - fprintf_unfiltered (mi->event_channel,
> - "breakpoint-created");
> - /* We want the output from gdb_breakpoint_query to go to
> - mi->event_channel. One approach would be to just call
> - gdb_breakpoint_query, and then use mi_out_put to send the current
> - content of mi_outout into mi->event_channel. However, that will
> - break if anything is output to mi_uiout prior to calling the
> - breakpoint_created notifications. So, we use
> - ui_out_redirect. */
> - ui_out_redirect (mi_uiout, mi->event_channel);
> - TRY
> + SWITCH_THRU_ALL_UIS (state)
> {
> - gdb_breakpoint_query (mi_uiout, b->number, NULL);
> - }
> - CATCH (e, RETURN_MASK_ERROR)
> - {
> - }
> - END_CATCH
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct ui_out *mi_uiout;
> + struct cleanup *old_chain;
> +
> + if (mi == NULL)
> + continue;
> +
> + mi_uiout = interp_ui_out (top_level_interpreter ());
> +
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + fprintf_unfiltered (mi->event_channel,
> + "breakpoint-created");
> + /* We want the output from gdb_breakpoint_query to go to
> + mi->event_channel. One approach would be to just call
> + gdb_breakpoint_query, and then use mi_out_put to send the current
> + content of mi_outout into mi->event_channel. However, that will
> + break if anything is output to mi_uiout prior to calling the
> + breakpoint_created notifications. So, we use
> + ui_out_redirect. */
> + ui_out_redirect (mi_uiout, mi->event_channel);
> + TRY
> + {
> + gdb_breakpoint_query (mi_uiout, b->number, NULL);
> + }
> + CATCH (e, RETURN_MASK_ERROR)
> + {
> + }
> + END_CATCH
>
> - ui_out_redirect (mi_uiout, NULL);
> + ui_out_redirect (mi_uiout, NULL);
>
> - gdb_flush (mi->event_channel);
> + gdb_flush (mi->event_channel);
>
> - do_cleanups (old_chain);
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification about deleted breakpoint. */
> @@ -849,8 +962,7 @@ mi_breakpoint_created (struct breakpoint *b)
> static void
> mi_breakpoint_deleted (struct breakpoint *b)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (mi_suppress_notification.breakpoint)
> return;
> @@ -858,15 +970,24 @@ mi_breakpoint_deleted (struct breakpoint *b)
> if (b->number <= 0)
> return;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel, "breakpoint-deleted,id=\"%d\"",
> - b->number);
> + if (mi == NULL)
> + continue;
>
> - gdb_flush (mi->event_channel);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - do_cleanups (old_chain);
> + fprintf_unfiltered (mi->event_channel, "breakpoint-deleted,id=\"%d\"",
> + b->number);
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification about modified breakpoint. */
> @@ -874,9 +995,7 @@ mi_breakpoint_deleted (struct breakpoint *b)
> static void
> mi_breakpoint_modified (struct breakpoint *b)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *mi_uiout = interp_ui_out (top_level_interpreter ());
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (mi_suppress_notification.breakpoint)
> return;
> @@ -884,44 +1003,61 @@ mi_breakpoint_modified (struct breakpoint *b)
> if (b->number <= 0)
> return;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> -
> - fprintf_unfiltered (mi->event_channel,
> - "breakpoint-modified");
> - /* We want the output from gdb_breakpoint_query to go to
> - mi->event_channel. One approach would be to just call
> - gdb_breakpoint_query, and then use mi_out_put to send the current
> - content of mi_outout into mi->event_channel. However, that will
> - break if anything is output to mi_uiout prior to calling the
> - breakpoint_created notifications. So, we use
> - ui_out_redirect. */
> - ui_out_redirect (mi_uiout, mi->event_channel);
> - TRY
> + SWITCH_THRU_ALL_UIS (state)
> {
> - gdb_breakpoint_query (mi_uiout, b->number, NULL);
> - }
> - CATCH (e, RETURN_MASK_ERROR)
> - {
> - }
> - END_CATCH
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
> +
> + if (mi == NULL)
> + continue;
> +
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> + fprintf_unfiltered (mi->event_channel,
> + "breakpoint-modified");
> + /* We want the output from gdb_breakpoint_query to go to
> + mi->event_channel. One approach would be to just call
> + gdb_breakpoint_query, and then use mi_out_put to send the current
> + content of mi_outout into mi->event_channel. However, that will
> + break if anything is output to mi_uiout prior to calling the
> + breakpoint_created notifications. So, we use
> + ui_out_redirect. */
> + ui_out_redirect (mi->mi_uiout, mi->event_channel);
> + TRY
> + {
> + gdb_breakpoint_query (mi->mi_uiout, b->number, NULL);
> + }
> + CATCH (e, RETURN_MASK_ERROR)
> + {
> + }
> + END_CATCH
>
> - ui_out_redirect (mi_uiout, NULL);
> + ui_out_redirect (mi->mi_uiout, NULL);
>
> - gdb_flush (mi->event_channel);
> + gdb_flush (mi->event_channel);
>
> - do_cleanups (old_chain);
> + do_cleanups (old_chain);
> + }
> }
>
> static int
> mi_output_running_pid (struct thread_info *info, void *arg)
> {
> ptid_t *ptid = (ptid_t *) arg;
> + struct switch_thru_all_uis state;
>
> - if (ptid_get_pid (*ptid) == ptid_get_pid (info->ptid))
> - fprintf_unfiltered (raw_stdout,
> - "*running,thread-id=\"%d\"\n",
> - info->global_num);
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> +
> + if (mi == NULL)
> + continue;
> +
> + if (ptid_get_pid (*ptid) == ptid_get_pid (info->ptid))
> + fprintf_unfiltered (raw_stdout,
> + "*running,thread-id=\"%d\"\n",
> + info->global_num);
> + }
>
> return 0;
> }
> @@ -939,19 +1075,8 @@ mi_inferior_count (struct inferior *inf, void *arg)
> }
>
> static void
> -mi_on_resume (ptid_t ptid)
> +mi_on_resume_1 (ptid_t ptid)
> {
> - struct thread_info *tp = NULL;
> -
> - if (ptid_equal (ptid, minus_one_ptid) || ptid_is_pid (ptid))
> - tp = inferior_thread ();
> - else
> - tp = find_thread_ptid (ptid);
> -
> - /* Suppress output while calling an inferior function. */
> - if (tp->control.in_infcall)
> - return;
> -
> /* To cater for older frontends, emit ^running, but do it only once
> per each command. We do it here, since at this point we know
> that the target was successfully resumed, and in non-async mode,
> @@ -1006,64 +1131,116 @@ mi_on_resume (ptid_t ptid)
> }
>
> static void
> -mi_solib_loaded (struct so_list *solib)
> +mi_on_resume (ptid_t ptid)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *uiout = interp_ui_out (top_level_interpreter ());
> - struct cleanup *old_chain;
> -
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + struct thread_info *tp = NULL;
> + struct switch_thru_all_uis state;
>
> - fprintf_unfiltered (mi->event_channel, "library-loaded");
> + if (ptid_equal (ptid, minus_one_ptid) || ptid_is_pid (ptid))
> + tp = inferior_thread ();
> + else
> + tp = find_thread_ptid (ptid);
>
> - ui_out_redirect (uiout, mi->event_channel);
> + /* Suppress output while calling an inferior function. */
> + if (tp->control.in_infcall)
> + return;
>
> - ui_out_field_string (uiout, "id", solib->so_original_name);
> - ui_out_field_string (uiout, "target-name", solib->so_original_name);
> - ui_out_field_string (uiout, "host-name", solib->so_name);
> - ui_out_field_int (uiout, "symbols-loaded", solib->symbols_loaded);
> - if (!gdbarch_has_global_solist (target_gdbarch ()))
> + SWITCH_THRU_ALL_UIS (state)
> {
> - ui_out_field_fmt (uiout, "thread-group", "i%d",
> - current_inferior ()->num);
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct cleanup *old_chain;
> +
> + if (mi == NULL)
> + continue;
> +
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + mi_on_resume_1 (ptid);
> +
> + do_cleanups (old_chain);
> }
> +}
>
> - ui_out_redirect (uiout, NULL);
> +static void
> +mi_solib_loaded (struct so_list *solib)
> +{
> + struct switch_thru_all_uis state;
>
> - gdb_flush (mi->event_channel);
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct ui_out *uiout;
> + struct cleanup *old_chain;
>
> - do_cleanups (old_chain);
> + if (mi == NULL)
> + continue;
> +
> + uiout = interp_ui_out (top_level_interpreter ());
> +
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
> +
> + fprintf_unfiltered (mi->event_channel, "library-loaded");
> +
> + ui_out_redirect (uiout, mi->event_channel);
> +
> + ui_out_field_string (uiout, "id", solib->so_original_name);
> + ui_out_field_string (uiout, "target-name", solib->so_original_name);
> + ui_out_field_string (uiout, "host-name", solib->so_name);
> + ui_out_field_int (uiout, "symbols-loaded", solib->symbols_loaded);
> + if (!gdbarch_has_global_solist (target_gdbarch ()))
> + {
> + ui_out_field_fmt (uiout, "thread-group", "i%d",
> + current_inferior ()->num);
> + }
> +
> + ui_out_redirect (uiout, NULL);
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> static void
> mi_solib_unloaded (struct so_list *solib)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *uiout = interp_ui_out (top_level_interpreter ());
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct ui_out *uiout;
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel, "library-unloaded");
> + if (mi == NULL)
> + continue;
>
> - ui_out_redirect (uiout, mi->event_channel);
> + uiout = interp_ui_out (top_level_interpreter ());
>
> - ui_out_field_string (uiout, "id", solib->so_original_name);
> - ui_out_field_string (uiout, "target-name", solib->so_original_name);
> - ui_out_field_string (uiout, "host-name", solib->so_name);
> - if (!gdbarch_has_global_solist (target_gdbarch ()))
> - {
> - ui_out_field_fmt (uiout, "thread-group", "i%d",
> - current_inferior ()->num);
> - }
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - ui_out_redirect (uiout, NULL);
> + fprintf_unfiltered (mi->event_channel, "library-unloaded");
>
> - gdb_flush (mi->event_channel);
> + ui_out_redirect (uiout, mi->event_channel);
>
> - do_cleanups (old_chain);
> + ui_out_field_string (uiout, "id", solib->so_original_name);
> + ui_out_field_string (uiout, "target-name", solib->so_original_name);
> + ui_out_field_string (uiout, "host-name", solib->so_name);
> + if (!gdbarch_has_global_solist (target_gdbarch ()))
> + {
> + ui_out_field_fmt (uiout, "thread-group", "i%d",
> + current_inferior ()->num);
> + }
> +
> + ui_out_redirect (uiout, NULL);
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification about the command parameter change. */
> @@ -1071,29 +1248,38 @@ mi_solib_unloaded (struct so_list *solib)
> static void
> mi_command_param_changed (const char *param, const char *value)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *mi_uiout = interp_ui_out (top_level_interpreter ());
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (mi_suppress_notification.cmd_param_changed)
> return;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct ui_out *mi_uiout;
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "cmd-param-changed");
> + if (mi == NULL)
> + continue;
>
> - ui_out_redirect (mi_uiout, mi->event_channel);
> + mi_uiout = interp_ui_out (top_level_interpreter ());
>
> - ui_out_field_string (mi_uiout, "param", param);
> - ui_out_field_string (mi_uiout, "value", value);
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - ui_out_redirect (mi_uiout, NULL);
> + fprintf_unfiltered (mi->event_channel, "cmd-param-changed");
>
> - gdb_flush (mi->event_channel);
> + ui_out_redirect (mi_uiout, mi->event_channel);
>
> - do_cleanups (old_chain);
> + ui_out_field_string (mi_uiout, "param", param);
> + ui_out_field_string (mi_uiout, "value", value);
> +
> + ui_out_redirect (mi_uiout, NULL);
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> /* Emit notification about the target memory change. */
> @@ -1102,49 +1288,58 @@ static void
> mi_memory_changed (struct inferior *inferior, CORE_ADDR memaddr,
> ssize_t len, const bfd_byte *myaddr)
> {
> - struct mi_interp *mi = (struct mi_interp *) top_level_interpreter_data ();
> - struct ui_out *mi_uiout = interp_ui_out (top_level_interpreter ());
> - struct obj_section *sec;
> - struct cleanup *old_chain;
> + struct switch_thru_all_uis state;
>
> if (mi_suppress_notification.memory)
> return;
>
> - old_chain = make_cleanup_restore_target_terminal ();
> - target_terminal_ours_for_output ();
> + SWITCH_THRU_ALL_UIS (state)
> + {
> + struct mi_interp *mi = as_mi_interp (top_level_interpreter ());
> + struct ui_out *mi_uiout;
> + struct obj_section *sec;
> + struct cleanup *old_chain;
>
> - fprintf_unfiltered (mi->event_channel,
> - "memory-changed");
> + if (mi == NULL)
> + continue;
>
> - ui_out_redirect (mi_uiout, mi->event_channel);
> + mi_uiout = interp_ui_out (top_level_interpreter ());
>
> - ui_out_field_fmt (mi_uiout, "thread-group", "i%d", inferior->num);
> - ui_out_field_core_addr (mi_uiout, "addr", target_gdbarch (), memaddr);
> - ui_out_field_fmt (mi_uiout, "len", "%s", hex_string (len));
> + old_chain = make_cleanup_restore_target_terminal ();
> + target_terminal_ours_for_output ();
>
> - /* Append 'type=code' into notification if MEMADDR falls in the range of
> - sections contain code. */
> - sec = find_pc_section (memaddr);
> - if (sec != NULL && sec->objfile != NULL)
> - {
> - flagword flags = bfd_get_section_flags (sec->objfile->obfd,
> - sec->the_bfd_section);
> + fprintf_unfiltered (mi->event_channel, "memory-changed");
>
> - if (flags & SEC_CODE)
> - ui_out_field_string (mi_uiout, "type", "code");
> - }
> + ui_out_redirect (mi_uiout, mi->event_channel);
>
> - ui_out_redirect (mi_uiout, NULL);
> + ui_out_field_fmt (mi_uiout, "thread-group", "i%d", inferior->num);
> + ui_out_field_core_addr (mi_uiout, "addr", target_gdbarch (), memaddr);
> + ui_out_field_fmt (mi_uiout, "len", "%s", hex_string (len));
>
> - gdb_flush (mi->event_channel);
> + /* Append 'type=code' into notification if MEMADDR falls in the range of
> + sections contain code. */
> + sec = find_pc_section (memaddr);
> + if (sec != NULL && sec->objfile != NULL)
> + {
> + flagword flags = bfd_get_section_flags (sec->objfile->obfd,
> + sec->the_bfd_section);
>
> - do_cleanups (old_chain);
> + if (flags & SEC_CODE)
> + ui_out_field_string (mi_uiout, "type", "code");
> + }
> +
> + ui_out_redirect (mi_uiout, NULL);
> +
> + gdb_flush (mi->event_channel);
> +
> + do_cleanups (old_chain);
> + }
> }
>
> static int
> report_initial_inferior (struct inferior *inf, void *closure)
> {
> - /* This function is called from mi_intepreter_init, and since
> + /* This function is called from mi_interpreter_init, and since
> mi_inferior_added assumes that inferior is fully initialized
> and top_level_interpreter_data is set, we cannot call
> it here. */
> @@ -1250,4 +1445,32 @@ _initialize_mi_interp (void)
> interp_factory_register (INTERP_MI2, mi_interp_factory);
> interp_factory_register (INTERP_MI3, mi_interp_factory);
> interp_factory_register (INTERP_MI, mi_interp_factory);
> +
> + observer_attach_signal_received (mi_on_signal_received);
> + observer_attach_end_stepping_range (mi_on_end_stepping_range);
> + observer_attach_signal_exited (mi_on_signal_exited);
> + observer_attach_exited (mi_on_exited);
> + observer_attach_no_history (mi_on_no_history);
> + observer_attach_new_thread (mi_new_thread);
> + observer_attach_thread_exit (mi_thread_exit);
> + observer_attach_inferior_added (mi_inferior_added);
> + observer_attach_inferior_appeared (mi_inferior_appeared);
> + observer_attach_inferior_exit (mi_inferior_exit);
> + observer_attach_inferior_removed (mi_inferior_removed);
> + observer_attach_record_changed (mi_record_changed);
> + observer_attach_normal_stop (mi_on_normal_stop);
> + observer_attach_target_resumed (mi_on_resume);
> + observer_attach_solib_loaded (mi_solib_loaded);
> + observer_attach_solib_unloaded (mi_solib_unloaded);
> + observer_attach_about_to_proceed (mi_about_to_proceed);
> + observer_attach_traceframe_changed (mi_traceframe_changed);
> + observer_attach_tsv_created (mi_tsv_created);
> + observer_attach_tsv_deleted (mi_tsv_deleted);
> + observer_attach_tsv_modified (mi_tsv_modified);
> + observer_attach_breakpoint_created (mi_breakpoint_created);
> + observer_attach_breakpoint_deleted (mi_breakpoint_deleted);
> + observer_attach_breakpoint_modified (mi_breakpoint_modified);
> + observer_attach_command_param_changed (mi_command_param_changed);
> + observer_attach_memory_changed (mi_memory_changed);
> + observer_attach_sync_execution_done (mi_on_sync_execution_done);
> }
> diff --git a/gdb/top.h b/gdb/top.h
> index f18b79e..805022f 100644
> --- a/gdb/top.h
> +++ b/gdb/top.h
> @@ -36,6 +36,9 @@ struct tl_interp_info;
>
> struct ui
> {
> + /* Pointer to next in singly-linked list. */
> + struct ui *next;
> +
> /* The UI's command line buffer. This is to used to accumulate
> input until we have a whole command line. */
> struct buffer line_buffer;
> @@ -83,8 +86,34 @@ struct ui
> struct ui_file *m_gdb_stdlog;
> };
>
> +/* The current UI. */
> extern struct ui *current_ui;
>
> +/* The list of all UIs. */
> +extern struct ui *ui_list;
> +
> +/* State for SWITCH_THRU_ALL_UIS. Declared here because it is meant
> + to be created on the stack, but should be treated as opaque. */
> +struct switch_thru_all_uis
> +{
> + struct ui *iter;
> + struct cleanup *old_chain;
> +};
> +
> +/* Functions to drive SWITCH_THRU_ALL_UIS. Though declared here by
> + necessity, these functions should not be used other than via the
> + SWITCH_THRU_ALL_UIS macro defined below. */
> +extern void switch_thru_all_uis_init (struct switch_thru_all_uis *state);
> +extern int switch_thru_all_uis_cond (struct switch_thru_all_uis *state);
> +extern void switch_thru_all_uis_next (struct switch_thru_all_uis *state);
> +
> + /* Traverse through all UI, and switch the current UI to the one
> + being iterated. */
> +#define SWITCH_THRU_ALL_UIS(STATE) \
> + for (switch_thru_all_uis_init (&STATE); \
> + switch_thru_all_uis_cond (&STATE); \
> + switch_thru_all_uis_next (&STATE)) \
The last backslash is not necessary I think.
I was wondering why you did not name this "ALL_UIS", using the same pattern
as ALL_INFERIORS & al, but then I realized it's because this one actually
sets current_ui before each iteration (and restores it at the end).
I guess the reason you need to do this is because, again, everything relies on
accessing the global current_ui, whereas it should be passed down as a parameter.
I am not saying it should be done in this series (it's a big task in itself), but
we can probably tackle it after. They way you did things should make it easy to
improve things later on.
next prev parent reply other threads:[~2016-05-19 15:16 UTC|newest]
Thread overview: 71+ messages / expand[flat|nested] mbox.gz Atom feed top
2016-05-06 12:35 [PATCH v3 00/34] Towards great frontend GDB consoles Pedro Alves
2016-05-06 12:35 ` [PATCH v3 02/34] [Ada catchpoints] Fix "warning: failed to get exception name: No definition of \"e.full_name\" in current context" Pedro Alves
2016-05-06 12:35 ` [PATCH v3 33/34] Make mi-break.exp always expect breakpoint commands output on the main UI Pedro Alves
2016-05-06 12:35 ` [PATCH v3 20/34] Make gdb_in_secondary_prompt_p() be per UI Pedro Alves
2016-05-06 12:35 ` [PATCH v3 14/34] Make command line editing (use of readline) " Pedro Alves
2016-05-06 12:35 ` [PATCH v3 21/34] Replace the sync_execution global with a new enum prompt_state tristate Pedro Alves
2016-05-06 12:35 ` [PATCH v3 01/34] Prepare gdb.python/mi-py-events.exp for Python/MI in separate channels Pedro Alves
2016-05-06 12:35 ` [PATCH v3 16/34] Make target_terminal_inferior/ours almost nops on non-main UIs Pedro Alves
2016-05-06 12:35 ` [PATCH v3 15/34] Always process target events in the main UI Pedro Alves
2016-05-06 12:35 ` [PATCH v3 03/34] Introduce "struct ui" Pedro Alves
2016-05-06 12:35 ` [PATCH v3 29/34] Add new command to create extra console/mi UI channels Pedro Alves
2016-05-26 18:34 ` Pedro Alves
2016-05-06 12:35 ` [PATCH v3 24/34] Push thread->control.command_interp to the struct thread_fsm Pedro Alves
2016-07-01 11:02 ` Thomas Preudhomme
[not found] ` <20144b4c-11ee-fc84-e3ad-b9992f14ce15@redhat.com>
2016-07-01 15:24 ` [PATCH] Build gdb.opt/inline-*.exp tests at -O0, rely on __attribute__((always_inline)) (was: Re: [PATCH v3 24/34] Push thread->control.command_interp to the struct thread_fsm) Thomas Preudhomme
2016-07-15 12:05 ` Thomas Preudhomme
2016-07-19 17:02 ` [PATCH] Build gdb.opt/inline-*.exp tests at -O0, rely on __attribute__((always_inline)) Pedro Alves
2016-07-20 16:35 ` Thomas Preudhomme
2016-05-06 12:36 ` [PATCH v3 31/34] Add testing infrastruture bits for running with MI on a separate UI Pedro Alves
2016-06-28 20:19 ` Simon Marchi
2016-06-29 10:50 ` Pedro Alves
2016-06-30 11:12 ` [pushed] Fix gdbserver/MI testing regression (was: Re: [PATCH v3 31/34] Add testing infrastruture bits for running with MI on a separate UI) Pedro Alves
2016-06-30 12:10 ` gdbserver/ada testing broken (was: Re: [pushed] Fix gdbserver/MI testing regression) Pedro Alves
2016-07-04 20:40 ` gdbserver/ada testing broken Simon Marchi
2016-07-05 15:28 ` Joel Brobecker
2016-07-05 15:47 ` Joel Brobecker
2016-07-05 16:36 ` gdbserver/ada testing broken (was: Re: [pushed] Fix gdbserver/MI testing regression) Joel Brobecker
2016-07-05 17:19 ` gdbserver/ada testing broken Simon Marchi
2016-07-06 13:23 ` Joel Brobecker
2016-07-06 14:28 ` Simon Marchi
2016-07-19 17:11 ` Pedro Alves
2016-07-04 17:22 ` [pushed] Fix gdbserver/MI testing regression Simon Marchi
2016-05-06 12:40 ` [PATCH v3 23/34] New function should_print_stop_to_console Pedro Alves
2016-05-06 12:40 ` [PATCH v3 13/34] Make current_ui_out be per UI Pedro Alves
2016-05-06 12:40 ` [PATCH v3 11/34] Make out and error streams " Pedro Alves
2016-05-06 12:41 ` [PATCH v3 06/34] Introduce interpreter factories Pedro Alves
2016-05-18 19:18 ` Simon Marchi
2016-05-26 18:11 ` Pedro Alves
2016-05-18 19:20 ` Simon Marchi
2016-05-26 18:08 ` Pedro Alves
2016-05-06 12:42 ` [PATCH v3 30/34] [DOC] Document support for running interpreters on separate UI channels Pedro Alves
2016-05-06 13:04 ` Eli Zaretskii
2016-05-26 11:11 ` Pedro Alves
2016-06-17 17:24 ` Pedro Alves
2016-06-17 20:02 ` Eli Zaretskii
2016-05-06 12:43 ` [PATCH v3 28/34] Make stdin be per UI Pedro Alves
2016-05-06 12:43 ` [PATCH v3 08/34] Always run async signal handlers in the main UI Pedro Alves
2016-05-19 19:28 ` Simon Marchi
2016-05-26 18:13 ` Pedro Alves
2016-05-26 18:15 ` Simon Marchi
2016-05-06 12:43 ` [PATCH v3 25/34] Only send sync execution command output to the UI that ran the command Pedro Alves
2016-05-06 12:43 ` [PATCH v3 07/34] Make the intepreters output to all UIs Pedro Alves
2016-05-19 15:16 ` Simon Marchi [this message]
2016-05-26 18:12 ` Pedro Alves
2016-05-06 12:43 ` [PATCH v3 10/34] Make input_fd be per UI Pedro Alves
2016-05-06 12:43 ` [PATCH v3 05/34] Make the interpreters " Pedro Alves
2016-05-18 17:51 ` Simon Marchi
2016-05-26 18:08 ` Pedro Alves
2016-05-06 12:43 ` [PATCH v3 04/34] Make gdb_stdout&co " Pedro Alves
2016-05-06 12:43 ` [PATCH v3 17/34] Introduce display_mi_prompt Pedro Alves
2016-05-06 12:43 ` [PATCH v3 12/34] Delete def_uiout Pedro Alves
2016-05-06 12:45 ` [PATCH v3 26/34] Make main_ui be heap allocated Pedro Alves
2016-05-06 12:45 ` [PATCH v3 22/34] Fix for spurious prompts in secondary UIs Pedro Alves
2016-05-06 12:45 ` [PATCH v3 27/34] Handle UI's terminal closing Pedro Alves
2016-05-06 12:45 ` [PATCH v3 32/34] Send deleted watchpoint-scope output to all UIs Pedro Alves
2016-05-06 12:45 ` [PATCH v3 34/34] Always switch fork child to the main UI Pedro Alves
2016-05-06 12:52 ` [PATCH v3 18/34] Make raw_stdout be per MI instance Pedro Alves
2016-05-06 12:53 ` [PATCH v3 19/34] Simplify starting the command event loop Pedro Alves
2016-05-06 12:53 ` [PATCH v3 09/34] Make instream be per UI Pedro Alves
2016-05-26 18:37 ` [PATCH v3 35/34] Add "new-ui console" tests Pedro Alves
2016-06-21 0:23 ` [pushed] Re: [PATCH v3 00/34] Towards great frontend GDB consoles Pedro Alves
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=573DD8A7.60806@ericsson.com \
--to=simon.marchi@ericsson.com \
--cc=gdb-patches@sourceware.org \
--cc=palves@redhat.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox