/* Process record and replay target for GDB, the GNU debugger.
- Copyright (C) 2008, 2009 Free Software Foundation, Inc.
+ Copyright (C) 2008-2013 Free Software Foundation, Inc.
This file is part of GDB.
#include "defs.h"
#include "gdbcmd.h"
-#include "regcache.h"
-#include "gdbthread.h"
-#include "event-top.h"
-#include "exceptions.h"
+#include "completer.h"
#include "record.h"
+#include "observer.h"
+#include "inferior.h"
+#include "common/common-utils.h"
+#include "cli/cli-utils.h"
+#include "disasm.h"
-#include <signal.h>
+#include <ctype.h>
-#define DEFAULT_RECORD_INSN_MAX_NUM 200000
+/* This is the debug switch for process record. */
+unsigned int record_debug = 0;
-#define RECORD_IS_REPLAY \
- (record_list->next || execution_direction == EXEC_REVERSE)
+/* The number of instructions to print in "record instruction-history". */
+static unsigned int record_insn_history_size = 10;
-/* These are the core struct of record function.
+struct cmd_list_element *record_cmdlist = NULL;
+struct cmd_list_element *set_record_cmdlist = NULL;
+struct cmd_list_element *show_record_cmdlist = NULL;
+struct cmd_list_element *info_record_cmdlist = NULL;
- An record_entry is a record of the value change of a register
- ("record_reg") or a part of memory ("record_mem"). And each
- instruction must has a struct record_entry ("record_end") that points out this
- is the last struct record_entry of this instruction.
+#define DEBUG(msg, args...) \
+ if (record_debug) \
+ fprintf_unfiltered (gdb_stdlog, "record: " msg "\n", ##args)
- Each struct record_entry is linked to "record_list" by "prev" and "next". */
+/* Find the record target in the target stack. */
-struct record_reg_entry
+static struct target_ops *
+find_record_target (void)
{
- int num;
- gdb_byte *val;
-};
+ struct target_ops *t;
-struct record_mem_entry
-{
- CORE_ADDR addr;
- int len;
- /* Set this flag if target memory for this entry
- can no longer be accessed. */
- int mem_entry_not_accessible;
- gdb_byte *val;
-};
-
-enum record_type
-{
- record_end = 0,
- record_reg,
- record_mem
-};
+ for (t = current_target.beneath; t != NULL; t = t->beneath)
+ if (t->to_stratum == record_stratum)
+ return t;
-struct record_entry
-{
- struct record_entry *prev;
- struct record_entry *next;
- enum record_type type;
- union
- {
- /* reg */
- struct record_reg_entry reg;
- /* mem */
- struct record_mem_entry mem;
- } u;
-};
+ return NULL;
+}
-/* This is the debug switch for process record. */
-int record_debug = 0;
-
-/* These list is for execution log. */
-static struct record_entry record_first;
-static struct record_entry *record_list = &record_first;
-static struct record_entry *record_arch_list_head = NULL;
-static struct record_entry *record_arch_list_tail = NULL;
-
-/* 1 ask user. 0 auto delete the last struct record_entry. */
-static int record_stop_at_limit = 1;
-static int record_insn_max_num = DEFAULT_RECORD_INSN_MAX_NUM;
-static int record_insn_num = 0;
-
-/* The target_ops of process record. */
-static struct target_ops record_ops;
-
-/* The beneath function pointers. */
-static struct target_ops *record_beneath_to_resume_ops;
-static void (*record_beneath_to_resume) (struct target_ops *, ptid_t, int,
- enum target_signal);
-static struct target_ops *record_beneath_to_wait_ops;
-static ptid_t (*record_beneath_to_wait) (struct target_ops *, ptid_t,
- struct target_waitstatus *,
- int);
-static struct target_ops *record_beneath_to_store_registers_ops;
-static void (*record_beneath_to_store_registers) (struct target_ops *,
- struct regcache *,
- int regno);
-static struct target_ops *record_beneath_to_xfer_partial_ops;
-static LONGEST (*record_beneath_to_xfer_partial) (struct target_ops *ops,
- enum target_object object,
- const char *annex,
- gdb_byte *readbuf,
- const gdb_byte *writebuf,
- ULONGEST offset,
- LONGEST len);
-static int (*record_beneath_to_insert_breakpoint) (struct gdbarch *,
- struct bp_target_info *);
-static int (*record_beneath_to_remove_breakpoint) (struct gdbarch *,
- struct bp_target_info *);
+/* Check that recording is active. Throw an error, if it isn't. */
-static void
-record_list_release (struct record_entry *rec)
+static struct target_ops *
+require_record_target (void)
{
- struct record_entry *tmp;
-
- if (!rec)
- return;
-
- while (rec->next)
- {
- rec = rec->next;
- }
+ struct target_ops *t;
- while (rec->prev)
- {
- tmp = rec;
- rec = rec->prev;
- if (tmp->type == record_reg)
- xfree (tmp->u.reg.val);
- else if (tmp->type == record_mem)
- xfree (tmp->u.mem.val);
- xfree (tmp);
- }
+ t = find_record_target ();
+ if (t == NULL)
+ error (_("No record target is currently active.\n"
+ "Use one of the \"target record-<tab><tab>\" commands first."));
- if (rec != &record_first)
- xfree (rec);
+ return t;
}
-static void
-record_list_release_next (void)
-{
- struct record_entry *rec = record_list;
- struct record_entry *tmp = rec->next;
- rec->next = NULL;
- while (tmp)
- {
- rec = tmp->next;
- if (tmp->type == record_reg)
- record_insn_num--;
- else if (tmp->type == record_reg)
- xfree (tmp->u.reg.val);
- else if (tmp->type == record_mem)
- xfree (tmp->u.mem.val);
- xfree (tmp);
- tmp = rec;
- }
-}
+/* See record.h. */
-static void
-record_list_release_first (void)
+int
+record_read_memory (struct gdbarch *gdbarch,
+ CORE_ADDR memaddr, gdb_byte *myaddr,
+ ssize_t len)
{
- struct record_entry *tmp = NULL;
- enum record_type type;
-
- if (!record_first.next)
- return;
-
- while (1)
- {
- type = record_first.next->type;
-
- if (type == record_reg)
- xfree (record_first.next->u.reg.val);
- else if (type == record_mem)
- xfree (record_first.next->u.mem.val);
- tmp = record_first.next;
- record_first.next = tmp->next;
- xfree (tmp);
-
- if (!record_first.next)
- {
- gdb_assert (record_insn_num == 1);
- break;
- }
+ int ret = target_read_memory (memaddr, myaddr, len);
- record_first.next->prev = &record_first;
+ if (ret != 0)
+ DEBUG ("error reading memory at addr %s len = %ld.\n",
+ paddress (gdbarch, memaddr), (long) len);
- if (type == record_end)
- break;
- }
-
- record_insn_num--;
+ return ret;
}
-/* Add a struct record_entry to record_arch_list. */
+/* Stop recording. */
static void
-record_arch_list_add (struct record_entry *rec)
+record_stop (struct target_ops *t)
{
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: record_arch_list_add %s.\n",
- host_address_to_string (rec));
+ DEBUG ("stop %s", t->to_shortname);
- if (record_arch_list_tail)
- {
- record_arch_list_tail->next = rec;
- rec->prev = record_arch_list_tail;
- record_arch_list_tail = rec;
- }
- else
- {
- record_arch_list_head = rec;
- record_arch_list_tail = rec;
- }
+ if (t->to_stop_recording != NULL)
+ t->to_stop_recording ();
}
-/* Record the value of a register NUM to record_arch_list. */
+/* Unpush the record target. */
-int
-record_arch_list_add_reg (struct regcache *regcache, int num)
+static void
+record_unpush (struct target_ops *t)
{
- struct record_entry *rec;
+ DEBUG ("unpush %s", t->to_shortname);
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: add register num = %d to "
- "record list.\n",
- num);
-
- rec = (struct record_entry *) xmalloc (sizeof (struct record_entry));
- rec->u.reg.val = (gdb_byte *) xmalloc (MAX_REGISTER_SIZE);
- rec->prev = NULL;
- rec->next = NULL;
- rec->type = record_reg;
- rec->u.reg.num = num;
-
- regcache_raw_read (regcache, num, rec->u.reg.val);
-
- record_arch_list_add (rec);
-
- return 0;
+ unpush_target (t);
}
-/* Record the value of a region of memory whose address is ADDR and
- length is LEN to record_arch_list. */
+/* See record.h. */
-int
-record_arch_list_add_mem (CORE_ADDR addr, int len)
+void
+record_disconnect (struct target_ops *t, char *args, int from_tty)
{
- struct record_entry *rec;
-
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: add mem addr = %s len = %d to "
- "record list.\n",
- paddress (target_gdbarch, addr), len);
-
- if (!addr)
- return 0;
-
- rec = (struct record_entry *) xmalloc (sizeof (struct record_entry));
- rec->u.mem.val = (gdb_byte *) xmalloc (len);
- rec->prev = NULL;
- rec->next = NULL;
- rec->type = record_mem;
- rec->u.mem.addr = addr;
- rec->u.mem.len = len;
- rec->u.mem.mem_entry_not_accessible = 0;
-
- if (target_read_memory (addr, rec->u.mem.val, len))
- {
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: error reading memory at "
- "addr = %s len = %d.\n",
- paddress (target_gdbarch, addr), len);
- xfree (rec->u.mem.val);
- xfree (rec);
- return -1;
- }
+ gdb_assert (t->to_stratum == record_stratum);
- record_arch_list_add (rec);
+ DEBUG ("disconnect %s", t->to_shortname);
- return 0;
+ record_stop (t);
+ record_unpush (t);
+
+ target_disconnect (args, from_tty);
}
-/* Add a record_end type struct record_entry to record_arch_list. */
+/* See record.h. */
-int
-record_arch_list_add_end (void)
+void
+record_detach (struct target_ops *t, char *args, int from_tty)
{
- struct record_entry *rec;
-
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: add end to arch list.\n");
+ gdb_assert (t->to_stratum == record_stratum);
- rec = (struct record_entry *) xmalloc (sizeof (struct record_entry));
- rec->prev = NULL;
- rec->next = NULL;
- rec->type = record_end;
+ DEBUG ("detach %s", t->to_shortname);
- record_arch_list_add (rec);
-
- return 0;
-}
+ record_stop (t);
+ record_unpush (t);
-static void
-record_check_insn_num (int set_terminal)
-{
- if (record_insn_max_num)
- {
- gdb_assert (record_insn_num <= record_insn_max_num);
- if (record_insn_num == record_insn_max_num)
- {
- /* Ask user what to do. */
- if (record_stop_at_limit)
- {
- int q;
- if (set_terminal)
- target_terminal_ours ();
- q = yquery (_("Do you want to auto delete previous execution "
- "log entries when record/replay buffer becomes "
- "full (record stop-at-limit)?"));
- if (set_terminal)
- target_terminal_inferior ();
- if (q)
- record_stop_at_limit = 0;
- else
- error (_("Process record: inferior program stopped."));
- }
- }
- }
+ target_detach (args, from_tty);
}
-/* Before inferior step (when GDB record the running message, inferior
- only can step), GDB will call this function to record the values to
- record_list. This function will call gdbarch_process_record to
- record the running message of inferior and set them to
- record_arch_list, and add it to record_list. */
+/* See record.h. */
-static void
-record_message_cleanups (void *ignore)
+void
+record_mourn_inferior (struct target_ops *t)
{
- record_list_release (record_arch_list_tail);
-}
+ gdb_assert (t->to_stratum == record_stratum);
-static int
-record_message (void *args)
-{
- int ret;
- struct regcache *regcache = args;
- struct cleanup *old_cleanups = make_cleanup (record_message_cleanups, 0);
+ DEBUG ("mourn inferior %s", t->to_shortname);
- record_arch_list_head = NULL;
- record_arch_list_tail = NULL;
+ /* It is safer to not stop recording. Resources will be freed when
+ threads are discarded. */
+ record_unpush (t);
- /* Check record_insn_num. */
- record_check_insn_num (1);
+ target_mourn_inferior ();
+}
- ret = gdbarch_process_record (get_regcache_arch (regcache),
- regcache,
- regcache_read_pc (regcache));
- if (ret > 0)
- error (_("Process record: inferior program stopped."));
- if (ret < 0)
- error (_("Process record: failed to record execution log."));
+/* See record.h. */
- discard_cleanups (old_cleanups);
+void
+record_kill (struct target_ops *t)
+{
+ gdb_assert (t->to_stratum == record_stratum);
- record_list->next = record_arch_list_head;
- record_arch_list_head->prev = record_list;
- record_list = record_arch_list_tail;
+ DEBUG ("kill %s", t->to_shortname);
- if (record_insn_num == record_insn_max_num && record_insn_max_num)
- record_list_release_first ();
- else
- record_insn_num++;
+ /* It is safer to not stop recording. Resources will be freed when
+ threads are discarded. */
+ record_unpush (t);
- return 1;
+ target_kill ();
}
-static int
-do_record_message (struct regcache *regcache)
+/* Implement "show record debug" command. */
+
+static void
+show_record_debug (struct ui_file *file, int from_tty,
+ struct cmd_list_element *c, const char *value)
{
- return catch_errors (record_message, regcache, NULL, RETURN_MASK_ALL);
+ fprintf_filtered (file, _("Debugging of process record target is %s.\n"),
+ value);
}
-/* Set to 1 if record_store_registers and record_xfer_partial
- doesn't need record. */
-
-static int record_gdb_operation_disable = 0;
+/* Alias for "target record". */
-struct cleanup *
-record_gdb_operation_disable_set (void)
+static void
+cmd_record_start (char *args, int from_tty)
{
- struct cleanup *old_cleanups = NULL;
-
- old_cleanups =
- make_cleanup_restore_integer (&record_gdb_operation_disable);
- record_gdb_operation_disable = 1;
-
- return old_cleanups;
+ execute_command ("target record-full", from_tty);
}
+/* Truncate the record log from the present point
+ of replay until the end. */
+
static void
-record_open (char *name, int from_tty)
+cmd_record_delete (char *args, int from_tty)
{
- struct target_ops *t;
+ require_record_target ();
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: record_open\n");
-
- /* check exec */
- if (!target_has_execution)
- error (_("Process record: the program is not being run."));
- if (non_stop)
- error (_("Process record target can't debug inferior in non-stop mode "
- "(non-stop)."));
- if (target_async_permitted)
- error (_("Process record target can't debug inferior in asynchronous "
- "mode (target-async)."));
-
- if (!gdbarch_process_record_p (target_gdbarch))
- error (_("Process record: the current architecture doesn't support "
- "record function."));
-
- /* Check if record target is already running. */
- if (current_target.to_stratum == record_stratum)
+ if (!target_record_is_replaying ())
{
- if (!nquery
- (_("Process record target already running, do you want to delete "
- "the old record log?")))
- return;
+ printf_unfiltered (_("Already at end of record list.\n"));
+ return;
}
- /*Reset the beneath function pointers. */
- record_beneath_to_resume = NULL;
- record_beneath_to_wait = NULL;
- record_beneath_to_store_registers = NULL;
- record_beneath_to_xfer_partial = NULL;
- record_beneath_to_insert_breakpoint = NULL;
- record_beneath_to_remove_breakpoint = NULL;
-
- /* Set the beneath function pointers. */
- for (t = current_target.beneath; t != NULL; t = t->beneath)
+ if (!target_supports_delete_record ())
{
- if (!record_beneath_to_resume)
- {
- record_beneath_to_resume = t->to_resume;
- record_beneath_to_resume_ops = t;
- }
- if (!record_beneath_to_wait)
- {
- record_beneath_to_wait = t->to_wait;
- record_beneath_to_wait_ops = t;
- }
- if (!record_beneath_to_store_registers)
- {
- record_beneath_to_store_registers = t->to_store_registers;
- record_beneath_to_store_registers_ops = t;
- }
- if (!record_beneath_to_xfer_partial)
- {
- record_beneath_to_xfer_partial = t->to_xfer_partial;
- record_beneath_to_xfer_partial_ops = t;
- }
- if (!record_beneath_to_insert_breakpoint)
- record_beneath_to_insert_breakpoint = t->to_insert_breakpoint;
- if (!record_beneath_to_remove_breakpoint)
- record_beneath_to_remove_breakpoint = t->to_remove_breakpoint;
+ printf_unfiltered (_("The current record target does not support "
+ "this operation.\n"));
+ return;
}
- if (!record_beneath_to_resume)
- error (_("Process record can't get to_resume."));
- if (!record_beneath_to_wait)
- error (_("Process record can't get to_wait."));
- if (!record_beneath_to_store_registers)
- error (_("Process record can't get to_store_registers."));
- if (!record_beneath_to_xfer_partial)
- error (_("Process record can't get to_xfer_partial."));
- if (!record_beneath_to_insert_breakpoint)
- error (_("Process record can't get to_insert_breakpoint."));
- if (!record_beneath_to_remove_breakpoint)
- error (_("Process record can't get to_remove_breakpoint."));
-
- push_target (&record_ops);
-
- /* Reset */
- record_insn_num = 0;
- record_list = &record_first;
- record_list->next = NULL;
+
+ if (!from_tty || query (_("Delete the log from this point forward "
+ "and begin to record the running message "
+ "at current PC?")))
+ target_delete_record ();
}
+/* Implement the "stoprecord" or "record stop" command. */
+
static void
-record_close (int quitting)
+cmd_record_stop (char *args, int from_tty)
{
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: record_close\n");
+ struct target_ops *t;
- record_list_release (record_list);
-}
+ t = require_record_target ();
-static int record_resume_step = 0;
-static enum target_signal record_resume_siggnal;
-static int record_resume_error;
+ record_stop (t);
+ record_unpush (t);
-static void
-record_resume (struct target_ops *ops, ptid_t ptid, int step,
- enum target_signal siggnal)
-{
- record_resume_step = step;
- record_resume_siggnal = siggnal;
+ printf_unfiltered (_("Process record is stopped and all execution "
+ "logs are deleted.\n"));
- if (!RECORD_IS_REPLAY)
- {
- if (do_record_message (get_current_regcache ()))
- {
- record_resume_error = 0;
- }
- else
- {
- record_resume_error = 1;
- return;
- }
- record_beneath_to_resume (record_beneath_to_resume_ops, ptid, 1,
- siggnal);
- }
+ observer_notify_record_changed (current_inferior (), 0);
}
-static int record_get_sig = 0;
+/* The "set record" command. */
static void
-record_sig_handler (int signo)
+set_record_command (char *args, int from_tty)
{
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: get a signal\n");
+ printf_unfiltered (_("\"set record\" must be followed "
+ "by an apporpriate subcommand.\n"));
+ help_list (set_record_cmdlist, "set record ", all_commands, gdb_stdout);
+}
- /* It will break the running inferior in replay mode. */
- record_resume_step = 1;
+/* The "show record" command. */
- /* It will let record_wait set inferior status to get the signal
- SIGINT. */
- record_get_sig = 1;
+static void
+show_record_command (char *args, int from_tty)
+{
+ cmd_show_list (show_record_cmdlist, from_tty, "");
}
+/* The "info record" command. */
+
static void
-record_wait_cleanups (void *ignore)
+info_record_command (char *args, int from_tty)
{
- if (execution_direction == EXEC_REVERSE)
+ struct target_ops *t;
+
+ t = find_record_target ();
+ if (t == NULL)
{
- if (record_list->next)
- record_list = record_list->next;
+ printf_filtered (_("No record target is currently active.\n"));
+ return;
}
- else
- record_list = record_list->prev;
+
+ printf_filtered (_("Active record target: %s\n"), t->to_shortname);
+ if (t->to_info_record != NULL)
+ t->to_info_record ();
}
-/* In replay mode, this function examines the recorded log and
- determines where to stop. */
+/* The "record save" command. */
-static ptid_t
-record_wait (struct target_ops *ops,
- ptid_t ptid, struct target_waitstatus *status,
- int options)
+static void
+cmd_record_save (char *args, int from_tty)
{
- struct cleanup *set_cleanups = record_gdb_operation_disable_set ();
+ char *recfilename, recfilename_buffer[40];
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: record_wait "
- "record_resume_step = %d\n",
- record_resume_step);
+ require_record_target ();
- if (!RECORD_IS_REPLAY)
+ if (args != NULL && *args != 0)
+ recfilename = args;
+ else
{
- if (record_resume_error)
- {
- /* If record_resume get error, return directly. */
- status->kind = TARGET_WAITKIND_STOPPED;
- status->value.sig = TARGET_SIGNAL_ABRT;
- return inferior_ptid;
- }
+ /* Default recfile name is "gdb_record.PID". */
+ xsnprintf (recfilename_buffer, sizeof (recfilename_buffer),
+ "gdb_record.%d", PIDGET (inferior_ptid));
+ recfilename = recfilename_buffer;
+ }
- if (record_resume_step)
- {
- /* This is a single step. */
- return record_beneath_to_wait (record_beneath_to_wait_ops,
- ptid, status, 0);
- }
- else
- {
- /* This is not a single step. */
- ptid_t ret;
- CORE_ADDR tmp_pc;
+ target_save_record (recfilename);
+}
- while (1)
- {
- ret = record_beneath_to_wait (record_beneath_to_wait_ops,
- ptid, status, 0);
-
- if (status->kind == TARGET_WAITKIND_STOPPED
- && status->value.sig == TARGET_SIGNAL_TRAP)
- {
- /* Check if there is a breakpoint. */
- registers_changed ();
- tmp_pc = regcache_read_pc (get_current_regcache ());
- if (breakpoint_inserted_here_p (tmp_pc))
- {
- /* There is a breakpoint. */
- CORE_ADDR decr_pc_after_break =
- gdbarch_decr_pc_after_break
- (get_regcache_arch (get_current_regcache ()));
- if (decr_pc_after_break)
- {
- regcache_write_pc (get_thread_regcache (ret),
- tmp_pc + decr_pc_after_break);
- }
- }
- else
- {
- /* There is not a breakpoint. */
- if (!do_record_message (get_current_regcache ()))
- {
- break;
- }
- record_beneath_to_resume (record_beneath_to_resume_ops,
- ptid, 1,
- record_resume_siggnal);
- continue;
- }
- }
-
- /* The inferior is broken by a breakpoint or a signal. */
- break;
- }
+/* "record goto" command. Argument is an instruction number,
+ as given by "info record".
- return ret;
- }
- }
- else
- {
- struct regcache *regcache = get_current_regcache ();
- struct gdbarch *gdbarch = get_regcache_arch (regcache);
- int continue_flag = 1;
- int first_record_end = 1;
- struct cleanup *old_cleanups = make_cleanup (record_wait_cleanups, 0);
- CORE_ADDR tmp_pc;
+ Rewinds the recording (forward or backward) to the given instruction. */
- status->kind = TARGET_WAITKIND_STOPPED;
+void
+cmd_record_goto (char *arg, int from_tty)
+{
+ require_record_target ();
- /* Check breakpoint when forward execute. */
- if (execution_direction == EXEC_FORWARD)
- {
- tmp_pc = regcache_read_pc (regcache);
- if (breakpoint_inserted_here_p (tmp_pc))
- {
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: break at %s.\n",
- paddress (gdbarch, tmp_pc));
- if (gdbarch_decr_pc_after_break (gdbarch)
- && !record_resume_step)
- regcache_write_pc (regcache,
- tmp_pc +
- gdbarch_decr_pc_after_break (gdbarch));
- goto replay_out;
- }
- }
+ if (arg == NULL || *arg == '\0')
+ error (_("Command requires an argument (insn number to go to)."));
- record_get_sig = 0;
- signal (SIGINT, record_sig_handler);
- /* If GDB is in terminal_inferior mode, it will not get the signal.
- And in GDB replay mode, GDB doesn't need to be in terminal_inferior
- mode, because inferior will not executed.
- Then set it to terminal_ours to make GDB get the signal. */
- target_terminal_ours ();
-
- /* In EXEC_FORWARD mode, record_list points to the tail of prev
- instruction. */
- if (execution_direction == EXEC_FORWARD && record_list->next)
- record_list = record_list->next;
-
- /* Loop over the record_list, looking for the next place to
- stop. */
- do
- {
- /* Check for beginning and end of log. */
- if (execution_direction == EXEC_REVERSE
- && record_list == &record_first)
- {
- /* Hit beginning of record log in reverse. */
- status->kind = TARGET_WAITKIND_NO_HISTORY;
- break;
- }
- if (execution_direction != EXEC_REVERSE && !record_list->next)
- {
- /* Hit end of record log going forward. */
- status->kind = TARGET_WAITKIND_NO_HISTORY;
- break;
- }
+ if (strncmp (arg, "start", strlen ("start")) == 0
+ || strncmp (arg, "begin", strlen ("begin")) == 0)
+ target_goto_record_begin ();
+ else if (strncmp (arg, "end", strlen ("end")) == 0)
+ target_goto_record_end ();
+ else
+ {
+ ULONGEST insn;
- /* Set ptid, register and memory according to record_list. */
- if (record_list->type == record_reg)
- {
- /* reg */
- gdb_byte reg[MAX_REGISTER_SIZE];
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: record_reg %s to "
- "inferior num = %d.\n",
- host_address_to_string (record_list),
- record_list->u.reg.num);
- regcache_cooked_read (regcache, record_list->u.reg.num, reg);
- regcache_cooked_write (regcache, record_list->u.reg.num,
- record_list->u.reg.val);
- memcpy (record_list->u.reg.val, reg, MAX_REGISTER_SIZE);
- }
- else if (record_list->type == record_mem)
- {
- /* mem */
- /* Nothing to do if the entry is flagged not_accessible. */
- if (!record_list->u.mem.mem_entry_not_accessible)
- {
- gdb_byte *mem = alloca (record_list->u.mem.len);
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: record_mem %s to "
- "inferior addr = %s len = %d.\n",
- host_address_to_string (record_list),
- paddress (gdbarch,
- record_list->u.mem.addr),
- record_list->u.mem.len);
-
- if (target_read_memory (record_list->u.mem.addr, mem,
- record_list->u.mem.len))
- {
- if (execution_direction != EXEC_REVERSE)
- error (_("Process record: error reading memory at "
- "addr = %s len = %d."),
- paddress (gdbarch, record_list->u.mem.addr),
- record_list->u.mem.len);
- else
- /* Read failed --
- flag entry as not_accessible. */
- record_list->u.mem.mem_entry_not_accessible = 1;
- }
- else
- {
- if (target_write_memory (record_list->u.mem.addr,
- record_list->u.mem.val,
- record_list->u.mem.len))
- {
- if (execution_direction != EXEC_REVERSE)
- error (_("Process record: error writing memory at "
- "addr = %s len = %d."),
- paddress (gdbarch, record_list->u.mem.addr),
- record_list->u.mem.len);
- else
- /* Write failed --
- flag entry as not_accessible. */
- record_list->u.mem.mem_entry_not_accessible = 1;
- }
- else
- {
- memcpy (record_list->u.mem.val, mem,
- record_list->u.mem.len);
- }
- }
- }
- }
- else
- {
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: record_end %s to "
- "inferior.\n",
- host_address_to_string (record_list));
-
- if (first_record_end && execution_direction == EXEC_REVERSE)
- {
- /* When reverse excute, the first record_end is the part of
- current instruction. */
- first_record_end = 0;
- }
- else
- {
- /* In EXEC_REVERSE mode, this is the record_end of prev
- instruction.
- In EXEC_FORWARD mode, this is the record_end of current
- instruction. */
- /* step */
- if (record_resume_step)
- {
- if (record_debug > 1)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: step.\n");
- continue_flag = 0;
- }
-
- /* check breakpoint */
- tmp_pc = regcache_read_pc (regcache);
- if (breakpoint_inserted_here_p (tmp_pc))
- {
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog,
- "Process record: break "
- "at %s.\n",
- paddress (gdbarch, tmp_pc));
- if (gdbarch_decr_pc_after_break (gdbarch)
- && execution_direction == EXEC_FORWARD
- && !record_resume_step)
- regcache_write_pc (regcache,
- tmp_pc +
- gdbarch_decr_pc_after_break (gdbarch));
- continue_flag = 0;
- }
- }
- }
+ insn = parse_and_eval_long (arg);
+ target_goto_record (insn);
+ }
+}
- if (continue_flag)
- {
- if (execution_direction == EXEC_REVERSE)
- {
- if (record_list->prev)
- record_list = record_list->prev;
- }
- else
- {
- if (record_list->next)
- record_list = record_list->next;
- }
- }
- }
- while (continue_flag);
+/* Read an instruction number from an argument string. */
- signal (SIGINT, handle_sigint);
+static ULONGEST
+get_insn_number (char **arg)
+{
+ ULONGEST number;
+ const char *begin, *end, *pos;
-replay_out:
- if (record_get_sig)
- status->value.sig = TARGET_SIGNAL_INT;
- else
- status->value.sig = TARGET_SIGNAL_TRAP;
+ begin = *arg;
+ pos = skip_spaces_const (begin);
- discard_cleanups (old_cleanups);
- }
+ if (!isdigit (*pos))
+ error (_("Expected positive number, got: %s."), pos);
- do_cleanups (set_cleanups);
- return inferior_ptid;
-}
+ number = strtoulst (pos, &end, 10);
-static void
-record_disconnect (struct target_ops *target, char *args, int from_tty)
-{
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: record_disconnect\n");
+ *arg += (end - begin);
- unpush_target (&record_ops);
- target_disconnect (args, from_tty);
+ return number;
}
-static void
-record_detach (struct target_ops *ops, char *args, int from_tty)
+/* Read a context size from an argument string. */
+
+static int
+get_context_size (char **arg)
{
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: record_detach\n");
+ char *pos;
+ int number;
- unpush_target (&record_ops);
- target_detach (args, from_tty);
-}
+ pos = skip_spaces (*arg);
-static void
-record_mourn_inferior (struct target_ops *ops)
-{
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: "
- "record_mourn_inferior\n");
+ if (!isdigit (*pos))
+ error (_("Expected positive number, got: %s."), pos);
- unpush_target (&record_ops);
- target_mourn_inferior ();
+ return strtol (pos, arg, 10);
}
-/* Close process record target before killing the inferior process. */
+/* Complain about junk at the end of an argument string. */
static void
-record_kill (struct target_ops *ops)
+no_chunk (char *arg)
{
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog, "Process record: record_kill\n");
-
- unpush_target (&record_ops);
- target_kill ();
+ if (*arg != 0)
+ error (_("Junk after argument: %s."), arg);
}
-/* Record registers change (by user or by GDB) to list as an instruction. */
+/* Read instruction-history modifiers from an argument string. */
-static void
-record_registers_change (struct regcache *regcache, int regnum)
+static int
+get_insn_history_modifiers (char **arg)
{
- /* Check record_insn_num. */
- record_check_insn_num (0);
+ int modifiers;
+ char *args;
- record_arch_list_head = NULL;
- record_arch_list_tail = NULL;
+ modifiers = 0;
+ args = *arg;
- if (regnum < 0)
- {
- int i;
- for (i = 0; i < gdbarch_num_regs (get_regcache_arch (regcache)); i++)
- {
- if (record_arch_list_add_reg (regcache, i))
- {
- record_list_release (record_arch_list_tail);
- error (_("Process record: failed to record execution log."));
- }
- }
- }
- else
- {
- if (record_arch_list_add_reg (regcache, regnum))
- {
- record_list_release (record_arch_list_tail);
- error (_("Process record: failed to record execution log."));
- }
- }
- if (record_arch_list_add_end ())
- {
- record_list_release (record_arch_list_tail);
- error (_("Process record: failed to record execution log."));
- }
- record_list->next = record_arch_list_head;
- record_arch_list_head->prev = record_list;
- record_list = record_arch_list_tail;
+ if (args == NULL)
+ return modifiers;
- if (record_insn_num == record_insn_max_num && record_insn_max_num)
- record_list_release_first ();
- else
- record_insn_num++;
-}
-
-static void
-record_store_registers (struct target_ops *ops, struct regcache *regcache,
- int regno)
-{
- if (!record_gdb_operation_disable)
+ while (*args == '/')
{
- if (RECORD_IS_REPLAY)
- {
- int n;
- struct cleanup *old_cleanups;
-
- /* Let user choose if he wants to write register or not. */
- if (regno < 0)
- n =
- nquery (_("Because GDB is in replay mode, changing the "
- "value of a register will make the execution "
- "log unusable from this point onward. "
- "Change all registers?"));
- else
- n =
- nquery (_("Because GDB is in replay mode, changing the value "
- "of a register will make the execution log unusable "
- "from this point onward. Change register %s?"),
- gdbarch_register_name (get_regcache_arch (regcache),
- regno));
-
- if (!n)
- {
- /* Invalidate the value of regcache that was set in function
- "regcache_raw_write". */
- if (regno < 0)
- {
- int i;
- for (i = 0;
- i < gdbarch_num_regs (get_regcache_arch (regcache));
- i++)
- regcache_invalidate (regcache, i);
- }
- else
- regcache_invalidate (regcache, regno);
-
- error (_("Process record canceled the operation."));
- }
+ ++args;
- /* Destroy the record from here forward. */
- record_list_release_next ();
- }
-
- record_registers_change (regcache, regno);
- }
- record_beneath_to_store_registers (record_beneath_to_store_registers_ops,
- regcache, regno);
-}
-
-/* Behavior is conditional on RECORD_IS_REPLAY.
- In replay mode, we cannot write memory unles we are willing to
- invalidate the record/replay log from this point forward. */
+ if (*args == '\0')
+ error (_("Missing modifier."));
-static LONGEST
-record_xfer_partial (struct target_ops *ops, enum target_object object,
- const char *annex, gdb_byte *readbuf,
- const gdb_byte *writebuf, ULONGEST offset, LONGEST len)
-{
- if (!record_gdb_operation_disable
- && (object == TARGET_OBJECT_MEMORY
- || object == TARGET_OBJECT_RAW_MEMORY) && writebuf)
- {
- if (RECORD_IS_REPLAY)
+ for (; *args; ++args)
{
- /* Let user choose if he wants to write memory or not. */
- if (!nquery (_("Because GDB is in replay mode, writing to memory "
- "will make the execution log unusable from this "
- "point onward. Write memory at address %s?"),
- paddress (target_gdbarch, offset)))
- error (_("Process record canceled the operation."));
-
- /* Destroy the record from here forward. */
- record_list_release_next ();
- }
+ if (isspace (*args))
+ break;
- /* Check record_insn_num */
- record_check_insn_num (0);
+ if (*args == '/')
+ continue;
- /* Record registers change to list as an instruction. */
- record_arch_list_head = NULL;
- record_arch_list_tail = NULL;
- if (record_arch_list_add_mem (offset, len))
- {
- record_list_release (record_arch_list_tail);
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog,
- _("Process record: failed to record "
- "execution log."));
- return -1;
- }
- if (record_arch_list_add_end ())
- {
- record_list_release (record_arch_list_tail);
- if (record_debug)
- fprintf_unfiltered (gdb_stdlog,
- _("Process record: failed to record "
- "execution log."));
- return -1;
+ switch (*args)
+ {
+ case 'm':
+ modifiers |= DISASSEMBLY_SOURCE;
+ modifiers |= DISASSEMBLY_FILENAME;
+ break;
+ case 'r':
+ modifiers |= DISASSEMBLY_RAW_INSN;
+ break;
+ case 'f':
+ modifiers |= DISASSEMBLY_OMIT_FNAME;
+ break;
+ default:
+ error (_("Invalid modifier: %c."), *args);
+ }
}
- record_list->next = record_arch_list_head;
- record_arch_list_head->prev = record_list;
- record_list = record_arch_list_tail;
- if (record_insn_num == record_insn_max_num && record_insn_max_num)
- record_list_release_first ();
- else
- record_insn_num++;
+ args = skip_spaces (args);
}
- return record_beneath_to_xfer_partial (record_beneath_to_xfer_partial_ops,
- object, annex, readbuf, writebuf,
- offset, len);
+ /* Update the argument string. */
+ *arg = args;
+
+ return modifiers;
}
-/* Behavior is conditional on RECORD_IS_REPLAY.
- We will not actually insert or remove breakpoints when replaying,
- nor when recording. */
+/* The "record instruction-history" command. */
-static int
-record_insert_breakpoint (struct gdbarch *gdbarch,
- struct bp_target_info *bp_tgt)
+static void
+cmd_record_insn_history (char *arg, int from_tty)
{
- if (!RECORD_IS_REPLAY)
- {
- struct cleanup *old_cleanups = record_gdb_operation_disable_set ();
- int ret = record_beneath_to_insert_breakpoint (gdbarch, bp_tgt);
+ int flags, size;
- do_cleanups (old_cleanups);
+ require_record_target ();
- return ret;
- }
+ flags = get_insn_history_modifiers (&arg);
- return 0;
-}
+ /* We use a signed size to also indicate the direction. Make sure that
+ unlimited remains unlimited. */
+ size = (int) record_insn_history_size;
+ if (size < 0)
+ size = INT_MAX;
-static int
-record_remove_breakpoint (struct gdbarch *gdbarch,
- struct bp_target_info *bp_tgt)
-{
- if (!RECORD_IS_REPLAY)
+ if (arg == NULL || *arg == 0 || strcmp (arg, "+") == 0)
+ target_insn_history (size, flags);
+ else if (strcmp (arg, "-") == 0)
+ target_insn_history (-size, flags);
+ else
{
- struct cleanup *old_cleanups = record_gdb_operation_disable_set ();
- int ret = record_beneath_to_remove_breakpoint (gdbarch, bp_tgt);
+ ULONGEST begin, end;
- do_cleanups (old_cleanups);
-
- return ret;
- }
+ begin = get_insn_number (&arg);
- return 0;
-}
+ if (*arg == ',')
+ {
+ arg = skip_spaces (++arg);
-static int
-record_can_execute_reverse (void)
-{
- return 1;
-}
+ if (*arg == '+')
+ {
+ arg += 1;
+ size = get_context_size (&arg);
-static void
-init_record_ops (void)
-{
- record_ops.to_shortname = "record";
- record_ops.to_longname = "Process record and replay target";
- record_ops.to_doc =
- "Log program while executing and replay execution from log.";
- record_ops.to_open = record_open;
- record_ops.to_close = record_close;
- record_ops.to_resume = record_resume;
- record_ops.to_wait = record_wait;
- record_ops.to_disconnect = record_disconnect;
- record_ops.to_detach = record_detach;
- record_ops.to_mourn_inferior = record_mourn_inferior;
- record_ops.to_kill = record_kill;
- record_ops.to_create_inferior = find_default_create_inferior;
- record_ops.to_store_registers = record_store_registers;
- record_ops.to_xfer_partial = record_xfer_partial;
- record_ops.to_insert_breakpoint = record_insert_breakpoint;
- record_ops.to_remove_breakpoint = record_remove_breakpoint;
- record_ops.to_can_execute_reverse = record_can_execute_reverse;
- record_ops.to_stratum = record_stratum;
- record_ops.to_magic = OPS_MAGIC;
-}
+ no_chunk (arg);
-static void
-show_record_debug (struct ui_file *file, int from_tty,
- struct cmd_list_element *c, const char *value)
-{
- fprintf_filtered (file, _("Debugging of process record target is %s.\n"),
- value);
-}
+ target_insn_history_from (begin, size, flags);
+ }
+ else if (*arg == '-')
+ {
+ arg += 1;
+ size = get_context_size (&arg);
-/* Alias for "target record". */
+ no_chunk (arg);
-static void
-cmd_record_start (char *args, int from_tty)
-{
- execute_command ("target record", from_tty);
-}
+ target_insn_history_from (begin, -size, flags);
+ }
+ else
+ {
+ end = get_insn_number (&arg);
-/* Truncate the record log from the present point
- of replay until the end. */
+ no_chunk (arg);
-static void
-cmd_record_delete (char *args, int from_tty)
-{
- if (current_target.to_stratum == record_stratum)
- {
- if (RECORD_IS_REPLAY)
- {
- if (!from_tty || query (_("Delete the log from this point forward "
- "and begin to record the running message "
- "at current PC?")))
- record_list_release_next ();
+ target_insn_history_range (begin, end, flags);
+ }
}
else
- printf_unfiltered (_("Already at end of record list.\n"));
-
- }
- else
- printf_unfiltered (_("Process record is not started.\n"));
-}
-
-/* Implement the "stoprecord" command. */
-
-static void
-cmd_record_stop (char *args, int from_tty)
-{
- if (current_target.to_stratum == record_stratum)
- {
- if (!record_list || !from_tty || query (_("Delete recorded log and "
- "stop recording?")))
- unpush_target (&record_ops);
- }
- else
- printf_unfiltered (_("Process record is not started.\n"));
-}
-
-/* Set upper limit of record log size. */
+ {
+ no_chunk (arg);
-static void
-set_record_insn_max_num (char *args, int from_tty, struct cmd_list_element *c)
-{
- if (record_insn_num > record_insn_max_num && record_insn_max_num)
- {
- printf_unfiltered (_("Record instructions number is bigger than "
- "record instructions max number. Auto delete "
- "the first ones?\n"));
+ target_insn_history_from (begin, size, flags);
+ }
- while (record_insn_num > record_insn_max_num)
- record_list_release_first ();
+ dont_repeat ();
}
}
-/* Print the current index into the record log (number of insns recorded
- so far). */
+/* Provide a prototype to silence -Wmissing-prototypes. */
+extern initialize_file_ftype _initialize_record;
-static void
-show_record_insn_number (char *ignore, int from_tty)
+void
+_initialize_record (void)
{
- printf_unfiltered (_("Record instruction number is %d.\n"),
- record_insn_num);
-}
+ struct cmd_list_element *c;
-static struct cmd_list_element *record_cmdlist, *set_record_cmdlist,
- *show_record_cmdlist, *info_record_cmdlist;
+ add_setshow_zuinteger_cmd ("record", no_class, &record_debug,
+ _("Set debugging of record/replay feature."),
+ _("Show debugging of record/replay feature."),
+ _("When enabled, debugging output for "
+ "record/replay feature is displayed."),
+ NULL, show_record_debug, &setdebuglist,
+ &showdebuglist);
-static void
-set_record_command (char *args, int from_tty)
-{
- printf_unfiltered (_("\
-\"set record\" must be followed by an apporpriate subcommand.\n"));
- help_list (set_record_cmdlist, "set record ", all_commands, gdb_stdout);
-}
+ add_setshow_uinteger_cmd ("instruction-history-size", no_class,
+ &record_insn_history_size, _("\
+Set number of instructions to print in \"record instruction-history\"."), _("\
+Show number of instructions to print in \"record instruction-history\"."),
+ NULL, NULL, NULL, &set_record_cmdlist,
+ &show_record_cmdlist);
-static void
-show_record_command (char *args, int from_tty)
-{
- cmd_show_list (show_record_cmdlist, from_tty, "");
-}
+ c = add_prefix_cmd ("record", class_obscure, cmd_record_start,
+ _("Start recording."),
+ &record_cmdlist, "record ", 0, &cmdlist);
+ set_cmd_completer (c, filename_completer);
-static void
-info_record_command (char *args, int from_tty)
-{
- cmd_show_list (info_record_cmdlist, from_tty, "");
-}
-
-void
-_initialize_record (void)
-{
- /* Init record_first. */
- record_first.prev = NULL;
- record_first.next = NULL;
- record_first.type = record_end;
-
- init_record_ops ();
- add_target (&record_ops);
-
- add_setshow_zinteger_cmd ("record", no_class, &record_debug,
- _("Set debugging of record/replay feature."),
- _("Show debugging of record/replay feature."),
- _("When enabled, debugging output for "
- "record/replay feature is displayed."),
- NULL, show_record_debug, &setdebuglist,
- &showdebuglist);
-
- add_prefix_cmd ("record", class_obscure, cmd_record_start,
- _("Abbreviated form of \"target record\" command."),
- &record_cmdlist, "record ", 0, &cmdlist);
add_com_alias ("rec", "record", class_obscure, 1);
add_prefix_cmd ("record", class_support, set_record_command,
_("Set record options"), &set_record_cmdlist,
"info record ", 0, &infolist);
add_alias_cmd ("rec", "record", class_obscure, 1, &infolist);
+ c = add_cmd ("save", class_obscure, cmd_record_save,
+ _("Save the execution log to a file.\n\
+Argument is optional filename.\n\
+Default filename is 'gdb_record.<process_id>'."),
+ &record_cmdlist);
+ set_cmd_completer (c, filename_completer);
add_cmd ("delete", class_obscure, cmd_record_delete,
_("Delete the rest of execution log and start recording it anew."),
&record_cmdlist);
add_alias_cmd ("s", "stop", class_obscure, 1, &record_cmdlist);
- /* Record instructions number limit command. */
- add_setshow_boolean_cmd ("stop-at-limit", no_class,
- &record_stop_at_limit, _("\
-Set whether record/replay stops when record/replay buffer becomes full."), _("\
-Show whether record/replay stops when record/replay buffer becomes full."), _("\
-Default is ON.\n\
-When ON, if the record/replay buffer becomes full, ask user what to do.\n\
-When OFF, if the record/replay buffer becomes full,\n\
-delete the oldest recorded instruction to make room for each new one."),
- NULL, NULL,
- &set_record_cmdlist, &show_record_cmdlist);
- add_setshow_zinteger_cmd ("insn-number-max", no_class,
- &record_insn_max_num,
- _("Set record/replay buffer limit."),
- _("Show record/replay buffer limit."), _("\
-Set the maximum number of instructions to be stored in the\n\
-record/replay buffer. Zero means unlimited. Default is 200000."),
- set_record_insn_max_num,
- NULL, &set_record_cmdlist, &show_record_cmdlist);
- add_cmd ("insn-number", class_obscure, show_record_insn_number,
- _("Show the current number of instructions in the "
- "record/replay buffer."), &info_record_cmdlist);
+ add_cmd ("goto", class_obscure, cmd_record_goto, _("\
+Restore the program to its state at instruction number N.\n\
+Argument is instruction number, as shown by 'info record'."),
+ &record_cmdlist);
+
+ add_cmd ("instruction-history", class_obscure, cmd_record_insn_history, _("\
+Print disassembled instructions stored in the execution log.\n\
+With a /m modifier, source lines are included (if available).\n\
+With a /r modifier, raw instructions in hex are included.\n\
+With a /f modifier, function names are omitted.\n\
+With no argument, disassembles ten more instructions after the previous \
+disassembly.\n\
+\"record instruction-history -\" disassembles ten instructions before a \
+previous disassembly.\n\
+One argument specifies an instruction number as shown by 'info record', and \
+ten instructions are disassembled after that instruction.\n\
+Two arguments with comma between them specify starting and ending instruction \
+numbers to disassemble.\n\
+If the second argument is preceded by '+' or '-', it specifies the distance \
+from the first argument.\n\
+The number of instructions to disassemble can be defined with \"set record \
+instruction-history-size\"."),
+ &record_cmdlist);
}