gdb/x86-linux-nat.c - gdb
Global variables defined
Data types defined
Functions defined
Macros defined
Source code
- #include "defs.h"
- #include "inferior.h"
- #include "elf/common.h"
- #include "gdb_proc_service.h"
- #include <sys/ptrace.h>
- #include <sys/user.h>
- #include <sys/procfs.h>
- #include <sys/uio.h>
- #include "x86-nat.h"
- #include "linux-nat.h"
- #ifndef __x86_64__
- #include "i386-linux-nat.h"
- #endif
- #include "x86-linux-nat.h"
- #include "i386-linux-tdep.h"
- #ifdef __x86_64__
- #include "amd64-linux-tdep.h"
- #endif
- #include "x86-xstate.h"
- #include "nat/linux-btrace.h"
- struct arch_lwp_info
- {
-
- int debug_registers_changed;
- };
- int have_ptrace_getregset = -1;
- static unsigned long
- x86_linux_dr_get (ptid_t ptid, int regnum)
- {
- int tid;
- unsigned long value;
- gdb_assert (ptid_lwp_p (ptid));
- tid = ptid_get_lwp (ptid);
- errno = 0;
- value = ptrace (PTRACE_PEEKUSER, tid,
- offsetof (struct user, u_debugreg[regnum]), 0);
- if (errno != 0)
- perror_with_name (_("Couldn't read debug register"));
- return value;
- }
- static void
- x86_linux_dr_set (ptid_t ptid, int regnum, unsigned long value)
- {
- int tid;
- gdb_assert (ptid_lwp_p (ptid));
- tid = ptid_get_lwp (ptid);
- errno = 0;
- ptrace (PTRACE_POKEUSER, tid,
- offsetof (struct user, u_debugreg[regnum]), value);
- if (errno != 0)
- perror_with_name (_("Couldn't write debug register"));
- }
- static CORE_ADDR
- x86_linux_dr_get_addr (int regnum)
- {
-
- gdb_assert (DR_FIRSTADDR <= regnum && regnum <= DR_LASTADDR);
- return x86_linux_dr_get (inferior_ptid, regnum);
- }
- static unsigned long
- x86_linux_dr_get_control (void)
- {
- return x86_linux_dr_get (inferior_ptid, DR_CONTROL);
- }
- static unsigned long
- x86_linux_dr_get_status (void)
- {
- return x86_linux_dr_get (inferior_ptid, DR_STATUS);
- }
- static int
- update_debug_registers_callback (struct lwp_info *lwp, void *arg)
- {
- if (lwp->arch_private == NULL)
- lwp->arch_private = XCNEW (struct arch_lwp_info);
-
- lwp->arch_private->debug_registers_changed = 1;
-
- if (!lwp->stopped)
- linux_stop_lwp (lwp);
-
- return 0;
- }
- static void
- x86_linux_dr_set_control (unsigned long control)
- {
- ptid_t pid_ptid = pid_to_ptid (ptid_get_pid (inferior_ptid));
- iterate_over_lwps (pid_ptid, update_debug_registers_callback, NULL);
- }
- static void
- x86_linux_dr_set_addr (int regnum, CORE_ADDR addr)
- {
- ptid_t pid_ptid = pid_to_ptid (ptid_get_pid (inferior_ptid));
- gdb_assert (regnum >= 0 && regnum <= DR_LASTADDR - DR_FIRSTADDR);
- iterate_over_lwps (pid_ptid, update_debug_registers_callback, NULL);
- }
- static void
- x86_linux_prepare_to_resume (struct lwp_info *lwp)
- {
- int clear_status = 0;
-
- if (lwp->arch_private == NULL)
- return;
- if (lwp->arch_private->debug_registers_changed)
- {
- struct x86_debug_reg_state *state
- = x86_debug_reg_state (ptid_get_pid (lwp->ptid));
- int i;
-
-
- x86_linux_dr_set (lwp->ptid, DR_CONTROL, 0);
- ALL_DEBUG_ADDRESS_REGISTERS (i)
- if (state->dr_ref_count[i] > 0)
- {
- x86_linux_dr_set (lwp->ptid, i, state->dr_mirror[i]);
-
- clear_status = 1;
- }
-
- if (state->dr_control_mirror != 0)
- x86_linux_dr_set (lwp->ptid, DR_CONTROL, state->dr_control_mirror);
- lwp->arch_private->debug_registers_changed = 0;
- }
- if (clear_status || lwp->stop_reason == LWP_STOPPED_BY_WATCHPOINT)
- x86_linux_dr_set (lwp->ptid, DR_STATUS, 0);
- }
- static void
- x86_linux_new_thread (struct lwp_info *lp)
- {
- struct arch_lwp_info *info = XCNEW (struct arch_lwp_info);
- info->debug_registers_changed = 1;
- lp->arch_private = info;
- }
- static void
- x86_linux_new_fork (struct lwp_info *parent, pid_t child_pid)
- {
- pid_t parent_pid;
- struct x86_debug_reg_state *parent_state;
- struct x86_debug_reg_state *child_state;
-
- if (parent->arch_private == NULL)
- return;
-
- parent_pid = ptid_get_pid (parent->ptid);
- parent_state = x86_debug_reg_state (parent_pid);
- child_state = x86_debug_reg_state (child_pid);
- *child_state = *parent_state;
- }
- static void (*super_post_startup_inferior) (struct target_ops *self,
- ptid_t ptid);
- static void
- x86_linux_child_post_startup_inferior (struct target_ops *self, ptid_t ptid)
- {
- x86_cleanup_dregs ();
- super_post_startup_inferior (self, ptid);
- }
- #ifdef __x86_64__
- #define AMD64_LINUX_USER64_CS 0x33
- #define AMD64_LINUX_X32_DS 0x2b
- #endif
- static const struct target_desc *
- x86_linux_read_description (struct target_ops *ops)
- {
- int tid;
- int is_64bit = 0;
- #ifdef __x86_64__
- int is_x32;
- #endif
- static uint64_t xcr0;
- uint64_t xcr0_features_bits;
-
- tid = ptid_get_lwp (inferior_ptid);
- if (tid == 0)
- tid = ptid_get_pid (inferior_ptid);
- #ifdef __x86_64__
- {
- unsigned long cs;
- unsigned long ds;
-
- errno = 0;
- cs = ptrace (PTRACE_PEEKUSER, tid,
- offsetof (struct user_regs_struct, cs), 0);
- if (errno != 0)
- perror_with_name (_("Couldn't get CS register"));
- is_64bit = cs == AMD64_LINUX_USER64_CS;
-
- errno = 0;
- ds = ptrace (PTRACE_PEEKUSER, tid,
- offsetof (struct user_regs_struct, ds), 0);
- if (errno != 0)
- perror_with_name (_("Couldn't get DS register"));
- is_x32 = ds == AMD64_LINUX_X32_DS;
- if (sizeof (void *) == 4 && is_64bit && !is_x32)
- error (_("Can't debug 64-bit process with 32-bit GDB"));
- }
- #elif HAVE_PTRACE_GETFPXREGS
- if (have_ptrace_getfpxregs == -1)
- {
- elf_fpxregset_t fpxregs;
- if (ptrace (PTRACE_GETFPXREGS, tid, 0, (int) &fpxregs) < 0)
- {
- have_ptrace_getfpxregs = 0;
- have_ptrace_getregset = 0;
- return tdesc_i386_mmx_linux;
- }
- }
- #endif
- if (have_ptrace_getregset == -1)
- {
- uint64_t xstateregs[(X86_XSTATE_SSE_SIZE / sizeof (uint64_t))];
- struct iovec iov;
- iov.iov_base = xstateregs;
- iov.iov_len = sizeof (xstateregs);
-
- if (ptrace (PTRACE_GETREGSET, tid,
- (unsigned int) NT_X86_XSTATE, &iov) < 0)
- have_ptrace_getregset = 0;
- else
- {
- have_ptrace_getregset = 1;
-
- xcr0 = xstateregs[(I386_LINUX_XSAVE_XCR0_OFFSET
- / sizeof (uint64_t))];
- }
- }
-
- if (have_ptrace_getregset)
- xcr0_features_bits = xcr0 & X86_XSTATE_ALL_MASK;
- else
- xcr0_features_bits = 0;
- if (is_64bit)
- {
- #ifdef __x86_64__
- switch (xcr0_features_bits)
- {
- case X86_XSTATE_MPX_AVX512_MASK:
- case X86_XSTATE_AVX512_MASK:
- if (is_x32)
- return tdesc_x32_avx512_linux;
- else
- return tdesc_amd64_avx512_linux;
- case X86_XSTATE_MPX_MASK:
- if (is_x32)
- return tdesc_x32_avx_linux;
- else
- return tdesc_amd64_mpx_linux;
- case X86_XSTATE_AVX_MASK:
- if (is_x32)
- return tdesc_x32_avx_linux;
- else
- return tdesc_amd64_avx_linux;
- default:
- if (is_x32)
- return tdesc_x32_linux;
- else
- return tdesc_amd64_linux;
- }
- #endif
- }
- else
- {
- switch (xcr0_features_bits)
- {
- case X86_XSTATE_MPX_AVX512_MASK:
- case X86_XSTATE_AVX512_MASK:
- return tdesc_i386_avx512_linux;
- case X86_XSTATE_MPX_MASK:
- return tdesc_i386_mpx_linux;
- case X86_XSTATE_AVX_MASK:
- return tdesc_i386_avx_linux;
- default:
- return tdesc_i386_linux;
- }
- }
- gdb_assert_not_reached ("failed to return tdesc");
- }
- static struct btrace_target_info *
- x86_linux_enable_btrace (struct target_ops *self, ptid_t ptid)
- {
- struct btrace_target_info *tinfo;
- struct gdbarch *gdbarch;
- errno = 0;
- tinfo = linux_enable_btrace (ptid);
- if (tinfo == NULL)
- error (_("Could not enable branch tracing for %s: %s."),
- target_pid_to_str (ptid), safe_strerror (errno));
-
- gdbarch = target_thread_architecture (ptid);
- tinfo->ptr_bits = gdbarch_ptr_bit (gdbarch);
- return tinfo;
- }
- static void
- x86_linux_disable_btrace (struct target_ops *self,
- struct btrace_target_info *tinfo)
- {
- enum btrace_error errcode = linux_disable_btrace (tinfo);
- if (errcode != BTRACE_ERR_NONE)
- error (_("Could not disable branch tracing."));
- }
- static void
- x86_linux_teardown_btrace (struct target_ops *self,
- struct btrace_target_info *tinfo)
- {
-
- linux_disable_btrace (tinfo);
- }
- static enum btrace_error
- x86_linux_read_btrace (struct target_ops *self,
- VEC (btrace_block_s) **data,
- struct btrace_target_info *btinfo,
- enum btrace_read_type type)
- {
- return linux_read_btrace (data, btinfo, type);
- }
- ps_err_e
- x86_linux_get_thread_area (pid_t pid, void *addr, unsigned int *base_addr)
- {
-
- unsigned int desc[4];
-
- gdb_assert (sizeof (int) == 4);
- #ifndef PTRACE_GET_THREAD_AREA
- #define PTRACE_GET_THREAD_AREA 25
- #endif
- if (ptrace (PTRACE_GET_THREAD_AREA, pid, addr, &desc) < 0)
- return PS_ERR;
- *base_addr = desc[1];
- return PS_OK;
- }
- struct target_ops *
- x86_linux_create_target (void)
- {
-
- struct target_ops *t = linux_target ();
-
- x86_use_watchpoints (t);
- x86_dr_low.set_control = x86_linux_dr_set_control;
- x86_dr_low.set_addr = x86_linux_dr_set_addr;
- x86_dr_low.get_addr = x86_linux_dr_get_addr;
- x86_dr_low.get_status = x86_linux_dr_get_status;
- x86_dr_low.get_control = x86_linux_dr_get_control;
- x86_set_debug_register_length (sizeof (void *));
-
- super_post_startup_inferior = t->to_post_startup_inferior;
- t->to_post_startup_inferior = x86_linux_child_post_startup_inferior;
-
- t->to_read_description = x86_linux_read_description;
-
- t->to_supports_btrace = linux_supports_btrace;
- t->to_enable_btrace = x86_linux_enable_btrace;
- t->to_disable_btrace = x86_linux_disable_btrace;
- t->to_teardown_btrace = x86_linux_teardown_btrace;
- t->to_read_btrace = x86_linux_read_btrace;
- return t;
- }
- void
- x86_linux_add_target (struct target_ops *t)
- {
- linux_nat_add_target (t);
- linux_nat_set_new_thread (t, x86_linux_new_thread);
- linux_nat_set_new_fork (t, x86_linux_new_fork);
- linux_nat_set_forget_process (t, x86_forget_process);
- linux_nat_set_prepare_to_resume (t, x86_linux_prepare_to_resume);
- }