| /* Process record and replay target for GDB, the GNU debugger. | 
 |  | 
 |    Copyright (C) 2013-2022 Free Software Foundation, Inc. | 
 |  | 
 |    This file is part of GDB. | 
 |  | 
 |    This program is free software; you can redistribute it and/or modify | 
 |    it under the terms of the GNU General Public License as published by | 
 |    the Free Software Foundation; either version 3 of the License, or | 
 |    (at your option) any later version. | 
 |  | 
 |    This program is distributed in the hope that it will be useful, | 
 |    but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 |    GNU General Public License for more details. | 
 |  | 
 |    You should have received a copy of the GNU General Public License | 
 |    along with this program.  If not, see <http://www.gnu.org/licenses/>.  */ | 
 |  | 
 | #include "defs.h" | 
 | #include "gdbcmd.h" | 
 | #include "regcache.h" | 
 | #include "gdbthread.h" | 
 | #include "inferior.h" | 
 | #include "event-top.h" | 
 | #include "completer.h" | 
 | #include "arch-utils.h" | 
 | #include "gdbcore.h" | 
 | #include "exec.h" | 
 | #include "record.h" | 
 | #include "record-full.h" | 
 | #include "elf-bfd.h" | 
 | #include "gcore.h" | 
 | #include "gdbsupport/event-loop.h" | 
 | #include "inf-loop.h" | 
 | #include "gdb_bfd.h" | 
 | #include "observable.h" | 
 | #include "infrun.h" | 
 | #include "gdbsupport/gdb_unlinker.h" | 
 | #include "gdbsupport/byte-vector.h" | 
 | #include "async-event.h" | 
 |  | 
 | #include <signal.h> | 
 |  | 
 | /* This module implements "target record-full", also known as "process | 
 |    record and replay".  This target sits on top of a "normal" target | 
 |    (a target that "has execution"), and provides a record and replay | 
 |    functionality, including reverse debugging. | 
 |  | 
 |    Target record has two modes: recording, and replaying. | 
 |  | 
 |    In record mode, we intercept the resume and wait methods. | 
 |    Whenever gdb resumes the target, we run the target in single step | 
 |    mode, and we build up an execution log in which, for each executed | 
 |    instruction, we record all changes in memory and register state. | 
 |    This is invisible to the user, to whom it just looks like an | 
 |    ordinary debugging session (except for performance degradation). | 
 |  | 
 |    In replay mode, instead of actually letting the inferior run as a | 
 |    process, we simulate its execution by playing back the recorded | 
 |    execution log.  For each instruction in the log, we simulate the | 
 |    instruction's side effects by duplicating the changes that it would | 
 |    have made on memory and registers.  */ | 
 |  | 
 | #define DEFAULT_RECORD_FULL_INSN_MAX_NUM	200000 | 
 |  | 
 | #define RECORD_FULL_IS_REPLAY \ | 
 |   (record_full_list->next || ::execution_direction == EXEC_REVERSE) | 
 |  | 
 | #define RECORD_FULL_FILE_MAGIC	netorder32(0x20091016) | 
 |  | 
 | /* These are the core structs of the process record functionality. | 
 |  | 
 |    A record_full_entry is a record of the value change of a register | 
 |    ("record_full_reg") or a part of memory ("record_full_mem").  And each | 
 |    instruction must have a struct record_full_entry ("record_full_end") | 
 |    that indicates that this is the last struct record_full_entry of this | 
 |    instruction. | 
 |  | 
 |    Each struct record_full_entry is linked to "record_full_list" by "prev" | 
 |    and "next" pointers.  */ | 
 |  | 
 | struct record_full_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; | 
 |   union | 
 |   { | 
 |     gdb_byte *ptr; | 
 |     gdb_byte buf[sizeof (gdb_byte *)]; | 
 |   } u; | 
 | }; | 
 |  | 
 | struct record_full_reg_entry | 
 | { | 
 |   unsigned short num; | 
 |   unsigned short len; | 
 |   union  | 
 |   { | 
 |     gdb_byte *ptr; | 
 |     gdb_byte buf[2 * sizeof (gdb_byte *)]; | 
 |   } u; | 
 | }; | 
 |  | 
 | struct record_full_end_entry | 
 | { | 
 |   enum gdb_signal sigval; | 
 |   ULONGEST insn_num; | 
 | }; | 
 |  | 
 | enum record_full_type | 
 | { | 
 |   record_full_end = 0, | 
 |   record_full_reg, | 
 |   record_full_mem | 
 | }; | 
 |  | 
 | /* This is the data structure that makes up the execution log. | 
 |  | 
 |    The execution log consists of a single linked list of entries | 
 |    of type "struct record_full_entry".  It is doubly linked so that it | 
 |    can be traversed in either direction. | 
 |  | 
 |    The start of the list is anchored by a struct called | 
 |    "record_full_first".  The pointer "record_full_list" either points | 
 |    to the last entry that was added to the list (in record mode), or to | 
 |    the next entry in the list that will be executed (in replay mode). | 
 |  | 
 |    Each list element (struct record_full_entry), in addition to next | 
 |    and prev pointers, consists of a union of three entry types: mem, | 
 |    reg, and end.  A field called "type" determines which entry type is | 
 |    represented by a given list element. | 
 |  | 
 |    Each instruction that is added to the execution log is represented | 
 |    by a variable number of list elements ('entries').  The instruction | 
 |    will have one "reg" entry for each register that is changed by  | 
 |    executing the instruction (including the PC in every case).  It  | 
 |    will also have one "mem" entry for each memory change.  Finally, | 
 |    each instruction will have an "end" entry that separates it from | 
 |    the changes associated with the next instruction.  */ | 
 |  | 
 | struct record_full_entry | 
 | { | 
 |   struct record_full_entry *prev; | 
 |   struct record_full_entry *next; | 
 |   enum record_full_type type; | 
 |   union | 
 |   { | 
 |     /* reg */ | 
 |     struct record_full_reg_entry reg; | 
 |     /* mem */ | 
 |     struct record_full_mem_entry mem; | 
 |     /* end */ | 
 |     struct record_full_end_entry end; | 
 |   } u; | 
 | }; | 
 |  | 
 | /* If true, query if PREC cannot record memory | 
 |    change of next instruction.  */ | 
 | bool record_full_memory_query = false; | 
 |  | 
 | struct record_full_core_buf_entry | 
 | { | 
 |   struct record_full_core_buf_entry *prev; | 
 |   struct target_section *p; | 
 |   bfd_byte *buf; | 
 | }; | 
 |  | 
 | /* Record buf with core target.  */ | 
 | static detached_regcache *record_full_core_regbuf = NULL; | 
 | static target_section_table record_full_core_sections; | 
 | static struct record_full_core_buf_entry *record_full_core_buf_list = NULL; | 
 |  | 
 | /* The following variables are used for managing the linked list that | 
 |    represents the execution log. | 
 |  | 
 |    record_full_first is the anchor that holds down the beginning of | 
 |    the list. | 
 |  | 
 |    record_full_list serves two functions: | 
 |      1) In record mode, it anchors the end of the list. | 
 |      2) In replay mode, it traverses the list and points to | 
 | 	the next instruction that must be emulated. | 
 |  | 
 |    record_full_arch_list_head and record_full_arch_list_tail are used | 
 |    to manage a separate list, which is used to build up the change | 
 |    elements of the currently executing instruction during record mode. | 
 |    When this instruction has been completely annotated in the "arch | 
 |    list", it will be appended to the main execution log.  */ | 
 |  | 
 | static struct record_full_entry record_full_first; | 
 | static struct record_full_entry *record_full_list = &record_full_first; | 
 | static struct record_full_entry *record_full_arch_list_head = NULL; | 
 | static struct record_full_entry *record_full_arch_list_tail = NULL; | 
 |  | 
 | /* true ask user. false auto delete the last struct record_full_entry.  */ | 
 | static bool record_full_stop_at_limit = true; | 
 | /* Maximum allowed number of insns in execution log.  */ | 
 | static unsigned int record_full_insn_max_num | 
 | 	= DEFAULT_RECORD_FULL_INSN_MAX_NUM; | 
 | /* Actual count of insns presently in execution log.  */ | 
 | static unsigned int record_full_insn_num = 0; | 
 | /* Count of insns logged so far (may be larger | 
 |    than count of insns presently in execution log).  */ | 
 | static ULONGEST record_full_insn_count; | 
 |  | 
 | static const char record_longname[] | 
 |   = N_("Process record and replay target"); | 
 | static const char record_doc[] | 
 |   = N_("Log program while executing and replay execution from log."); | 
 |  | 
 | /* Base class implementing functionality common to both the | 
 |    "record-full" and "record-core" targets.  */ | 
 |  | 
 | class record_full_base_target : public target_ops | 
 | { | 
 | public: | 
 |   const target_info &info () const override = 0; | 
 |  | 
 |   strata stratum () const override { return record_stratum; } | 
 |  | 
 |   void close () override; | 
 |   void async (bool) override; | 
 |   ptid_t wait (ptid_t, struct target_waitstatus *, target_wait_flags) override; | 
 |   bool stopped_by_watchpoint () override; | 
 |   bool stopped_data_address (CORE_ADDR *) override; | 
 |  | 
 |   bool stopped_by_sw_breakpoint () override; | 
 |   bool supports_stopped_by_sw_breakpoint () override; | 
 |  | 
 |   bool stopped_by_hw_breakpoint () override; | 
 |   bool supports_stopped_by_hw_breakpoint () override; | 
 |  | 
 |   bool can_execute_reverse () override; | 
 |  | 
 |   /* Add bookmark target methods.  */ | 
 |   gdb_byte *get_bookmark (const char *, int) override; | 
 |   void goto_bookmark (const gdb_byte *, int) override; | 
 |   enum exec_direction_kind execution_direction () override; | 
 |   enum record_method record_method (ptid_t ptid) override; | 
 |   void info_record () override; | 
 |   void save_record (const char *filename) override; | 
 |   bool supports_delete_record () override; | 
 |   void delete_record () override; | 
 |   bool record_is_replaying (ptid_t ptid) override; | 
 |   bool record_will_replay (ptid_t ptid, int dir) override; | 
 |   void record_stop_replaying () override; | 
 |   void goto_record_begin () override; | 
 |   void goto_record_end () override; | 
 |   void goto_record (ULONGEST insn) override; | 
 | }; | 
 |  | 
 | /* The "record-full" target.  */ | 
 |  | 
 | static const target_info record_full_target_info = { | 
 |   "record-full", | 
 |   record_longname, | 
 |   record_doc, | 
 | }; | 
 |  | 
 | class record_full_target final : public record_full_base_target | 
 | { | 
 | public: | 
 |   const target_info &info () const override | 
 |   { return record_full_target_info; } | 
 |  | 
 |   void resume (ptid_t, int, enum gdb_signal) override; | 
 |   void disconnect (const char *, int) override; | 
 |   void detach (inferior *, int) override; | 
 |   void mourn_inferior () override; | 
 |   void kill () override; | 
 |   void store_registers (struct regcache *, int) override; | 
 |   enum target_xfer_status xfer_partial (enum target_object object, | 
 | 					const char *annex, | 
 | 					gdb_byte *readbuf, | 
 | 					const gdb_byte *writebuf, | 
 | 					ULONGEST offset, ULONGEST len, | 
 | 					ULONGEST *xfered_len) override; | 
 |   int insert_breakpoint (struct gdbarch *, | 
 | 			 struct bp_target_info *) override; | 
 |   int remove_breakpoint (struct gdbarch *, | 
 | 			 struct bp_target_info *, | 
 | 			 enum remove_bp_reason) override; | 
 | }; | 
 |  | 
 | /* The "record-core" target.  */ | 
 |  | 
 | static const target_info record_full_core_target_info = { | 
 |   "record-core", | 
 |   record_longname, | 
 |   record_doc, | 
 | }; | 
 |  | 
 | class record_full_core_target final : public record_full_base_target | 
 | { | 
 | public: | 
 |   const target_info &info () const override | 
 |   { return record_full_core_target_info; } | 
 |  | 
 |   void resume (ptid_t, int, enum gdb_signal) override; | 
 |   void disconnect (const char *, int) override; | 
 |   void kill () override; | 
 |   void fetch_registers (struct regcache *regcache, int regno) override; | 
 |   void prepare_to_store (struct regcache *regcache) override; | 
 |   void store_registers (struct regcache *, int) override; | 
 |   enum target_xfer_status xfer_partial (enum target_object object, | 
 | 					const char *annex, | 
 | 					gdb_byte *readbuf, | 
 | 					const gdb_byte *writebuf, | 
 | 					ULONGEST offset, ULONGEST len, | 
 | 					ULONGEST *xfered_len) override; | 
 |   int insert_breakpoint (struct gdbarch *, | 
 | 			 struct bp_target_info *) override; | 
 |   int remove_breakpoint (struct gdbarch *, | 
 | 			 struct bp_target_info *, | 
 | 			 enum remove_bp_reason) override; | 
 |  | 
 |   bool has_execution (inferior *inf) override; | 
 | }; | 
 |  | 
 | static record_full_target record_full_ops; | 
 | static record_full_core_target record_full_core_ops; | 
 |  | 
 | void | 
 | record_full_target::detach (inferior *inf, int from_tty) | 
 | { | 
 |   record_detach (this, inf, from_tty); | 
 | } | 
 |  | 
 | void | 
 | record_full_target::disconnect (const char *args, int from_tty) | 
 | { | 
 |   record_disconnect (this, args, from_tty); | 
 | } | 
 |  | 
 | void | 
 | record_full_core_target::disconnect (const char *args, int from_tty) | 
 | { | 
 |   record_disconnect (this, args, from_tty); | 
 | } | 
 |  | 
 | void | 
 | record_full_target::mourn_inferior () | 
 | { | 
 |   record_mourn_inferior (this); | 
 | } | 
 |  | 
 | void | 
 | record_full_target::kill () | 
 | { | 
 |   record_kill (this); | 
 | } | 
 |  | 
 | /* See record-full.h.  */ | 
 |  | 
 | int | 
 | record_full_is_used (void) | 
 | { | 
 |   struct target_ops *t; | 
 |  | 
 |   t = find_record_target (); | 
 |   return (t == &record_full_ops | 
 | 	  || t == &record_full_core_ops); | 
 | } | 
 |  | 
 |  | 
 | /* Command lists for "set/show record full".  */ | 
 | static struct cmd_list_element *set_record_full_cmdlist; | 
 | static struct cmd_list_element *show_record_full_cmdlist; | 
 |  | 
 | /* Command list for "record full".  */ | 
 | static struct cmd_list_element *record_full_cmdlist; | 
 |  | 
 | static void record_full_goto_insn (struct record_full_entry *entry, | 
 | 				   enum exec_direction_kind dir); | 
 |  | 
 | /* Alloc and free functions for record_full_reg, record_full_mem, and | 
 |    record_full_end entries.  */ | 
 |  | 
 | /* Alloc a record_full_reg record entry.  */ | 
 |  | 
 | static inline struct record_full_entry * | 
 | record_full_reg_alloc (struct regcache *regcache, int regnum) | 
 | { | 
 |   struct record_full_entry *rec; | 
 |   struct gdbarch *gdbarch = regcache->arch (); | 
 |  | 
 |   rec = XCNEW (struct record_full_entry); | 
 |   rec->type = record_full_reg; | 
 |   rec->u.reg.num = regnum; | 
 |   rec->u.reg.len = register_size (gdbarch, regnum); | 
 |   if (rec->u.reg.len > sizeof (rec->u.reg.u.buf)) | 
 |     rec->u.reg.u.ptr = (gdb_byte *) xmalloc (rec->u.reg.len); | 
 |  | 
 |   return rec; | 
 | } | 
 |  | 
 | /* Free a record_full_reg record entry.  */ | 
 |  | 
 | static inline void | 
 | record_full_reg_release (struct record_full_entry *rec) | 
 | { | 
 |   gdb_assert (rec->type == record_full_reg); | 
 |   if (rec->u.reg.len > sizeof (rec->u.reg.u.buf)) | 
 |     xfree (rec->u.reg.u.ptr); | 
 |   xfree (rec); | 
 | } | 
 |  | 
 | /* Alloc a record_full_mem record entry.  */ | 
 |  | 
 | static inline struct record_full_entry * | 
 | record_full_mem_alloc (CORE_ADDR addr, int len) | 
 | { | 
 |   struct record_full_entry *rec; | 
 |  | 
 |   rec = XCNEW (struct record_full_entry); | 
 |   rec->type = record_full_mem; | 
 |   rec->u.mem.addr = addr; | 
 |   rec->u.mem.len = len; | 
 |   if (rec->u.mem.len > sizeof (rec->u.mem.u.buf)) | 
 |     rec->u.mem.u.ptr = (gdb_byte *) xmalloc (len); | 
 |  | 
 |   return rec; | 
 | } | 
 |  | 
 | /* Free a record_full_mem record entry.  */ | 
 |  | 
 | static inline void | 
 | record_full_mem_release (struct record_full_entry *rec) | 
 | { | 
 |   gdb_assert (rec->type == record_full_mem); | 
 |   if (rec->u.mem.len > sizeof (rec->u.mem.u.buf)) | 
 |     xfree (rec->u.mem.u.ptr); | 
 |   xfree (rec); | 
 | } | 
 |  | 
 | /* Alloc a record_full_end record entry.  */ | 
 |  | 
 | static inline struct record_full_entry * | 
 | record_full_end_alloc (void) | 
 | { | 
 |   struct record_full_entry *rec; | 
 |  | 
 |   rec = XCNEW (struct record_full_entry); | 
 |   rec->type = record_full_end; | 
 |  | 
 |   return rec; | 
 | } | 
 |  | 
 | /* Free a record_full_end record entry.  */ | 
 |  | 
 | static inline void | 
 | record_full_end_release (struct record_full_entry *rec) | 
 | { | 
 |   xfree (rec); | 
 | } | 
 |  | 
 | /* Free one record entry, any type. | 
 |    Return entry->type, in case caller wants to know.  */ | 
 |  | 
 | static inline enum record_full_type | 
 | record_full_entry_release (struct record_full_entry *rec) | 
 | { | 
 |   enum record_full_type type = rec->type; | 
 |  | 
 |   switch (type) { | 
 |   case record_full_reg: | 
 |     record_full_reg_release (rec); | 
 |     break; | 
 |   case record_full_mem: | 
 |     record_full_mem_release (rec); | 
 |     break; | 
 |   case record_full_end: | 
 |     record_full_end_release (rec); | 
 |     break; | 
 |   } | 
 |   return type; | 
 | } | 
 |  | 
 | /* Free all record entries in list pointed to by REC.  */ | 
 |  | 
 | static void | 
 | record_full_list_release (struct record_full_entry *rec) | 
 | { | 
 |   if (!rec) | 
 |     return; | 
 |  | 
 |   while (rec->next) | 
 |     rec = rec->next; | 
 |  | 
 |   while (rec->prev) | 
 |     { | 
 |       rec = rec->prev; | 
 |       record_full_entry_release (rec->next); | 
 |     } | 
 |  | 
 |   if (rec == &record_full_first) | 
 |     { | 
 |       record_full_insn_num = 0; | 
 |       record_full_first.next = NULL; | 
 |     } | 
 |   else | 
 |     record_full_entry_release (rec); | 
 | } | 
 |  | 
 | /* Free all record entries forward of the given list position.  */ | 
 |  | 
 | static void | 
 | record_full_list_release_following (struct record_full_entry *rec) | 
 | { | 
 |   struct record_full_entry *tmp = rec->next; | 
 |  | 
 |   rec->next = NULL; | 
 |   while (tmp) | 
 |     { | 
 |       rec = tmp->next; | 
 |       if (record_full_entry_release (tmp) == record_full_end) | 
 | 	{ | 
 | 	  record_full_insn_num--; | 
 | 	  record_full_insn_count--; | 
 | 	} | 
 |       tmp = rec; | 
 |     } | 
 | } | 
 |  | 
 | /* Delete the first instruction from the beginning of the log, to make | 
 |    room for adding a new instruction at the end of the log. | 
 |  | 
 |    Note -- this function does not modify record_full_insn_num.  */ | 
 |  | 
 | static void | 
 | record_full_list_release_first (void) | 
 | { | 
 |   struct record_full_entry *tmp; | 
 |  | 
 |   if (!record_full_first.next) | 
 |     return; | 
 |  | 
 |   /* Loop until a record_full_end.  */ | 
 |   while (1) | 
 |     { | 
 |       /* Cut record_full_first.next out of the linked list.  */ | 
 |       tmp = record_full_first.next; | 
 |       record_full_first.next = tmp->next; | 
 |       tmp->next->prev = &record_full_first; | 
 |  | 
 |       /* tmp is now isolated, and can be deleted.  */ | 
 |       if (record_full_entry_release (tmp) == record_full_end) | 
 | 	break;	/* End loop at first record_full_end.  */ | 
 |  | 
 |       if (!record_full_first.next) | 
 | 	{ | 
 | 	  gdb_assert (record_full_insn_num == 1); | 
 | 	  break;	/* End loop when list is empty.  */ | 
 | 	} | 
 |     } | 
 | } | 
 |  | 
 | /* Add a struct record_full_entry to record_full_arch_list.  */ | 
 |  | 
 | static void | 
 | record_full_arch_list_add (struct record_full_entry *rec) | 
 | { | 
 |   if (record_debug > 1) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"Process record: record_full_arch_list_add %s.\n", | 
 | 		host_address_to_string (rec)); | 
 |  | 
 |   if (record_full_arch_list_tail) | 
 |     { | 
 |       record_full_arch_list_tail->next = rec; | 
 |       rec->prev = record_full_arch_list_tail; | 
 |       record_full_arch_list_tail = rec; | 
 |     } | 
 |   else | 
 |     { | 
 |       record_full_arch_list_head = rec; | 
 |       record_full_arch_list_tail = rec; | 
 |     } | 
 | } | 
 |  | 
 | /* Return the value storage location of a record entry.  */ | 
 | static inline gdb_byte * | 
 | record_full_get_loc (struct record_full_entry *rec) | 
 | { | 
 |   switch (rec->type) { | 
 |   case record_full_mem: | 
 |     if (rec->u.mem.len > sizeof (rec->u.mem.u.buf)) | 
 |       return rec->u.mem.u.ptr; | 
 |     else | 
 |       return rec->u.mem.u.buf; | 
 |   case record_full_reg: | 
 |     if (rec->u.reg.len > sizeof (rec->u.reg.u.buf)) | 
 |       return rec->u.reg.u.ptr; | 
 |     else | 
 |       return rec->u.reg.u.buf; | 
 |   case record_full_end: | 
 |   default: | 
 |     gdb_assert_not_reached ("unexpected record_full_entry type"); | 
 |     return NULL; | 
 |   } | 
 | } | 
 |  | 
 | /* Record the value of a register NUM to record_full_arch_list.  */ | 
 |  | 
 | int | 
 | record_full_arch_list_add_reg (struct regcache *regcache, int regnum) | 
 | { | 
 |   struct record_full_entry *rec; | 
 |  | 
 |   if (record_debug > 1) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"Process record: add register num = %d to " | 
 | 		"record list.\n", | 
 | 		regnum); | 
 |  | 
 |   rec = record_full_reg_alloc (regcache, regnum); | 
 |  | 
 |   regcache->raw_read (regnum, record_full_get_loc (rec)); | 
 |  | 
 |   record_full_arch_list_add (rec); | 
 |  | 
 |   return 0; | 
 | } | 
 |  | 
 | /* Record the value of a region of memory whose address is ADDR and | 
 |    length is LEN to record_full_arch_list.  */ | 
 |  | 
 | int | 
 | record_full_arch_list_add_mem (CORE_ADDR addr, int len) | 
 | { | 
 |   struct record_full_entry *rec; | 
 |  | 
 |   if (record_debug > 1) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"Process record: add mem addr = %s len = %d to " | 
 | 		"record list.\n", | 
 | 		paddress (target_gdbarch (), addr), len); | 
 |  | 
 |   if (!addr)	/* FIXME: Why?  Some arch must permit it...  */ | 
 |     return 0; | 
 |  | 
 |   rec = record_full_mem_alloc (addr, len); | 
 |  | 
 |   if (record_read_memory (target_gdbarch (), addr, | 
 | 			  record_full_get_loc (rec), len)) | 
 |     { | 
 |       record_full_mem_release (rec); | 
 |       return -1; | 
 |     } | 
 |  | 
 |   record_full_arch_list_add (rec); | 
 |  | 
 |   return 0; | 
 | } | 
 |  | 
 | /* Add a record_full_end type struct record_full_entry to | 
 |    record_full_arch_list.  */ | 
 |  | 
 | int | 
 | record_full_arch_list_add_end (void) | 
 | { | 
 |   struct record_full_entry *rec; | 
 |  | 
 |   if (record_debug > 1) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"Process record: add end to arch list.\n"); | 
 |  | 
 |   rec = record_full_end_alloc (); | 
 |   rec->u.end.sigval = GDB_SIGNAL_0; | 
 |   rec->u.end.insn_num = ++record_full_insn_count; | 
 |  | 
 |   record_full_arch_list_add (rec); | 
 |  | 
 |   return 0; | 
 | } | 
 |  | 
 | static void | 
 | record_full_check_insn_num (void) | 
 | { | 
 |   if (record_full_insn_num == record_full_insn_max_num) | 
 |     { | 
 |       /* Ask user what to do.  */ | 
 |       if (record_full_stop_at_limit) | 
 | 	{ | 
 | 	  if (!yquery (_("Do you want to auto delete previous execution " | 
 | 			"log entries when record/replay buffer becomes " | 
 | 			"full (record full stop-at-limit)?"))) | 
 | 	    error (_("Process record: stopped by user.")); | 
 | 	  record_full_stop_at_limit = 0; | 
 | 	} | 
 |     } | 
 | } | 
 |  | 
 | /* Before inferior step (when GDB record the running message, inferior | 
 |    only can step), GDB will call this function to record the values to | 
 |    record_full_list.  This function will call gdbarch_process_record to | 
 |    record the running message of inferior and set them to | 
 |    record_full_arch_list, and add it to record_full_list.  */ | 
 |  | 
 | static void | 
 | record_full_message (struct regcache *regcache, enum gdb_signal signal) | 
 | { | 
 |   int ret; | 
 |   struct gdbarch *gdbarch = regcache->arch (); | 
 |  | 
 |   try | 
 |     { | 
 |       record_full_arch_list_head = NULL; | 
 |       record_full_arch_list_tail = NULL; | 
 |  | 
 |       /* Check record_full_insn_num.  */ | 
 |       record_full_check_insn_num (); | 
 |  | 
 |       /* If gdb sends a signal value to target_resume, | 
 | 	 save it in the 'end' field of the previous instruction. | 
 |  | 
 | 	 Maybe process record should record what really happened, | 
 | 	 rather than what gdb pretends has happened. | 
 |  | 
 | 	 So if Linux delivered the signal to the child process during | 
 | 	 the record mode, we will record it and deliver it again in | 
 | 	 the replay mode. | 
 |  | 
 | 	 If user says "ignore this signal" during the record mode, then | 
 | 	 it will be ignored again during the replay mode (no matter if | 
 | 	 the user says something different, like "deliver this signal" | 
 | 	 during the replay mode). | 
 |  | 
 | 	 User should understand that nothing he does during the replay | 
 | 	 mode will change the behavior of the child.  If he tries, | 
 | 	 then that is a user error. | 
 |  | 
 | 	 But we should still deliver the signal to gdb during the replay, | 
 | 	 if we delivered it during the recording.  Therefore we should | 
 | 	 record the signal during record_full_wait, not | 
 | 	 record_full_resume.  */ | 
 |       if (record_full_list != &record_full_first)  /* FIXME better way | 
 | 						      to check */ | 
 | 	{ | 
 | 	  gdb_assert (record_full_list->type == record_full_end); | 
 | 	  record_full_list->u.end.sigval = signal; | 
 | 	} | 
 |  | 
 |       if (signal == GDB_SIGNAL_0 | 
 | 	  || !gdbarch_process_record_signal_p (gdbarch)) | 
 | 	ret = gdbarch_process_record (gdbarch, | 
 | 				      regcache, | 
 | 				      regcache_read_pc (regcache)); | 
 |       else | 
 | 	ret = gdbarch_process_record_signal (gdbarch, | 
 | 					     regcache, | 
 | 					     signal); | 
 |  | 
 |       if (ret > 0) | 
 | 	error (_("Process record: inferior program stopped.")); | 
 |       if (ret < 0) | 
 | 	error (_("Process record: failed to record execution log.")); | 
 |     } | 
 |   catch (const gdb_exception &ex) | 
 |     { | 
 |       record_full_list_release (record_full_arch_list_tail); | 
 |       throw; | 
 |     } | 
 |  | 
 |   record_full_list->next = record_full_arch_list_head; | 
 |   record_full_arch_list_head->prev = record_full_list; | 
 |   record_full_list = record_full_arch_list_tail; | 
 |  | 
 |   if (record_full_insn_num == record_full_insn_max_num) | 
 |     record_full_list_release_first (); | 
 |   else | 
 |     record_full_insn_num++; | 
 | } | 
 |  | 
 | static bool | 
 | record_full_message_wrapper_safe (struct regcache *regcache, | 
 | 				  enum gdb_signal signal) | 
 | { | 
 |   try | 
 |     { | 
 |       record_full_message (regcache, signal); | 
 |     } | 
 |   catch (const gdb_exception &ex) | 
 |     { | 
 |       exception_print (gdb_stderr, ex); | 
 |       return false; | 
 |     } | 
 |  | 
 |   return true; | 
 | } | 
 |  | 
 | /* Set to 1 if record_full_store_registers and record_full_xfer_partial | 
 |    doesn't need record.  */ | 
 |  | 
 | static int record_full_gdb_operation_disable = 0; | 
 |  | 
 | scoped_restore_tmpl<int> | 
 | record_full_gdb_operation_disable_set (void) | 
 | { | 
 |   return make_scoped_restore (&record_full_gdb_operation_disable, 1); | 
 | } | 
 |  | 
 | /* Flag set to TRUE for target_stopped_by_watchpoint.  */ | 
 | static enum target_stop_reason record_full_stop_reason | 
 |   = TARGET_STOPPED_BY_NO_REASON; | 
 |  | 
 | /* Execute one instruction from the record log.  Each instruction in | 
 |    the log will be represented by an arbitrary sequence of register | 
 |    entries and memory entries, followed by an 'end' entry.  */ | 
 |  | 
 | static inline void | 
 | record_full_exec_insn (struct regcache *regcache, | 
 | 		       struct gdbarch *gdbarch, | 
 | 		       struct record_full_entry *entry) | 
 | { | 
 |   switch (entry->type) | 
 |     { | 
 |     case record_full_reg: /* reg */ | 
 |       { | 
 | 	gdb::byte_vector reg (entry->u.reg.len); | 
 |  | 
 | 	if (record_debug > 1) | 
 | 	  gdb_printf (gdb_stdlog, | 
 | 		      "Process record: record_full_reg %s to " | 
 | 		      "inferior num = %d.\n", | 
 | 		      host_address_to_string (entry), | 
 | 		      entry->u.reg.num); | 
 |  | 
 | 	regcache->cooked_read (entry->u.reg.num, reg.data ()); | 
 | 	regcache->cooked_write (entry->u.reg.num, record_full_get_loc (entry)); | 
 | 	memcpy (record_full_get_loc (entry), reg.data (), entry->u.reg.len); | 
 |       } | 
 |       break; | 
 |  | 
 |     case record_full_mem: /* mem */ | 
 |       { | 
 | 	/* Nothing to do if the entry is flagged not_accessible.  */ | 
 | 	if (!entry->u.mem.mem_entry_not_accessible) | 
 | 	  { | 
 | 	    gdb::byte_vector mem (entry->u.mem.len); | 
 |  | 
 | 	    if (record_debug > 1) | 
 | 	      gdb_printf (gdb_stdlog, | 
 | 			  "Process record: record_full_mem %s to " | 
 | 			  "inferior addr = %s len = %d.\n", | 
 | 			  host_address_to_string (entry), | 
 | 			  paddress (gdbarch, entry->u.mem.addr), | 
 | 			  entry->u.mem.len); | 
 |  | 
 | 	    if (record_read_memory (gdbarch, | 
 | 				    entry->u.mem.addr, mem.data (), | 
 | 				    entry->u.mem.len)) | 
 | 	      entry->u.mem.mem_entry_not_accessible = 1; | 
 | 	    else | 
 | 	      { | 
 | 		if (target_write_memory (entry->u.mem.addr,  | 
 | 					 record_full_get_loc (entry), | 
 | 					 entry->u.mem.len)) | 
 | 		  { | 
 | 		    entry->u.mem.mem_entry_not_accessible = 1; | 
 | 		    if (record_debug) | 
 | 		      warning (_("Process record: error writing memory at " | 
 | 				 "addr = %s len = %d."), | 
 | 			       paddress (gdbarch, entry->u.mem.addr), | 
 | 			       entry->u.mem.len); | 
 | 		  } | 
 | 		else | 
 | 		  { | 
 | 		    memcpy (record_full_get_loc (entry), mem.data (), | 
 | 			    entry->u.mem.len); | 
 |  | 
 | 		    /* We've changed memory --- check if a hardware | 
 | 		       watchpoint should trap.  Note that this | 
 | 		       presently assumes the target beneath supports | 
 | 		       continuable watchpoints.  On non-continuable | 
 | 		       watchpoints target, we'll want to check this | 
 | 		       _before_ actually doing the memory change, and | 
 | 		       not doing the change at all if the watchpoint | 
 | 		       traps.  */ | 
 | 		    if (hardware_watchpoint_inserted_in_range | 
 | 			(regcache->aspace (), | 
 | 			 entry->u.mem.addr, entry->u.mem.len)) | 
 | 		      record_full_stop_reason = TARGET_STOPPED_BY_WATCHPOINT; | 
 | 		  } | 
 | 	      } | 
 | 	  } | 
 |       } | 
 |       break; | 
 |     } | 
 | } | 
 |  | 
 | static void record_full_restore (void); | 
 |  | 
 | /* Asynchronous signal handle registered as event loop source for when | 
 |    we have pending events ready to be passed to the core.  */ | 
 |  | 
 | static struct async_event_handler *record_full_async_inferior_event_token; | 
 |  | 
 | static void | 
 | record_full_async_inferior_event_handler (gdb_client_data data) | 
 | { | 
 |   inferior_event_handler (INF_REG_EVENT); | 
 | } | 
 |  | 
 | /* Open the process record target for 'core' files.  */ | 
 |  | 
 | static void | 
 | record_full_core_open_1 (const char *name, int from_tty) | 
 | { | 
 |   struct regcache *regcache = get_current_regcache (); | 
 |   int regnum = gdbarch_num_regs (regcache->arch ()); | 
 |   int i; | 
 |  | 
 |   /* Get record_full_core_regbuf.  */ | 
 |   target_fetch_registers (regcache, -1); | 
 |   record_full_core_regbuf = new detached_regcache (regcache->arch (), false); | 
 |  | 
 |   for (i = 0; i < regnum; i ++) | 
 |     record_full_core_regbuf->raw_supply (i, *regcache); | 
 |  | 
 |   record_full_core_sections = build_section_table (core_bfd); | 
 |  | 
 |   current_inferior ()->push_target (&record_full_core_ops); | 
 |   record_full_restore (); | 
 | } | 
 |  | 
 | /* Open the process record target for 'live' processes.  */ | 
 |  | 
 | static void | 
 | record_full_open_1 (const char *name, int from_tty) | 
 | { | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Process record: record_full_open_1\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 (!gdbarch_process_record_p (target_gdbarch ())) | 
 |     error (_("Process record: the current architecture doesn't support " | 
 | 	     "record function.")); | 
 |  | 
 |   current_inferior ()->push_target (&record_full_ops); | 
 | } | 
 |  | 
 | static void record_full_init_record_breakpoints (void); | 
 |  | 
 | /* Open the process record target.  */ | 
 |  | 
 | static void | 
 | record_full_open (const char *name, int from_tty) | 
 | { | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Process record: record_full_open\n"); | 
 |  | 
 |   record_preopen (); | 
 |  | 
 |   /* Reset */ | 
 |   record_full_insn_num = 0; | 
 |   record_full_insn_count = 0; | 
 |   record_full_list = &record_full_first; | 
 |   record_full_list->next = NULL; | 
 |  | 
 |   if (core_bfd) | 
 |     record_full_core_open_1 (name, from_tty); | 
 |   else | 
 |     record_full_open_1 (name, from_tty); | 
 |  | 
 |   /* Register extra event sources in the event loop.  */ | 
 |   record_full_async_inferior_event_token | 
 |     = create_async_event_handler (record_full_async_inferior_event_handler, | 
 | 				  NULL, "record-full"); | 
 |  | 
 |   record_full_init_record_breakpoints (); | 
 |  | 
 |   gdb::observers::record_changed.notify (current_inferior (),  1, "full", NULL); | 
 | } | 
 |  | 
 | /* "close" target method.  Close the process record target.  */ | 
 |  | 
 | void | 
 | record_full_base_target::close () | 
 | { | 
 |   struct record_full_core_buf_entry *entry; | 
 |  | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Process record: record_full_close\n"); | 
 |  | 
 |   record_full_list_release (record_full_list); | 
 |  | 
 |   /* Release record_full_core_regbuf.  */ | 
 |   if (record_full_core_regbuf) | 
 |     { | 
 |       delete record_full_core_regbuf; | 
 |       record_full_core_regbuf = NULL; | 
 |     } | 
 |  | 
 |   /* Release record_full_core_buf_list.  */ | 
 |   while (record_full_core_buf_list) | 
 |     { | 
 |       entry = record_full_core_buf_list; | 
 |       record_full_core_buf_list = record_full_core_buf_list->prev; | 
 |       xfree (entry); | 
 |     } | 
 |  | 
 |   if (record_full_async_inferior_event_token) | 
 |     delete_async_event_handler (&record_full_async_inferior_event_token); | 
 | } | 
 |  | 
 | /* "async" target method.  */ | 
 |  | 
 | void | 
 | record_full_base_target::async (bool enable) | 
 | { | 
 |   if (enable) | 
 |     mark_async_event_handler (record_full_async_inferior_event_token); | 
 |   else | 
 |     clear_async_event_handler (record_full_async_inferior_event_token); | 
 |  | 
 |   beneath ()->async (enable); | 
 | } | 
 |  | 
 | /* The PTID and STEP arguments last passed to | 
 |    record_full_target::resume.  */ | 
 | static ptid_t record_full_resume_ptid = null_ptid; | 
 | static int record_full_resume_step = 0; | 
 |  | 
 | /* True if we've been resumed, and so each record_full_wait call should | 
 |    advance execution.  If this is false, record_full_wait will return a | 
 |    TARGET_WAITKIND_IGNORE.  */ | 
 | static int record_full_resumed = 0; | 
 |  | 
 | /* The execution direction of the last resume we got.  This is | 
 |    necessary for async mode.  Vis (order is not strictly accurate): | 
 |  | 
 |    1. user has the global execution direction set to forward | 
 |    2. user does a reverse-step command | 
 |    3. record_full_resume is called with global execution direction | 
 |       temporarily switched to reverse | 
 |    4. GDB's execution direction is reverted back to forward | 
 |    5. target record notifies event loop there's an event to handle | 
 |    6. infrun asks the target which direction was it going, and switches | 
 |       the global execution direction accordingly (to reverse) | 
 |    7. infrun polls an event out of the record target, and handles it | 
 |    8. GDB goes back to the event loop, and goto #4. | 
 | */ | 
 | static enum exec_direction_kind record_full_execution_dir = EXEC_FORWARD; | 
 |  | 
 | /* "resume" target method.  Resume the process record target.  */ | 
 |  | 
 | void | 
 | record_full_target::resume (ptid_t ptid, int step, enum gdb_signal signal) | 
 | { | 
 |   record_full_resume_ptid = inferior_ptid; | 
 |   record_full_resume_step = step; | 
 |   record_full_resumed = 1; | 
 |   record_full_execution_dir = ::execution_direction; | 
 |  | 
 |   if (!RECORD_FULL_IS_REPLAY) | 
 |     { | 
 |       struct gdbarch *gdbarch = target_thread_architecture (ptid); | 
 |  | 
 |       record_full_message (get_current_regcache (), signal); | 
 |  | 
 |       if (!step) | 
 | 	{ | 
 | 	  /* This is not hard single step.  */ | 
 | 	  if (!gdbarch_software_single_step_p (gdbarch)) | 
 | 	    { | 
 | 	      /* This is a normal continue.  */ | 
 | 	      step = 1; | 
 | 	    } | 
 | 	  else | 
 | 	    { | 
 | 	      /* This arch supports soft single step.  */ | 
 | 	      if (thread_has_single_step_breakpoints_set (inferior_thread ())) | 
 | 		{ | 
 | 		  /* This is a soft single step.  */ | 
 | 		  record_full_resume_step = 1; | 
 | 		} | 
 | 	      else | 
 | 		step = !insert_single_step_breakpoints (gdbarch); | 
 | 	    } | 
 | 	} | 
 |  | 
 |       /* Make sure the target beneath reports all signals.  */ | 
 |       target_pass_signals ({}); | 
 |  | 
 |       this->beneath ()->resume (ptid, step, signal); | 
 |     } | 
 | } | 
 |  | 
 | static int record_full_get_sig = 0; | 
 |  | 
 | /* SIGINT signal handler, registered by "wait" method.  */ | 
 |  | 
 | static void | 
 | record_full_sig_handler (int signo) | 
 | { | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Process record: get a signal\n"); | 
 |  | 
 |   /* It will break the running inferior in replay mode.  */ | 
 |   record_full_resume_step = 1; | 
 |  | 
 |   /* It will let record_full_wait set inferior status to get the signal | 
 |      SIGINT.  */ | 
 |   record_full_get_sig = 1; | 
 | } | 
 |  | 
 | /* "wait" target method for process record target. | 
 |  | 
 |    In record mode, the target is always run in singlestep mode | 
 |    (even when gdb says to continue).  The wait method intercepts | 
 |    the stop events and determines which ones are to be passed on to | 
 |    gdb.  Most stop events are just singlestep events that gdb is not | 
 |    to know about, so the wait method just records them and keeps | 
 |    singlestepping. | 
 |  | 
 |    In replay mode, this function emulates the recorded execution log,  | 
 |    one instruction at a time (forward or backward), and determines  | 
 |    where to stop.  */ | 
 |  | 
 | static ptid_t | 
 | record_full_wait_1 (struct target_ops *ops, | 
 | 		    ptid_t ptid, struct target_waitstatus *status, | 
 | 		    target_wait_flags options) | 
 | { | 
 |   scoped_restore restore_operation_disable | 
 |     = record_full_gdb_operation_disable_set (); | 
 |  | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"Process record: record_full_wait " | 
 | 		"record_full_resume_step = %d, " | 
 | 		"record_full_resumed = %d, direction=%s\n", | 
 | 		record_full_resume_step, record_full_resumed, | 
 | 		record_full_execution_dir == EXEC_FORWARD | 
 | 		? "forward" : "reverse"); | 
 |  | 
 |   if (!record_full_resumed) | 
 |     { | 
 |       gdb_assert ((options & TARGET_WNOHANG) != 0); | 
 |  | 
 |       /* No interesting event.  */ | 
 |       status->set_ignore (); | 
 |       return minus_one_ptid; | 
 |     } | 
 |  | 
 |   record_full_get_sig = 0; | 
 |   signal (SIGINT, record_full_sig_handler); | 
 |  | 
 |   record_full_stop_reason = TARGET_STOPPED_BY_NO_REASON; | 
 |  | 
 |   if (!RECORD_FULL_IS_REPLAY && ops != &record_full_core_ops) | 
 |     { | 
 |       if (record_full_resume_step) | 
 | 	{ | 
 | 	  /* This is a single step.  */ | 
 | 	  return ops->beneath ()->wait (ptid, status, options); | 
 | 	} | 
 |       else | 
 | 	{ | 
 | 	  /* This is not a single step.  */ | 
 | 	  ptid_t ret; | 
 | 	  CORE_ADDR tmp_pc; | 
 | 	  struct gdbarch *gdbarch | 
 | 	    = target_thread_architecture (record_full_resume_ptid); | 
 |  | 
 | 	  while (1) | 
 | 	    { | 
 | 	      ret = ops->beneath ()->wait (ptid, status, options); | 
 | 	      if (status->kind () == TARGET_WAITKIND_IGNORE) | 
 | 		{ | 
 | 		  if (record_debug) | 
 | 		    gdb_printf (gdb_stdlog, | 
 | 				"Process record: record_full_wait " | 
 | 				"target beneath not done yet\n"); | 
 | 		  return ret; | 
 | 		} | 
 |  | 
 | 	      for (thread_info *tp : all_non_exited_threads ()) | 
 | 		delete_single_step_breakpoints (tp); | 
 |  | 
 | 	      if (record_full_resume_step) | 
 | 		return ret; | 
 |  | 
 | 	      /* Is this a SIGTRAP?  */ | 
 | 	      if (status->kind () == TARGET_WAITKIND_STOPPED | 
 | 		  && status->sig () == GDB_SIGNAL_TRAP) | 
 | 		{ | 
 | 		  struct regcache *regcache; | 
 | 		  enum target_stop_reason *stop_reason_p | 
 | 		    = &record_full_stop_reason; | 
 |  | 
 | 		  /* Yes -- this is likely our single-step finishing, | 
 | 		     but check if there's any reason the core would be | 
 | 		     interested in the event.  */ | 
 |  | 
 | 		  registers_changed (); | 
 | 		  switch_to_thread (current_inferior ()->process_target (), | 
 | 				    ret); | 
 | 		  regcache = get_current_regcache (); | 
 | 		  tmp_pc = regcache_read_pc (regcache); | 
 | 		  const struct address_space *aspace = regcache->aspace (); | 
 |  | 
 | 		  if (target_stopped_by_watchpoint ()) | 
 | 		    { | 
 | 		      /* Always interested in watchpoints.  */ | 
 | 		    } | 
 | 		  else if (record_check_stopped_by_breakpoint (aspace, tmp_pc, | 
 | 							       stop_reason_p)) | 
 | 		    { | 
 | 		      /* There is a breakpoint here.  Let the core | 
 | 			 handle it.  */ | 
 | 		    } | 
 | 		  else | 
 | 		    { | 
 | 		      /* This is a single-step trap.  Record the | 
 | 			 insn and issue another step. | 
 | 			 FIXME: this part can be a random SIGTRAP too. | 
 | 			 But GDB cannot handle it.  */ | 
 | 		      int step = 1; | 
 |  | 
 | 		      if (!record_full_message_wrapper_safe (regcache, | 
 | 							     GDB_SIGNAL_0)) | 
 | 			{ | 
 | 			   status->set_stopped (GDB_SIGNAL_0); | 
 | 			   break; | 
 | 			} | 
 |  | 
 | 		      process_stratum_target *proc_target | 
 | 			= current_inferior ()->process_target (); | 
 |  | 
 | 		      if (gdbarch_software_single_step_p (gdbarch)) | 
 | 			{ | 
 | 			  /* Try to insert the software single step breakpoint. | 
 | 			     If insert success, set step to 0.  */ | 
 | 			  set_executing (proc_target, inferior_ptid, false); | 
 | 			  SCOPE_EXIT | 
 | 			    { | 
 | 			      set_executing (proc_target, inferior_ptid, true); | 
 | 			    }; | 
 |  | 
 | 			  reinit_frame_cache (); | 
 | 			  step = !insert_single_step_breakpoints (gdbarch); | 
 | 			} | 
 |  | 
 | 		      if (record_debug) | 
 | 			gdb_printf (gdb_stdlog, | 
 | 				    "Process record: record_full_wait " | 
 | 				    "issuing one more step in the " | 
 | 				    "target beneath\n"); | 
 | 		      ops->beneath ()->resume (ptid, step, GDB_SIGNAL_0); | 
 | 		      proc_target->commit_resumed_state = true; | 
 | 		      proc_target->commit_resumed (); | 
 | 		      proc_target->commit_resumed_state = false; | 
 | 		      continue; | 
 | 		    } | 
 | 		} | 
 |  | 
 | 	      /* The inferior is broken by a breakpoint or a signal.  */ | 
 | 	      break; | 
 | 	    } | 
 |  | 
 | 	  return ret; | 
 | 	} | 
 |     } | 
 |   else | 
 |     { | 
 |       switch_to_thread (current_inferior ()->process_target (), | 
 | 			record_full_resume_ptid); | 
 |       struct regcache *regcache = get_current_regcache (); | 
 |       struct gdbarch *gdbarch = regcache->arch (); | 
 |       const struct address_space *aspace = regcache->aspace (); | 
 |       int continue_flag = 1; | 
 |       int first_record_full_end = 1; | 
 |  | 
 |       try | 
 | 	{ | 
 | 	  CORE_ADDR tmp_pc; | 
 |  | 
 | 	  record_full_stop_reason = TARGET_STOPPED_BY_NO_REASON; | 
 | 	  status->set_stopped (GDB_SIGNAL_0); | 
 |  | 
 | 	  /* Check breakpoint when forward execute.  */ | 
 | 	  if (execution_direction == EXEC_FORWARD) | 
 | 	    { | 
 | 	      tmp_pc = regcache_read_pc (regcache); | 
 | 	      if (record_check_stopped_by_breakpoint (aspace, tmp_pc, | 
 | 						      &record_full_stop_reason)) | 
 | 		{ | 
 | 		  if (record_debug) | 
 | 		    gdb_printf (gdb_stdlog, | 
 | 				"Process record: break at %s.\n", | 
 | 				paddress (gdbarch, tmp_pc)); | 
 | 		  goto replay_out; | 
 | 		} | 
 | 	    } | 
 |  | 
 | 	  /* 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_full_list points to the tail of prev | 
 | 	     instruction.  */ | 
 | 	  if (execution_direction == EXEC_FORWARD && record_full_list->next) | 
 | 	    record_full_list = record_full_list->next; | 
 |  | 
 | 	  /* Loop over the record_full_list, looking for the next place to | 
 | 	     stop.  */ | 
 | 	  do | 
 | 	    { | 
 | 	      /* Check for beginning and end of log.  */ | 
 | 	      if (execution_direction == EXEC_REVERSE | 
 | 		  && record_full_list == &record_full_first) | 
 | 		{ | 
 | 		  /* Hit beginning of record log in reverse.  */ | 
 | 		  status->set_no_history (); | 
 | 		  break; | 
 | 		} | 
 | 	      if (execution_direction != EXEC_REVERSE | 
 | 		  && !record_full_list->next) | 
 | 		{ | 
 | 		  /* Hit end of record log going forward.  */ | 
 | 		  status->set_no_history (); | 
 | 		  break; | 
 | 		} | 
 |  | 
 | 	      record_full_exec_insn (regcache, gdbarch, record_full_list); | 
 |  | 
 | 	      if (record_full_list->type == record_full_end) | 
 | 		{ | 
 | 		  if (record_debug > 1) | 
 | 		    gdb_printf | 
 | 		      (gdb_stdlog, | 
 | 		       "Process record: record_full_end %s to " | 
 | 		       "inferior.\n", | 
 | 		       host_address_to_string (record_full_list)); | 
 |  | 
 | 		  if (first_record_full_end | 
 | 		      && execution_direction == EXEC_REVERSE) | 
 | 		    { | 
 | 		      /* When reverse execute, the first | 
 | 			 record_full_end is the part of current | 
 | 			 instruction.  */ | 
 | 		      first_record_full_end = 0; | 
 | 		    } | 
 | 		  else | 
 | 		    { | 
 | 		      /* In EXEC_REVERSE mode, this is the | 
 | 			 record_full_end of prev instruction.  In | 
 | 			 EXEC_FORWARD mode, this is the | 
 | 			 record_full_end of current instruction.  */ | 
 | 		      /* step */ | 
 | 		      if (record_full_resume_step) | 
 | 			{ | 
 | 			  if (record_debug > 1) | 
 | 			    gdb_printf (gdb_stdlog, | 
 | 					"Process record: step.\n"); | 
 | 			  continue_flag = 0; | 
 | 			} | 
 |  | 
 | 		      /* check breakpoint */ | 
 | 		      tmp_pc = regcache_read_pc (regcache); | 
 | 		      if (record_check_stopped_by_breakpoint | 
 | 			  (aspace, tmp_pc, &record_full_stop_reason)) | 
 | 			{ | 
 | 			  if (record_debug) | 
 | 			    gdb_printf (gdb_stdlog, | 
 | 					"Process record: break " | 
 | 					"at %s.\n", | 
 | 					paddress (gdbarch, tmp_pc)); | 
 |  | 
 | 			  continue_flag = 0; | 
 | 			} | 
 |  | 
 | 		      if (record_full_stop_reason | 
 | 			  == TARGET_STOPPED_BY_WATCHPOINT) | 
 | 			{ | 
 | 			  if (record_debug) | 
 | 			    gdb_printf (gdb_stdlog, | 
 | 					"Process record: hit hw " | 
 | 					"watchpoint.\n"); | 
 | 			  continue_flag = 0; | 
 | 			} | 
 | 		      /* Check target signal */ | 
 | 		      if (record_full_list->u.end.sigval != GDB_SIGNAL_0) | 
 | 			/* FIXME: better way to check */ | 
 | 			continue_flag = 0; | 
 | 		    } | 
 | 		} | 
 |  | 
 | 	      if (continue_flag) | 
 | 		{ | 
 | 		  if (execution_direction == EXEC_REVERSE) | 
 | 		    { | 
 | 		      if (record_full_list->prev) | 
 | 			record_full_list = record_full_list->prev; | 
 | 		    } | 
 | 		  else | 
 | 		    { | 
 | 		      if (record_full_list->next) | 
 | 			record_full_list = record_full_list->next; | 
 | 		    } | 
 | 		} | 
 | 	    } | 
 | 	  while (continue_flag); | 
 |  | 
 | 	replay_out: | 
 | 	  if (status->kind () == TARGET_WAITKIND_STOPPED) | 
 | 	    { | 
 | 	      if (record_full_get_sig) | 
 | 		status->set_stopped (GDB_SIGNAL_INT); | 
 | 	      else if (record_full_list->u.end.sigval != GDB_SIGNAL_0) | 
 | 		/* FIXME: better way to check */ | 
 | 		status->set_stopped (record_full_list->u.end.sigval); | 
 | 	      else | 
 | 		status->set_stopped (GDB_SIGNAL_TRAP); | 
 | 	    } | 
 | 	} | 
 |       catch (const gdb_exception &ex) | 
 | 	{ | 
 | 	  if (execution_direction == EXEC_REVERSE) | 
 | 	    { | 
 | 	      if (record_full_list->next) | 
 | 		record_full_list = record_full_list->next; | 
 | 	    } | 
 | 	  else | 
 | 	    record_full_list = record_full_list->prev; | 
 |  | 
 | 	  throw; | 
 | 	} | 
 |     } | 
 |  | 
 |   signal (SIGINT, handle_sigint); | 
 |  | 
 |   return inferior_ptid; | 
 | } | 
 |  | 
 | ptid_t | 
 | record_full_base_target::wait (ptid_t ptid, struct target_waitstatus *status, | 
 | 			       target_wait_flags options) | 
 | { | 
 |   ptid_t return_ptid; | 
 |  | 
 |   clear_async_event_handler (record_full_async_inferior_event_token); | 
 |  | 
 |   return_ptid = record_full_wait_1 (this, ptid, status, options); | 
 |   if (status->kind () != TARGET_WAITKIND_IGNORE) | 
 |     { | 
 |       /* We're reporting a stop.  Make sure any spurious | 
 | 	 target_wait(WNOHANG) doesn't advance the target until the | 
 | 	 core wants us resumed again.  */ | 
 |       record_full_resumed = 0; | 
 |     } | 
 |   return return_ptid; | 
 | } | 
 |  | 
 | bool | 
 | record_full_base_target::stopped_by_watchpoint () | 
 | { | 
 |   if (RECORD_FULL_IS_REPLAY) | 
 |     return record_full_stop_reason == TARGET_STOPPED_BY_WATCHPOINT; | 
 |   else | 
 |     return beneath ()->stopped_by_watchpoint (); | 
 | } | 
 |  | 
 | bool | 
 | record_full_base_target::stopped_data_address (CORE_ADDR *addr_p) | 
 | { | 
 |   if (RECORD_FULL_IS_REPLAY) | 
 |     return false; | 
 |   else | 
 |     return this->beneath ()->stopped_data_address (addr_p); | 
 | } | 
 |  | 
 | /* The stopped_by_sw_breakpoint method of target record-full.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::stopped_by_sw_breakpoint () | 
 | { | 
 |   return record_full_stop_reason == TARGET_STOPPED_BY_SW_BREAKPOINT; | 
 | } | 
 |  | 
 | /* The supports_stopped_by_sw_breakpoint method of target | 
 |    record-full.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::supports_stopped_by_sw_breakpoint () | 
 | { | 
 |   return true; | 
 | } | 
 |  | 
 | /* The stopped_by_hw_breakpoint method of target record-full.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::stopped_by_hw_breakpoint () | 
 | { | 
 |   return record_full_stop_reason == TARGET_STOPPED_BY_HW_BREAKPOINT; | 
 | } | 
 |  | 
 | /* The supports_stopped_by_sw_breakpoint method of target | 
 |    record-full.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::supports_stopped_by_hw_breakpoint () | 
 | { | 
 |   return true; | 
 | } | 
 |  | 
 | /* Record registers change (by user or by GDB) to list as an instruction.  */ | 
 |  | 
 | static void | 
 | record_full_registers_change (struct regcache *regcache, int regnum) | 
 | { | 
 |   /* Check record_full_insn_num.  */ | 
 |   record_full_check_insn_num (); | 
 |  | 
 |   record_full_arch_list_head = NULL; | 
 |   record_full_arch_list_tail = NULL; | 
 |  | 
 |   if (regnum < 0) | 
 |     { | 
 |       int i; | 
 |  | 
 |       for (i = 0; i < gdbarch_num_regs (regcache->arch ()); i++) | 
 | 	{ | 
 | 	  if (record_full_arch_list_add_reg (regcache, i)) | 
 | 	    { | 
 | 	      record_full_list_release (record_full_arch_list_tail); | 
 | 	      error (_("Process record: failed to record execution log.")); | 
 | 	    } | 
 | 	} | 
 |     } | 
 |   else | 
 |     { | 
 |       if (record_full_arch_list_add_reg (regcache, regnum)) | 
 | 	{ | 
 | 	  record_full_list_release (record_full_arch_list_tail); | 
 | 	  error (_("Process record: failed to record execution log.")); | 
 | 	} | 
 |     } | 
 |   if (record_full_arch_list_add_end ()) | 
 |     { | 
 |       record_full_list_release (record_full_arch_list_tail); | 
 |       error (_("Process record: failed to record execution log.")); | 
 |     } | 
 |   record_full_list->next = record_full_arch_list_head; | 
 |   record_full_arch_list_head->prev = record_full_list; | 
 |   record_full_list = record_full_arch_list_tail; | 
 |  | 
 |   if (record_full_insn_num == record_full_insn_max_num) | 
 |     record_full_list_release_first (); | 
 |   else | 
 |     record_full_insn_num++; | 
 | } | 
 |  | 
 | /* "store_registers" method for process record target.  */ | 
 |  | 
 | void | 
 | record_full_target::store_registers (struct regcache *regcache, int regno) | 
 | { | 
 |   if (!record_full_gdb_operation_disable) | 
 |     { | 
 |       if (RECORD_FULL_IS_REPLAY) | 
 | 	{ | 
 | 	  int n; | 
 |  | 
 | 	  /* Let user choose if he wants to write register or not.  */ | 
 | 	  if (regno < 0) | 
 | 	    n = | 
 | 	      query (_("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 = | 
 | 	      query (_("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 (regcache->arch (), | 
 | 					       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 (regcache->arch ()); | 
 | 		       i++) | 
 | 		    regcache->invalidate (i); | 
 | 		} | 
 | 	      else | 
 | 		regcache->invalidate (regno); | 
 |  | 
 | 	      error (_("Process record canceled the operation.")); | 
 | 	    } | 
 |  | 
 | 	  /* Destroy the record from here forward.  */ | 
 | 	  record_full_list_release_following (record_full_list); | 
 | 	} | 
 |  | 
 |       record_full_registers_change (regcache, regno); | 
 |     } | 
 |   this->beneath ()->store_registers (regcache, regno); | 
 | } | 
 |  | 
 | /* "xfer_partial" method.  Behavior is conditional on | 
 |    RECORD_FULL_IS_REPLAY. | 
 |    In replay mode, we cannot write memory unles we are willing to | 
 |    invalidate the record/replay log from this point forward.  */ | 
 |  | 
 | enum target_xfer_status | 
 | record_full_target::xfer_partial (enum target_object object, | 
 | 				  const char *annex, gdb_byte *readbuf, | 
 | 				  const gdb_byte *writebuf, ULONGEST offset, | 
 | 				  ULONGEST len, ULONGEST *xfered_len) | 
 | { | 
 |   if (!record_full_gdb_operation_disable | 
 |       && (object == TARGET_OBJECT_MEMORY | 
 | 	  || object == TARGET_OBJECT_RAW_MEMORY) && writebuf) | 
 |     { | 
 |       if (RECORD_FULL_IS_REPLAY) | 
 | 	{ | 
 | 	  /* Let user choose if he wants to write memory or not.  */ | 
 | 	  if (!query (_("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_full_list_release_following (record_full_list); | 
 | 	} | 
 |  | 
 |       /* Check record_full_insn_num */ | 
 |       record_full_check_insn_num (); | 
 |  | 
 |       /* Record registers change to list as an instruction.  */ | 
 |       record_full_arch_list_head = NULL; | 
 |       record_full_arch_list_tail = NULL; | 
 |       if (record_full_arch_list_add_mem (offset, len)) | 
 | 	{ | 
 | 	  record_full_list_release (record_full_arch_list_tail); | 
 | 	  if (record_debug) | 
 | 	    gdb_printf (gdb_stdlog, | 
 | 			"Process record: failed to record " | 
 | 			"execution log."); | 
 | 	  return TARGET_XFER_E_IO; | 
 | 	} | 
 |       if (record_full_arch_list_add_end ()) | 
 | 	{ | 
 | 	  record_full_list_release (record_full_arch_list_tail); | 
 | 	  if (record_debug) | 
 | 	    gdb_printf (gdb_stdlog, | 
 | 			"Process record: failed to record " | 
 | 			"execution log."); | 
 | 	  return TARGET_XFER_E_IO; | 
 | 	} | 
 |       record_full_list->next = record_full_arch_list_head; | 
 |       record_full_arch_list_head->prev = record_full_list; | 
 |       record_full_list = record_full_arch_list_tail; | 
 |  | 
 |       if (record_full_insn_num == record_full_insn_max_num) | 
 | 	record_full_list_release_first (); | 
 |       else | 
 | 	record_full_insn_num++; | 
 |     } | 
 |  | 
 |   return this->beneath ()->xfer_partial (object, annex, readbuf, writebuf, | 
 | 					 offset, len, xfered_len); | 
 | } | 
 |  | 
 | /* This structure represents a breakpoint inserted while the record | 
 |    target is active.  We use this to know when to install/remove | 
 |    breakpoints in/from the target beneath.  For example, a breakpoint | 
 |    may be inserted while recording, but removed when not replaying nor | 
 |    recording.  In that case, the breakpoint had not been inserted on | 
 |    the target beneath, so we should not try to remove it there.  */ | 
 |  | 
 | struct record_full_breakpoint | 
 | { | 
 |   record_full_breakpoint (struct address_space *address_space_, | 
 | 			  CORE_ADDR addr_, | 
 | 			  bool in_target_beneath_) | 
 |     : address_space (address_space_), | 
 |       addr (addr_), | 
 |       in_target_beneath (in_target_beneath_) | 
 |   { | 
 |   } | 
 |  | 
 |   /* The address and address space the breakpoint was set at.  */ | 
 |   struct address_space *address_space; | 
 |   CORE_ADDR addr; | 
 |  | 
 |   /* True when the breakpoint has been also installed in the target | 
 |      beneath.  This will be false for breakpoints set during replay or | 
 |      when recording.  */ | 
 |   bool in_target_beneath; | 
 | }; | 
 |  | 
 | /* The list of breakpoints inserted while the record target is | 
 |    active.  */ | 
 | static std::vector<record_full_breakpoint> record_full_breakpoints; | 
 |  | 
 | /* Sync existing breakpoints to record_full_breakpoints.  */ | 
 |  | 
 | static void | 
 | record_full_init_record_breakpoints (void) | 
 | { | 
 |   record_full_breakpoints.clear (); | 
 |  | 
 |   for (bp_location *loc : all_bp_locations ()) | 
 |     { | 
 |       if (loc->loc_type != bp_loc_software_breakpoint) | 
 | 	continue; | 
 |  | 
 |       if (loc->inserted) | 
 | 	record_full_breakpoints.emplace_back | 
 | 	  (loc->target_info.placed_address_space, | 
 | 	   loc->target_info.placed_address, 1); | 
 |     } | 
 | } | 
 |  | 
 | /* Behavior is conditional on RECORD_FULL_IS_REPLAY.  We will not actually | 
 |    insert or remove breakpoints in the real target when replaying, nor | 
 |    when recording.  */ | 
 |  | 
 | int | 
 | record_full_target::insert_breakpoint (struct gdbarch *gdbarch, | 
 | 				       struct bp_target_info *bp_tgt) | 
 | { | 
 |   bool in_target_beneath = false; | 
 |  | 
 |   if (!RECORD_FULL_IS_REPLAY) | 
 |     { | 
 |       /* When recording, we currently always single-step, so we don't | 
 | 	 really need to install regular breakpoints in the inferior. | 
 | 	 However, we do have to insert software single-step | 
 | 	 breakpoints, in case the target can't hardware step.  To keep | 
 | 	 things simple, we always insert.  */ | 
 |  | 
 |       scoped_restore restore_operation_disable | 
 | 	= record_full_gdb_operation_disable_set (); | 
 |  | 
 |       int ret = this->beneath ()->insert_breakpoint (gdbarch, bp_tgt); | 
 |       if (ret != 0) | 
 | 	return ret; | 
 |  | 
 |       in_target_beneath = true; | 
 |     } | 
 |  | 
 |   /* Use the existing entries if found in order to avoid duplication | 
 |      in record_full_breakpoints.  */ | 
 |  | 
 |   for (const record_full_breakpoint &bp : record_full_breakpoints) | 
 |     { | 
 |       if (bp.addr == bp_tgt->placed_address | 
 | 	  && bp.address_space == bp_tgt->placed_address_space) | 
 | 	{ | 
 | 	  gdb_assert (bp.in_target_beneath == in_target_beneath); | 
 | 	  return 0; | 
 | 	} | 
 |     } | 
 |  | 
 |   record_full_breakpoints.emplace_back (bp_tgt->placed_address_space, | 
 | 					bp_tgt->placed_address, | 
 | 					in_target_beneath); | 
 |   return 0; | 
 | } | 
 |  | 
 | /* "remove_breakpoint" method for process record target.  */ | 
 |  | 
 | int | 
 | record_full_target::remove_breakpoint (struct gdbarch *gdbarch, | 
 | 				       struct bp_target_info *bp_tgt, | 
 | 				       enum remove_bp_reason reason) | 
 | { | 
 |   for (auto iter = record_full_breakpoints.begin (); | 
 |        iter != record_full_breakpoints.end (); | 
 |        ++iter) | 
 |     { | 
 |       struct record_full_breakpoint &bp = *iter; | 
 |  | 
 |       if (bp.addr == bp_tgt->placed_address | 
 | 	  && bp.address_space == bp_tgt->placed_address_space) | 
 | 	{ | 
 | 	  if (bp.in_target_beneath) | 
 | 	    { | 
 | 	      scoped_restore restore_operation_disable | 
 | 		= record_full_gdb_operation_disable_set (); | 
 |  | 
 | 	      int ret = this->beneath ()->remove_breakpoint (gdbarch, bp_tgt, | 
 | 							     reason); | 
 | 	      if (ret != 0) | 
 | 		return ret; | 
 | 	    } | 
 |  | 
 | 	  if (reason == REMOVE_BREAKPOINT) | 
 | 	    unordered_remove (record_full_breakpoints, iter); | 
 | 	  return 0; | 
 | 	} | 
 |     } | 
 |  | 
 |   gdb_assert_not_reached ("removing unknown breakpoint"); | 
 | } | 
 |  | 
 | /* "can_execute_reverse" method for process record target.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::can_execute_reverse () | 
 | { | 
 |   return true; | 
 | } | 
 |  | 
 | /* "get_bookmark" method for process record and prec over core.  */ | 
 |  | 
 | gdb_byte * | 
 | record_full_base_target::get_bookmark (const char *args, int from_tty) | 
 | { | 
 |   char *ret = NULL; | 
 |  | 
 |   /* Return stringified form of instruction count.  */ | 
 |   if (record_full_list && record_full_list->type == record_full_end) | 
 |     ret = xstrdup (pulongest (record_full_list->u.end.insn_num)); | 
 |  | 
 |   if (record_debug) | 
 |     { | 
 |       if (ret) | 
 | 	gdb_printf (gdb_stdlog, | 
 | 		    "record_full_get_bookmark returns %s\n", ret); | 
 |       else | 
 | 	gdb_printf (gdb_stdlog, | 
 | 		    "record_full_get_bookmark returns NULL\n"); | 
 |     } | 
 |   return (gdb_byte *) ret; | 
 | } | 
 |  | 
 | /* "goto_bookmark" method for process record and prec over core.  */ | 
 |  | 
 | void | 
 | record_full_base_target::goto_bookmark (const gdb_byte *raw_bookmark, | 
 | 					int from_tty) | 
 | { | 
 |   const char *bookmark = (const char *) raw_bookmark; | 
 |  | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"record_full_goto_bookmark receives %s\n", bookmark); | 
 |  | 
 |   std::string name_holder; | 
 |   if (bookmark[0] == '\'' || bookmark[0] == '\"') | 
 |     { | 
 |       if (bookmark[strlen (bookmark) - 1] != bookmark[0]) | 
 | 	error (_("Unbalanced quotes: %s"), bookmark); | 
 |  | 
 |       name_holder = std::string (bookmark + 1, strlen (bookmark) - 2); | 
 |       bookmark = name_holder.c_str (); | 
 |     } | 
 |  | 
 |   record_goto (bookmark); | 
 | } | 
 |  | 
 | enum exec_direction_kind | 
 | record_full_base_target::execution_direction () | 
 | { | 
 |   return record_full_execution_dir; | 
 | } | 
 |  | 
 | /* The record_method method of target record-full.  */ | 
 |  | 
 | enum record_method | 
 | record_full_base_target::record_method (ptid_t ptid) | 
 | { | 
 |   return RECORD_METHOD_FULL; | 
 | } | 
 |  | 
 | void | 
 | record_full_base_target::info_record () | 
 | { | 
 |   struct record_full_entry *p; | 
 |  | 
 |   if (RECORD_FULL_IS_REPLAY) | 
 |     gdb_printf (_("Replay mode:\n")); | 
 |   else | 
 |     gdb_printf (_("Record mode:\n")); | 
 |  | 
 |   /* Find entry for first actual instruction in the log.  */ | 
 |   for (p = record_full_first.next; | 
 |        p != NULL && p->type != record_full_end; | 
 |        p = p->next) | 
 |     ; | 
 |  | 
 |   /* Do we have a log at all?  */ | 
 |   if (p != NULL && p->type == record_full_end) | 
 |     { | 
 |       /* Display instruction number for first instruction in the log.  */ | 
 |       gdb_printf (_("Lowest recorded instruction number is %s.\n"), | 
 | 		  pulongest (p->u.end.insn_num)); | 
 |  | 
 |       /* If in replay mode, display where we are in the log.  */ | 
 |       if (RECORD_FULL_IS_REPLAY) | 
 | 	gdb_printf (_("Current instruction number is %s.\n"), | 
 | 		    pulongest (record_full_list->u.end.insn_num)); | 
 |  | 
 |       /* Display instruction number for last instruction in the log.  */ | 
 |       gdb_printf (_("Highest recorded instruction number is %s.\n"), | 
 | 		  pulongest (record_full_insn_count)); | 
 |  | 
 |       /* Display log count.  */ | 
 |       gdb_printf (_("Log contains %u instructions.\n"), | 
 | 		  record_full_insn_num); | 
 |     } | 
 |   else | 
 |     gdb_printf (_("No instructions have been logged.\n")); | 
 |  | 
 |   /* Display max log size.  */ | 
 |   gdb_printf (_("Max logged instructions is %u.\n"), | 
 | 	      record_full_insn_max_num); | 
 | } | 
 |  | 
 | bool | 
 | record_full_base_target::supports_delete_record () | 
 | { | 
 |   return true; | 
 | } | 
 |  | 
 | /* The "delete_record" target method.  */ | 
 |  | 
 | void | 
 | record_full_base_target::delete_record () | 
 | { | 
 |   record_full_list_release_following (record_full_list); | 
 | } | 
 |  | 
 | /* The "record_is_replaying" target method.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::record_is_replaying (ptid_t ptid) | 
 | { | 
 |   return RECORD_FULL_IS_REPLAY; | 
 | } | 
 |  | 
 | /* The "record_will_replay" target method.  */ | 
 |  | 
 | bool | 
 | record_full_base_target::record_will_replay (ptid_t ptid, int dir) | 
 | { | 
 |   /* We can currently only record when executing forwards.  Should we be able | 
 |      to record when executing backwards on targets that support reverse | 
 |      execution, this needs to be changed.  */ | 
 |  | 
 |   return RECORD_FULL_IS_REPLAY || dir == EXEC_REVERSE; | 
 | } | 
 |  | 
 | /* Go to a specific entry.  */ | 
 |  | 
 | static void | 
 | record_full_goto_entry (struct record_full_entry *p) | 
 | { | 
 |   if (p == NULL) | 
 |     error (_("Target insn not found.")); | 
 |   else if (p == record_full_list) | 
 |     error (_("Already at target insn.")); | 
 |   else if (p->u.end.insn_num > record_full_list->u.end.insn_num) | 
 |     { | 
 |       gdb_printf (_("Go forward to insn number %s\n"), | 
 | 		  pulongest (p->u.end.insn_num)); | 
 |       record_full_goto_insn (p, EXEC_FORWARD); | 
 |     } | 
 |   else | 
 |     { | 
 |       gdb_printf (_("Go backward to insn number %s\n"), | 
 | 		  pulongest (p->u.end.insn_num)); | 
 |       record_full_goto_insn (p, EXEC_REVERSE); | 
 |     } | 
 |  | 
 |   registers_changed (); | 
 |   reinit_frame_cache (); | 
 |   inferior_thread ()->set_stop_pc (regcache_read_pc (get_current_regcache ())); | 
 |   print_stack_frame (get_selected_frame (NULL), 1, SRC_AND_LOC, 1); | 
 | } | 
 |  | 
 | /* The "goto_record_begin" target method.  */ | 
 |  | 
 | void | 
 | record_full_base_target::goto_record_begin () | 
 | { | 
 |   struct record_full_entry *p = NULL; | 
 |  | 
 |   for (p = &record_full_first; p != NULL; p = p->next) | 
 |     if (p->type == record_full_end) | 
 |       break; | 
 |  | 
 |   record_full_goto_entry (p); | 
 | } | 
 |  | 
 | /* The "goto_record_end" target method.  */ | 
 |  | 
 | void | 
 | record_full_base_target::goto_record_end () | 
 | { | 
 |   struct record_full_entry *p = NULL; | 
 |  | 
 |   for (p = record_full_list; p->next != NULL; p = p->next) | 
 |     ; | 
 |   for (; p!= NULL; p = p->prev) | 
 |     if (p->type == record_full_end) | 
 |       break; | 
 |  | 
 |   record_full_goto_entry (p); | 
 | } | 
 |  | 
 | /* The "goto_record" target method.  */ | 
 |  | 
 | void | 
 | record_full_base_target::goto_record (ULONGEST target_insn) | 
 | { | 
 |   struct record_full_entry *p = NULL; | 
 |  | 
 |   for (p = &record_full_first; p != NULL; p = p->next) | 
 |     if (p->type == record_full_end && p->u.end.insn_num == target_insn) | 
 |       break; | 
 |  | 
 |   record_full_goto_entry (p); | 
 | } | 
 |  | 
 | /* The "record_stop_replaying" target method.  */ | 
 |  | 
 | void | 
 | record_full_base_target::record_stop_replaying () | 
 | { | 
 |   goto_record_end (); | 
 | } | 
 |  | 
 | /* "resume" method for prec over corefile.  */ | 
 |  | 
 | void | 
 | record_full_core_target::resume (ptid_t ptid, int step, | 
 | 				 enum gdb_signal signal) | 
 | { | 
 |   record_full_resume_step = step; | 
 |   record_full_resumed = 1; | 
 |   record_full_execution_dir = ::execution_direction; | 
 | } | 
 |  | 
 | /* "kill" method for prec over corefile.  */ | 
 |  | 
 | void | 
 | record_full_core_target::kill () | 
 | { | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Process record: record_full_core_kill\n"); | 
 |  | 
 |   current_inferior ()->unpush_target (this); | 
 | } | 
 |  | 
 | /* "fetch_registers" method for prec over corefile.  */ | 
 |  | 
 | void | 
 | record_full_core_target::fetch_registers (struct regcache *regcache, | 
 | 					  int regno) | 
 | { | 
 |   if (regno < 0) | 
 |     { | 
 |       int num = gdbarch_num_regs (regcache->arch ()); | 
 |       int i; | 
 |  | 
 |       for (i = 0; i < num; i ++) | 
 | 	regcache->raw_supply (i, *record_full_core_regbuf); | 
 |     } | 
 |   else | 
 |     regcache->raw_supply (regno, *record_full_core_regbuf); | 
 | } | 
 |  | 
 | /* "prepare_to_store" method for prec over corefile.  */ | 
 |  | 
 | void | 
 | record_full_core_target::prepare_to_store (struct regcache *regcache) | 
 | { | 
 | } | 
 |  | 
 | /* "store_registers" method for prec over corefile.  */ | 
 |  | 
 | void | 
 | record_full_core_target::store_registers (struct regcache *regcache, | 
 | 					  int regno) | 
 | { | 
 |   if (record_full_gdb_operation_disable) | 
 |     record_full_core_regbuf->raw_supply (regno, *regcache); | 
 |   else | 
 |     error (_("You can't do that without a process to debug.")); | 
 | } | 
 |  | 
 | /* "xfer_partial" method for prec over corefile.  */ | 
 |  | 
 | enum target_xfer_status | 
 | record_full_core_target::xfer_partial (enum target_object object, | 
 | 				       const char *annex, gdb_byte *readbuf, | 
 | 				       const gdb_byte *writebuf, ULONGEST offset, | 
 | 				       ULONGEST len, ULONGEST *xfered_len) | 
 | { | 
 |   if (object == TARGET_OBJECT_MEMORY) | 
 |     { | 
 |       if (record_full_gdb_operation_disable || !writebuf) | 
 | 	{ | 
 | 	  for (target_section &p : record_full_core_sections) | 
 | 	    { | 
 | 	      if (offset >= p.addr) | 
 | 		{ | 
 | 		  struct record_full_core_buf_entry *entry; | 
 | 		  ULONGEST sec_offset; | 
 |  | 
 | 		  if (offset >= p.endaddr) | 
 | 		    continue; | 
 |  | 
 | 		  if (offset + len > p.endaddr) | 
 | 		    len = p.endaddr - offset; | 
 |  | 
 | 		  sec_offset = offset - p.addr; | 
 |  | 
 | 		  /* Read readbuf or write writebuf p, offset, len.  */ | 
 | 		  /* Check flags.  */ | 
 | 		  if (p.the_bfd_section->flags & SEC_CONSTRUCTOR | 
 | 		      || (p.the_bfd_section->flags & SEC_HAS_CONTENTS) == 0) | 
 | 		    { | 
 | 		      if (readbuf) | 
 | 			memset (readbuf, 0, len); | 
 |  | 
 | 		      *xfered_len = len; | 
 | 		      return TARGET_XFER_OK; | 
 | 		    } | 
 | 		  /* Get record_full_core_buf_entry.  */ | 
 | 		  for (entry = record_full_core_buf_list; entry; | 
 | 		       entry = entry->prev) | 
 | 		    if (entry->p == &p) | 
 | 		      break; | 
 | 		  if (writebuf) | 
 | 		    { | 
 | 		      if (!entry) | 
 | 			{ | 
 | 			  /* Add a new entry.  */ | 
 | 			  entry = XNEW (struct record_full_core_buf_entry); | 
 | 			  entry->p = &p; | 
 | 			  if (!bfd_malloc_and_get_section | 
 | 				(p.the_bfd_section->owner, | 
 | 				 p.the_bfd_section, | 
 | 				 &entry->buf)) | 
 | 			    { | 
 | 			      xfree (entry); | 
 | 			      return TARGET_XFER_EOF; | 
 | 			    } | 
 | 			  entry->prev = record_full_core_buf_list; | 
 | 			  record_full_core_buf_list = entry; | 
 | 			} | 
 |  | 
 | 		      memcpy (entry->buf + sec_offset, writebuf, | 
 | 			      (size_t) len); | 
 | 		    } | 
 | 		  else | 
 | 		    { | 
 | 		      if (!entry) | 
 | 			return this->beneath ()->xfer_partial (object, annex, | 
 | 							       readbuf, writebuf, | 
 | 							       offset, len, | 
 | 							       xfered_len); | 
 |  | 
 | 		      memcpy (readbuf, entry->buf + sec_offset, | 
 | 			      (size_t) len); | 
 | 		    } | 
 |  | 
 | 		  *xfered_len = len; | 
 | 		  return TARGET_XFER_OK; | 
 | 		} | 
 | 	    } | 
 |  | 
 | 	  return TARGET_XFER_E_IO; | 
 | 	} | 
 |       else | 
 | 	error (_("You can't do that without a process to debug.")); | 
 |     } | 
 |  | 
 |   return this->beneath ()->xfer_partial (object, annex, | 
 | 					 readbuf, writebuf, offset, len, | 
 | 					 xfered_len); | 
 | } | 
 |  | 
 | /* "insert_breakpoint" method for prec over corefile.  */ | 
 |  | 
 | int | 
 | record_full_core_target::insert_breakpoint (struct gdbarch *gdbarch, | 
 | 					    struct bp_target_info *bp_tgt) | 
 | { | 
 |   return 0; | 
 | } | 
 |  | 
 | /* "remove_breakpoint" method for prec over corefile.  */ | 
 |  | 
 | int | 
 | record_full_core_target::remove_breakpoint (struct gdbarch *gdbarch, | 
 | 					    struct bp_target_info *bp_tgt, | 
 | 					    enum remove_bp_reason reason) | 
 | { | 
 |   return 0; | 
 | } | 
 |  | 
 | /* "has_execution" method for prec over corefile.  */ | 
 |  | 
 | bool | 
 | record_full_core_target::has_execution (inferior *inf) | 
 | { | 
 |   return true; | 
 | } | 
 |  | 
 | /* Record log save-file format | 
 |    Version 1 (never released) | 
 |  | 
 |    Header: | 
 |      4 bytes: magic number htonl(0x20090829). | 
 |        NOTE: be sure to change whenever this file format changes! | 
 |  | 
 |    Records: | 
 |      record_full_end: | 
 |        1 byte:  record type (record_full_end, see enum record_full_type). | 
 |      record_full_reg: | 
 |        1 byte:  record type (record_full_reg, see enum record_full_type). | 
 |        8 bytes: register id (network byte order). | 
 |        MAX_REGISTER_SIZE bytes: register value. | 
 |      record_full_mem: | 
 |        1 byte:  record type (record_full_mem, see enum record_full_type). | 
 |        8 bytes: memory length (network byte order). | 
 |        8 bytes: memory address (network byte order). | 
 |        n bytes: memory value (n == memory length). | 
 |  | 
 |    Version 2 | 
 |      4 bytes: magic number netorder32(0x20091016). | 
 |        NOTE: be sure to change whenever this file format changes! | 
 |  | 
 |    Records: | 
 |      record_full_end: | 
 |        1 byte:  record type (record_full_end, see enum record_full_type). | 
 |        4 bytes: signal | 
 |        4 bytes: instruction count | 
 |      record_full_reg: | 
 |        1 byte:  record type (record_full_reg, see enum record_full_type). | 
 |        4 bytes: register id (network byte order). | 
 |        n bytes: register value (n == actual register size). | 
 | 		(eg. 4 bytes for x86 general registers). | 
 |      record_full_mem: | 
 |        1 byte:  record type (record_full_mem, see enum record_full_type). | 
 |        4 bytes: memory length (network byte order). | 
 |        8 bytes: memory address (network byte order). | 
 |        n bytes: memory value (n == memory length). | 
 |  | 
 | */ | 
 |  | 
 | /* bfdcore_read -- read bytes from a core file section.  */ | 
 |  | 
 | static inline void | 
 | bfdcore_read (bfd *obfd, asection *osec, void *buf, int len, int *offset) | 
 | { | 
 |   int ret = bfd_get_section_contents (obfd, osec, buf, *offset, len); | 
 |  | 
 |   if (ret) | 
 |     *offset += len; | 
 |   else | 
 |     error (_("Failed to read %d bytes from core file %s ('%s')."), | 
 | 	   len, bfd_get_filename (obfd), | 
 | 	   bfd_errmsg (bfd_get_error ())); | 
 | } | 
 |  | 
 | static inline uint64_t | 
 | netorder64 (uint64_t input) | 
 | { | 
 |   uint64_t ret; | 
 |  | 
 |   store_unsigned_integer ((gdb_byte *) &ret, sizeof (ret),  | 
 | 			  BFD_ENDIAN_BIG, input); | 
 |   return ret; | 
 | } | 
 |  | 
 | static inline uint32_t | 
 | netorder32 (uint32_t input) | 
 | { | 
 |   uint32_t ret; | 
 |  | 
 |   store_unsigned_integer ((gdb_byte *) &ret, sizeof (ret),  | 
 | 			  BFD_ENDIAN_BIG, input); | 
 |   return ret; | 
 | } | 
 |  | 
 | /* Restore the execution log from a core_bfd file.  */ | 
 | static void | 
 | record_full_restore (void) | 
 | { | 
 |   uint32_t magic; | 
 |   struct record_full_entry *rec; | 
 |   asection *osec; | 
 |   uint32_t osec_size; | 
 |   int bfd_offset = 0; | 
 |   struct regcache *regcache; | 
 |  | 
 |   /* We restore the execution log from the open core bfd, | 
 |      if there is one.  */ | 
 |   if (core_bfd == NULL) | 
 |     return; | 
 |  | 
 |   /* "record_full_restore" can only be called when record list is empty.  */ | 
 |   gdb_assert (record_full_first.next == NULL); | 
 |   | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Restoring recording from core file.\n"); | 
 |  | 
 |   /* Now need to find our special note section.  */ | 
 |   osec = bfd_get_section_by_name (core_bfd, "null0"); | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Find precord section %s.\n", | 
 | 		osec ? "succeeded" : "failed"); | 
 |   if (osec == NULL) | 
 |     return; | 
 |   osec_size = bfd_section_size (osec); | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "%s", bfd_section_name (osec)); | 
 |  | 
 |   /* Check the magic code.  */ | 
 |   bfdcore_read (core_bfd, osec, &magic, sizeof (magic), &bfd_offset); | 
 |   if (magic != RECORD_FULL_FILE_MAGIC) | 
 |     error (_("Version mis-match or file format error in core file %s."), | 
 | 	   bfd_get_filename (core_bfd)); | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"  Reading 4-byte magic cookie " | 
 | 		"RECORD_FULL_FILE_MAGIC (0x%s)\n", | 
 | 		phex_nz (netorder32 (magic), 4)); | 
 |  | 
 |   /* Restore the entries in recfd into record_full_arch_list_head and | 
 |      record_full_arch_list_tail.  */ | 
 |   record_full_arch_list_head = NULL; | 
 |   record_full_arch_list_tail = NULL; | 
 |   record_full_insn_num = 0; | 
 |  | 
 |   try | 
 |     { | 
 |       regcache = get_current_regcache (); | 
 |  | 
 |       while (1) | 
 | 	{ | 
 | 	  uint8_t rectype; | 
 | 	  uint32_t regnum, len, signal, count; | 
 | 	  uint64_t addr; | 
 |  | 
 | 	  /* We are finished when offset reaches osec_size.  */ | 
 | 	  if (bfd_offset >= osec_size) | 
 | 	    break; | 
 | 	  bfdcore_read (core_bfd, osec, &rectype, sizeof (rectype), &bfd_offset); | 
 |  | 
 | 	  switch (rectype) | 
 | 	    { | 
 | 	    case record_full_reg: /* reg */ | 
 | 	      /* Get register number to regnum.  */ | 
 | 	      bfdcore_read (core_bfd, osec, ®num, | 
 | 			    sizeof (regnum), &bfd_offset); | 
 | 	      regnum = netorder32 (regnum); | 
 |  | 
 | 	      rec = record_full_reg_alloc (regcache, regnum); | 
 |  | 
 | 	      /* Get val.  */ | 
 | 	      bfdcore_read (core_bfd, osec, record_full_get_loc (rec), | 
 | 			    rec->u.reg.len, &bfd_offset); | 
 |  | 
 | 	      if (record_debug) | 
 | 		gdb_printf (gdb_stdlog, | 
 | 			    "  Reading register %d (1 " | 
 | 			    "plus %lu plus %d bytes)\n", | 
 | 			    rec->u.reg.num, | 
 | 			    (unsigned long) sizeof (regnum), | 
 | 			    rec->u.reg.len); | 
 | 	      break; | 
 |  | 
 | 	    case record_full_mem: /* mem */ | 
 | 	      /* Get len.  */ | 
 | 	      bfdcore_read (core_bfd, osec, &len, | 
 | 			    sizeof (len), &bfd_offset); | 
 | 	      len = netorder32 (len); | 
 |  | 
 | 	      /* Get addr.  */ | 
 | 	      bfdcore_read (core_bfd, osec, &addr, | 
 | 			    sizeof (addr), &bfd_offset); | 
 | 	      addr = netorder64 (addr); | 
 |  | 
 | 	      rec = record_full_mem_alloc (addr, len); | 
 |  | 
 | 	      /* Get val.  */ | 
 | 	      bfdcore_read (core_bfd, osec, record_full_get_loc (rec), | 
 | 			    rec->u.mem.len, &bfd_offset); | 
 |  | 
 | 	      if (record_debug) | 
 | 		gdb_printf (gdb_stdlog, | 
 | 			    "  Reading memory %s (1 plus " | 
 | 			    "%lu plus %lu plus %d bytes)\n", | 
 | 			    paddress (get_current_arch (), | 
 | 				      rec->u.mem.addr), | 
 | 			    (unsigned long) sizeof (addr), | 
 | 			    (unsigned long) sizeof (len), | 
 | 			    rec->u.mem.len); | 
 | 	      break; | 
 |  | 
 | 	    case record_full_end: /* end */ | 
 | 	      rec = record_full_end_alloc (); | 
 | 	      record_full_insn_num ++; | 
 |  | 
 | 	      /* Get signal value.  */ | 
 | 	      bfdcore_read (core_bfd, osec, &signal, | 
 | 			    sizeof (signal), &bfd_offset); | 
 | 	      signal = netorder32 (signal); | 
 | 	      rec->u.end.sigval = (enum gdb_signal) signal; | 
 |  | 
 | 	      /* Get insn count.  */ | 
 | 	      bfdcore_read (core_bfd, osec, &count, | 
 | 			    sizeof (count), &bfd_offset); | 
 | 	      count = netorder32 (count); | 
 | 	      rec->u.end.insn_num = count; | 
 | 	      record_full_insn_count = count + 1; | 
 | 	      if (record_debug) | 
 | 		gdb_printf (gdb_stdlog, | 
 | 			    "  Reading record_full_end (1 + " | 
 | 			    "%lu + %lu bytes), offset == %s\n", | 
 | 			    (unsigned long) sizeof (signal), | 
 | 			    (unsigned long) sizeof (count), | 
 | 			    paddress (get_current_arch (), | 
 | 				      bfd_offset)); | 
 | 	      break; | 
 |  | 
 | 	    default: | 
 | 	      error (_("Bad entry type in core file %s."), | 
 | 		     bfd_get_filename (core_bfd)); | 
 | 	      break; | 
 | 	    } | 
 |  | 
 | 	  /* Add rec to record arch list.  */ | 
 | 	  record_full_arch_list_add (rec); | 
 | 	} | 
 |     } | 
 |   catch (const gdb_exception &ex) | 
 |     { | 
 |       record_full_list_release (record_full_arch_list_tail); | 
 |       throw; | 
 |     } | 
 |  | 
 |   /* Add record_full_arch_list_head to the end of record list.  */ | 
 |   record_full_first.next = record_full_arch_list_head; | 
 |   record_full_arch_list_head->prev = &record_full_first; | 
 |   record_full_arch_list_tail->next = NULL; | 
 |   record_full_list = &record_full_first; | 
 |  | 
 |   /* Update record_full_insn_max_num.  */ | 
 |   if (record_full_insn_num > record_full_insn_max_num) | 
 |     { | 
 |       record_full_insn_max_num = record_full_insn_num; | 
 |       warning (_("Auto increase record/replay buffer limit to %u."), | 
 | 	       record_full_insn_max_num); | 
 |     } | 
 |  | 
 |   /* Succeeded.  */ | 
 |   gdb_printf (_("Restored records from core file %s.\n"), | 
 | 	      bfd_get_filename (core_bfd)); | 
 |  | 
 |   print_stack_frame (get_selected_frame (NULL), 1, SRC_AND_LOC, 1); | 
 | } | 
 |  | 
 | /* bfdcore_write -- write bytes into a core file section.  */ | 
 |  | 
 | static inline void | 
 | bfdcore_write (bfd *obfd, asection *osec, void *buf, int len, int *offset) | 
 | { | 
 |   int ret = bfd_set_section_contents (obfd, osec, buf, *offset, len); | 
 |  | 
 |   if (ret) | 
 |     *offset += len; | 
 |   else | 
 |     error (_("Failed to write %d bytes to core file %s ('%s')."), | 
 | 	   len, bfd_get_filename (obfd), | 
 | 	   bfd_errmsg (bfd_get_error ())); | 
 | } | 
 |  | 
 | /* Restore the execution log from a file.  We use a modified elf | 
 |    corefile format, with an extra section for our data.  */ | 
 |  | 
 | static void | 
 | cmd_record_full_restore (const char *args, int from_tty) | 
 | { | 
 |   core_file_command (args, from_tty); | 
 |   record_full_open (args, from_tty); | 
 | } | 
 |  | 
 | /* Save the execution log to a file.  We use a modified elf corefile | 
 |    format, with an extra section for our data.  */ | 
 |  | 
 | void | 
 | record_full_base_target::save_record (const char *recfilename) | 
 | { | 
 |   struct record_full_entry *cur_record_full_list; | 
 |   uint32_t magic; | 
 |   struct regcache *regcache; | 
 |   struct gdbarch *gdbarch; | 
 |   int save_size = 0; | 
 |   asection *osec = NULL; | 
 |   int bfd_offset = 0; | 
 |  | 
 |   /* Open the save file.  */ | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, "Saving execution log to core file '%s'\n", | 
 | 		recfilename); | 
 |  | 
 |   /* Open the output file.  */ | 
 |   gdb_bfd_ref_ptr obfd (create_gcore_bfd (recfilename)); | 
 |  | 
 |   /* Arrange to remove the output file on failure.  */ | 
 |   gdb::unlinker unlink_file (recfilename); | 
 |  | 
 |   /* Save the current record entry to "cur_record_full_list".  */ | 
 |   cur_record_full_list = record_full_list; | 
 |  | 
 |   /* Get the values of regcache and gdbarch.  */ | 
 |   regcache = get_current_regcache (); | 
 |   gdbarch = regcache->arch (); | 
 |  | 
 |   /* Disable the GDB operation record.  */ | 
 |   scoped_restore restore_operation_disable | 
 |     = record_full_gdb_operation_disable_set (); | 
 |  | 
 |   /* Reverse execute to the begin of record list.  */ | 
 |   while (1) | 
 |     { | 
 |       /* Check for beginning and end of log.  */ | 
 |       if (record_full_list == &record_full_first) | 
 | 	break; | 
 |  | 
 |       record_full_exec_insn (regcache, gdbarch, record_full_list); | 
 |  | 
 |       if (record_full_list->prev) | 
 | 	record_full_list = record_full_list->prev; | 
 |     } | 
 |  | 
 |   /* Compute the size needed for the extra bfd section.  */ | 
 |   save_size = 4;	/* magic cookie */ | 
 |   for (record_full_list = record_full_first.next; record_full_list; | 
 |        record_full_list = record_full_list->next) | 
 |     switch (record_full_list->type) | 
 |       { | 
 |       case record_full_end: | 
 | 	save_size += 1 + 4 + 4; | 
 | 	break; | 
 |       case record_full_reg: | 
 | 	save_size += 1 + 4 + record_full_list->u.reg.len; | 
 | 	break; | 
 |       case record_full_mem: | 
 | 	save_size += 1 + 4 + 8 + record_full_list->u.mem.len; | 
 | 	break; | 
 |       } | 
 |  | 
 |   /* Make the new bfd section.  */ | 
 |   osec = bfd_make_section_anyway_with_flags (obfd.get (), "precord", | 
 | 					     SEC_HAS_CONTENTS | 
 | 					     | SEC_READONLY); | 
 |   if (osec == NULL) | 
 |     error (_("Failed to create 'precord' section for corefile %s: %s"), | 
 | 	   recfilename, | 
 | 	   bfd_errmsg (bfd_get_error ())); | 
 |   bfd_set_section_size (osec, save_size); | 
 |   bfd_set_section_vma (osec, 0); | 
 |   bfd_set_section_alignment (osec, 0); | 
 |  | 
 |   /* Save corefile state.  */ | 
 |   write_gcore_file (obfd.get ()); | 
 |  | 
 |   /* Write out the record log.  */ | 
 |   /* Write the magic code.  */ | 
 |   magic = RECORD_FULL_FILE_MAGIC; | 
 |   if (record_debug) | 
 |     gdb_printf (gdb_stdlog, | 
 | 		"  Writing 4-byte magic cookie " | 
 | 		"RECORD_FULL_FILE_MAGIC (0x%s)\n", | 
 | 		phex_nz (magic, 4)); | 
 |   bfdcore_write (obfd.get (), osec, &magic, sizeof (magic), &bfd_offset); | 
 |  | 
 |   /* Save the entries to recfd and forward execute to the end of | 
 |      record list.  */ | 
 |   record_full_list = &record_full_first; | 
 |   while (1) | 
 |     { | 
 |       /* Save entry.  */ | 
 |       if (record_full_list != &record_full_first) | 
 | 	{ | 
 | 	  uint8_t type; | 
 | 	  uint32_t regnum, len, signal, count; | 
 | 	  uint64_t addr; | 
 |  | 
 | 	  type = record_full_list->type; | 
 | 	  bfdcore_write (obfd.get (), osec, &type, sizeof (type), &bfd_offset); | 
 |  | 
 | 	  switch (record_full_list->type) | 
 | 	    { | 
 | 	    case record_full_reg: /* reg */ | 
 | 	      if (record_debug) | 
 | 		gdb_printf (gdb_stdlog, | 
 | 			    "  Writing register %d (1 " | 
 | 			    "plus %lu plus %d bytes)\n", | 
 | 			    record_full_list->u.reg.num, | 
 | 			    (unsigned long) sizeof (regnum), | 
 | 			    record_full_list->u.reg.len); | 
 |  | 
 | 	      /* Write regnum.  */ | 
 | 	      regnum = netorder32 (record_full_list->u.reg.num); | 
 | 	      bfdcore_write (obfd.get (), osec, ®num, | 
 | 			     sizeof (regnum), &bfd_offset); | 
 |  | 
 | 	      /* Write regval.  */ | 
 | 	      bfdcore_write (obfd.get (), osec, | 
 | 			     record_full_get_loc (record_full_list), | 
 | 			     record_full_list->u.reg.len, &bfd_offset); | 
 | 	      break; | 
 |  | 
 | 	    case record_full_mem: /* mem */ | 
 | 	      if (record_debug) | 
 | 		gdb_printf (gdb_stdlog, | 
 | 			    "  Writing memory %s (1 plus " | 
 | 			    "%lu plus %lu plus %d bytes)\n", | 
 | 			    paddress (gdbarch, | 
 | 				      record_full_list->u.mem.addr), | 
 | 			    (unsigned long) sizeof (addr), | 
 | 			    (unsigned long) sizeof (len), | 
 | 			    record_full_list->u.mem.len); | 
 |  | 
 | 	      /* Write memlen.  */ | 
 | 	      len = netorder32 (record_full_list->u.mem.len); | 
 | 	      bfdcore_write (obfd.get (), osec, &len, sizeof (len), | 
 | 			     &bfd_offset); | 
 |  | 
 | 	      /* Write memaddr.  */ | 
 | 	      addr = netorder64 (record_full_list->u.mem.addr); | 
 | 	      bfdcore_write (obfd.get (), osec, &addr,  | 
 | 			     sizeof (addr), &bfd_offset); | 
 |  | 
 | 	      /* Write memval.  */ | 
 | 	      bfdcore_write (obfd.get (), osec, | 
 | 			     record_full_get_loc (record_full_list), | 
 | 			     record_full_list->u.mem.len, &bfd_offset); | 
 | 	      break; | 
 |  | 
 | 	      case record_full_end: | 
 | 		if (record_debug) | 
 | 		  gdb_printf (gdb_stdlog, | 
 | 			      "  Writing record_full_end (1 + " | 
 | 			      "%lu + %lu bytes)\n",  | 
 | 			      (unsigned long) sizeof (signal), | 
 | 			      (unsigned long) sizeof (count)); | 
 | 		/* Write signal value.  */ | 
 | 		signal = netorder32 (record_full_list->u.end.sigval); | 
 | 		bfdcore_write (obfd.get (), osec, &signal, | 
 | 			       sizeof (signal), &bfd_offset); | 
 |  | 
 | 		/* Write insn count.  */ | 
 | 		count = netorder32 (record_full_list->u.end.insn_num); | 
 | 		bfdcore_write (obfd.get (), osec, &count, | 
 | 			       sizeof (count), &bfd_offset); | 
 | 		break; | 
 | 	    } | 
 | 	} | 
 |  | 
 |       /* Execute entry.  */ | 
 |       record_full_exec_insn (regcache, gdbarch, record_full_list); | 
 |  | 
 |       if (record_full_list->next) | 
 | 	record_full_list = record_full_list->next; | 
 |       else | 
 | 	break; | 
 |     } | 
 |  | 
 |   /* Reverse execute to cur_record_full_list.  */ | 
 |   while (1) | 
 |     { | 
 |       /* Check for beginning and end of log.  */ | 
 |       if (record_full_list == cur_record_full_list) | 
 | 	break; | 
 |  | 
 |       record_full_exec_insn (regcache, gdbarch, record_full_list); | 
 |  | 
 |       if (record_full_list->prev) | 
 | 	record_full_list = record_full_list->prev; | 
 |     } | 
 |  | 
 |   unlink_file.keep (); | 
 |  | 
 |   /* Succeeded.  */ | 
 |   gdb_printf (_("Saved core file %s with execution log.\n"), | 
 | 	      recfilename); | 
 | } | 
 |  | 
 | /* record_full_goto_insn -- rewind the record log (forward or backward, | 
 |    depending on DIR) to the given entry, changing the program state | 
 |    correspondingly.  */ | 
 |  | 
 | static void | 
 | record_full_goto_insn (struct record_full_entry *entry, | 
 | 		       enum exec_direction_kind dir) | 
 | { | 
 |   scoped_restore restore_operation_disable | 
 |     = record_full_gdb_operation_disable_set (); | 
 |   struct regcache *regcache = get_current_regcache (); | 
 |   struct gdbarch *gdbarch = regcache->arch (); | 
 |  | 
 |   /* Assume everything is valid: we will hit the entry, | 
 |      and we will not hit the end of the recording.  */ | 
 |  | 
 |   if (dir == EXEC_FORWARD) | 
 |     record_full_list = record_full_list->next; | 
 |  | 
 |   do | 
 |     { | 
 |       record_full_exec_insn (regcache, gdbarch, record_full_list); | 
 |       if (dir == EXEC_REVERSE) | 
 | 	record_full_list = record_full_list->prev; | 
 |       else | 
 | 	record_full_list = record_full_list->next; | 
 |     } while (record_full_list != entry); | 
 | } | 
 |  | 
 | /* Alias for "target record-full".  */ | 
 |  | 
 | static void | 
 | cmd_record_full_start (const char *args, int from_tty) | 
 | { | 
 |   execute_command ("target record-full", from_tty); | 
 | } | 
 |  | 
 | static void | 
 | set_record_full_insn_max_num (const char *args, int from_tty, | 
 | 			      struct cmd_list_element *c) | 
 | { | 
 |   if (record_full_insn_num > record_full_insn_max_num) | 
 |     { | 
 |       /* Count down record_full_insn_num while releasing records from list.  */ | 
 |       while (record_full_insn_num > record_full_insn_max_num) | 
 |        { | 
 | 	 record_full_list_release_first (); | 
 | 	 record_full_insn_num--; | 
 |        } | 
 |     } | 
 | } | 
 |  | 
 | void _initialize_record_full (); | 
 | void | 
 | _initialize_record_full () | 
 | { | 
 |   struct cmd_list_element *c; | 
 |  | 
 |   /* Init record_full_first.  */ | 
 |   record_full_first.prev = NULL; | 
 |   record_full_first.next = NULL; | 
 |   record_full_first.type = record_full_end; | 
 |  | 
 |   add_target (record_full_target_info, record_full_open); | 
 |   add_deprecated_target_alias (record_full_target_info, "record"); | 
 |   add_target (record_full_core_target_info, record_full_open); | 
 |  | 
 |   add_prefix_cmd ("full", class_obscure, cmd_record_full_start, | 
 | 		  _("Start full execution recording."), &record_full_cmdlist, | 
 | 		  0, &record_cmdlist); | 
 |  | 
 |   cmd_list_element *record_full_restore_cmd | 
 |     = add_cmd ("restore", class_obscure, cmd_record_full_restore, | 
 | 	       _("Restore the execution log from a file.\n\ | 
 | Argument is filename.  File must be created with 'record save'."), | 
 | 	       &record_full_cmdlist); | 
 |   set_cmd_completer (record_full_restore_cmd, filename_completer); | 
 |  | 
 |   /* Deprecate the old version without "full" prefix.  */ | 
 |   c = add_alias_cmd ("restore", record_full_restore_cmd, class_obscure, 1, | 
 | 		     &record_cmdlist); | 
 |   set_cmd_completer (c, filename_completer); | 
 |   deprecate_cmd (c, "record full restore"); | 
 |  | 
 |   add_setshow_prefix_cmd ("full", class_support, | 
 | 			  _("Set record options."), | 
 | 			  _("Show record options."), | 
 | 			  &set_record_full_cmdlist, | 
 | 			  &show_record_full_cmdlist, | 
 | 			  &set_record_cmdlist, | 
 | 			  &show_record_cmdlist); | 
 |  | 
 |   /* Record instructions number limit command.  */ | 
 |   set_show_commands set_record_full_stop_at_limit_cmds | 
 |     = add_setshow_boolean_cmd ("stop-at-limit", no_class, | 
 | 			       &record_full_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_full_cmdlist, | 
 | 			       &show_record_full_cmdlist); | 
 |  | 
 |   c = add_alias_cmd ("stop-at-limit", | 
 | 		     set_record_full_stop_at_limit_cmds.set, no_class, 1, | 
 | 		     &set_record_cmdlist); | 
 |   deprecate_cmd (c, "set record full stop-at-limit"); | 
 |  | 
 |   c = add_alias_cmd ("stop-at-limit", | 
 | 		     set_record_full_stop_at_limit_cmds.show, no_class, 1, | 
 | 		     &show_record_cmdlist); | 
 |   deprecate_cmd (c, "show record full stop-at-limit"); | 
 |  | 
 |   set_show_commands record_full_insn_number_max_cmds | 
 |     = add_setshow_uinteger_cmd ("insn-number-max", no_class, | 
 | 				&record_full_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.  A value of either \"unlimited\" or zero means no\n\ | 
 | limit.  Default is 200000."), | 
 | 				set_record_full_insn_max_num, | 
 | 				NULL, &set_record_full_cmdlist, | 
 | 				&show_record_full_cmdlist); | 
 |  | 
 |   c = add_alias_cmd ("insn-number-max", record_full_insn_number_max_cmds.set, | 
 | 		     no_class, 1, &set_record_cmdlist); | 
 |   deprecate_cmd (c, "set record full insn-number-max"); | 
 |  | 
 |   c = add_alias_cmd ("insn-number-max", record_full_insn_number_max_cmds.show, | 
 | 		     no_class, 1, &show_record_cmdlist); | 
 |   deprecate_cmd (c, "show record full insn-number-max"); | 
 |  | 
 |   set_show_commands record_full_memory_query_cmds | 
 |     = add_setshow_boolean_cmd ("memory-query", no_class, | 
 | 			       &record_full_memory_query, _("\ | 
 | Set whether query if PREC cannot record memory change of next instruction."), | 
 | 			       _("\ | 
 | Show whether query if PREC cannot record memory change of next instruction."), | 
 | 			       _("\ | 
 | Default is OFF.\n\ | 
 | When ON, query if PREC cannot record memory change of next instruction."), | 
 | 			       NULL, NULL, | 
 | 			       &set_record_full_cmdlist, | 
 | 			       &show_record_full_cmdlist); | 
 |  | 
 |   c = add_alias_cmd ("memory-query", record_full_memory_query_cmds.set, | 
 | 		     no_class, 1, &set_record_cmdlist); | 
 |   deprecate_cmd (c, "set record full memory-query"); | 
 |  | 
 |   c = add_alias_cmd ("memory-query", record_full_memory_query_cmds.show, | 
 | 		     no_class, 1,&show_record_cmdlist); | 
 |   deprecate_cmd (c, "show record full memory-query"); | 
 | } |