aboutsummaryrefslogtreecommitdiff
path: root/gdb/infrun.c
diff options
context:
space:
mode:
Diffstat (limited to 'gdb/infrun.c')
-rw-r--r--gdb/infrun.c123
1 files changed, 116 insertions, 7 deletions
diff --git a/gdb/infrun.c b/gdb/infrun.c
index 339acf8..80cc891 100644
--- a/gdb/infrun.c
+++ b/gdb/infrun.c
@@ -32,8 +32,8 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
#include "annotate.h"
#include "symfile.h" /* for overlay functions */
#include "top.h"
-
#include <signal.h>
+#include "event-loop.h"
/* Prototypes for local functions */
@@ -53,9 +53,14 @@ static void delete_breakpoint_current_contents PARAMS ((PTR));
static void set_follow_fork_mode_command PARAMS ((char *arg, int from_tty, struct cmd_list_element *c));
+static void complete_execution PARAMS ((void));
+
int inferior_ignoring_startup_exec_events = 0;
int inferior_ignoring_leading_exec_events = 0;
+/* In asynchronous mode, but simulating synchronous execution. */
+int sync_execution = 0;
+
/* wait_for_inferior and normal_stop use this to notify the user
when the inferior stopped in a different thread than it had been
running in. */
@@ -1006,9 +1011,13 @@ The same program may be running in another process.");
/* Wait for it to stop (if not standalone)
and in any case decode why it stopped, and act accordingly. */
-
- wait_for_inferior ();
- normal_stop ();
+ /* Do this only if we are not using the event loop, or if the target
+ does not support asynchronous execution. */
+ if (!async_p || !target_has_async)
+ {
+ wait_for_inferior ();
+ normal_stop ();
+ }
}
/* Record the pc and sp of the program the last time it stopped.
@@ -1021,7 +1030,6 @@ static char *prev_func_name;
/* Start remote-debugging of a machine over a serial link. */
-
void
start_remote ()
{
@@ -1029,8 +1037,24 @@ start_remote ()
init_wait_for_inferior ();
stop_soon_quietly = 1;
trap_expected = 0;
- wait_for_inferior ();
- normal_stop ();
+
+ /* Go on waiting only in case gdb is not started in async mode, or
+ in case the target doesn't support async execution. */
+ if (!async_p || !target_has_async)
+ {
+ wait_for_inferior ();
+ normal_stop ();
+ }
+ else
+ {
+ /* The 'tar rem' command should always look synchronous,
+ i.e. display the prompt only once it has connected and
+ started the target. */
+ sync_execution = 1;
+ push_prompt ("", "", "");
+ delete_file_handler (input_fd);
+ target_executing = 1;
+ }
}
/* Initialize static vars when a new inferior begins. */
@@ -1179,6 +1203,72 @@ wait_for_inferior ()
do_cleanups (old_cleanups);
}
+/* Asynchronous version of wait_for_inferior. It is called by the
+ event loop whenever a change of state is detected on the file
+ descriptor corresponding to the target. It can be called more than
+ once to complete a single execution command. In such cases we need
+ to keep the state in a global variable ASYNC_ECSS. If it is the
+ last time that this function is called for a single execution
+ command, then report to the user that the inferior has stopped, and
+ do the necessary cleanups. */
+
+struct execution_control_state async_ecss;
+struct execution_control_state *async_ecs;
+
+void
+fetch_inferior_event ()
+{
+ static struct cleanup *old_cleanups;
+
+ async_ecs = &async_ecss;
+
+ if (!async_ecs->wait_some_more)
+ {
+ old_cleanups = make_exec_cleanup (delete_breakpoint_current_contents,
+ &step_resume_breakpoint);
+ make_exec_cleanup (delete_breakpoint_current_contents,
+ &through_sigtramp_breakpoint);
+
+ /* Fill in with reasonable starting values. */
+ init_execution_control_state (async_ecs);
+
+ thread_step_needed = 0;
+
+ /* We'll update this if & when we switch to a new thread. */
+ if (may_switch_from_inferior_pid)
+ switched_from_inferior_pid = inferior_pid;
+
+ overlay_cache_invalid = 1;
+
+ /* We have to invalidate the registers BEFORE calling target_wait
+ because they can be loaded from the target while in target_wait.
+ This makes remote debugging a bit more efficient for those
+ targets that provide critical registers as part of their normal
+ status mechanism. */
+
+ registers_changed ();
+ }
+
+ if (target_wait_hook)
+ async_ecs->pid = target_wait_hook (async_ecs->waiton_pid, async_ecs->wp);
+ else
+ async_ecs->pid = target_wait (async_ecs->waiton_pid, async_ecs->wp);
+
+ /* Now figure out what to do with the result of the result. */
+ handle_inferior_event (async_ecs);
+
+ if (!async_ecs->wait_some_more)
+ {
+ do_exec_cleanups (old_cleanups);
+ normal_stop ();
+ /* Is there anything left to do for the command issued to
+ complete? */
+ do_all_continuations ();
+ /* Reset things after target has stopped for the async commands. */
+ complete_execution ();
+ }
+}
+
/* Prepare an execution control state for looping through a
wait_for_inferior-type loop. */
@@ -3011,6 +3101,25 @@ stopped_for_shlib_catchpoint (bs, cp_p)
}
+/* Reset proper settings after an asynchronous command has finished.
+ If the execution command was in synchronous mode, register stdin
+ with the event loop, and reset the prompt. */
+static void
+complete_execution ()
+{
+extern cleanup_sigint_signal_handler PARAMS ((void));
+
+ if (sync_execution)
+ {
+ add_file_handler (input_fd, (file_handler_func *) call_readline, 0);
+ pop_prompt ();
+ sync_execution = 0;
+ cleanup_sigint_signal_handler ();
+ display_gdb_prompt (0);
+ }
+ target_executing = 0;
+}
+
/* Here to return control to GDB when the inferior stops for real.
Print appropriate messages, remove breakpoints, give terminal our modes.