d9f719f1ad
Currently, to open a target, with "target TARGET_NAME", GDB finds the target_ops instance with "TARGET_NAME" as short name, and then calls its target_ops::open virtual method. In reality, there's no actual target/name lookup, a pointer to the target_ops object was associated with the "target TARGET_NAME" command at add_target time (when GDB is initialized), as the command's context. This creates a chicken and egg situation. Consider the case of wanting to open multiple remote connections. We want to be able to have one remote target_ops instance per connection, but, if we're not connected yet, so we don't yet have an instance to call target->open() on... This patch fixes this by separating out common info about a target_ops to a separate structure (shortname, longname, doc), and changing the add_target routine to take a reference to such an object instead of a pointer to a target_ops, and a pointer to a factory function that is responsible to open an instance of the corresponding target when the user types "target TARGET_NAME". -extern void add_target (struct target_ops *); +extern void add_target (const target_info &info, target_open_ftype *func); I.e. this factory function replaces the target_ops::open virtual method. For static/singleton targets, nothing changes, the target_open_ftype function pushes the global target_ops instance on the target stack. At target_close time, the connection is tor down, but the global target_ops object remains live. However, targets that support being open multiple times will make their target_open_ftype routine allocate a new target_ops instance on the heap [e.g., new remote_target()], and push that on the stack. At target_close time, the new object is destroyed (by the target_ops::close virtual method). Both the core target and the remote targets will support being open multiple times (others could/should too, but those were my stopping point), but not in this patch yet. We need to get rid of more globals first before that'd be useful. Native targets are somewhat special, given find_default_run_target & friends. Those routines also expect to return a target_ops pointer, even before we've open the target. However, we'll never need more than one instance of the native target, so we can assume/require that native targets are global/simpletons, and have the backends register a pointer to the native target_ops. Since all native targets inherit inf_child_target, we can centralize that registration. See add_inf_child_target, get_native_target/set_native_target and find_default_run_target. gdb/ChangeLog: 2018-05-02 Pedro Alves <palves@redhat.com> * aarch64-fbsd-nat.c (_initialize_aarch64_fbsd_nat): Use add_inf_child_target. * aarch64-linux-nat.c (_initialize_aarch64_linux_nat): Use add_inf_child_target. * aix-thread.c (aix_thread_target_info): New. (aix_thread_target) <shortname, longname, doc>: Delete. <info>: New. * alpha-bsd-nat.c (_initialize_alphabsd_nat): Use add_inf_child_target. * alpha-linux-nat.c (_initialize_alpha_linux_nat): Use add_inf_child_target. * amd64-fbsd-nat.c (_initialize_amd64fbsd_nat): Use add_inf_child_target. * amd64-linux-nat.c (_initialize_amd64_linux_nat): Use add_inf_child_target. * amd64-nbsd-nat.c (_initialize_amd64nbsd_nat): Use add_inf_child_target. * amd64-obsd-nat.c (_initialize_amd64obsd_nat): Use add_inf_child_target. * arm-fbsd-nat.c (_initialize_arm_fbsd_nat): Use add_inf_child_target. * arm-linux-nat.c (_initialize_arm_linux_nat): Use add_inf_child_target. * arm-nbsd-nat.c (_initialize_arm_netbsd_nat): Use add_inf_child_target. * bfd-target.c (target_bfd_target_info): New. (target_bfd) <shortname, longname, doc>: Delete. <info>: New. * bsd-kvm.c (bsd_kvm_target_info): New. (bsd_kvm_target) <shortname, longname, doc>: Delete. <info>: New. (bsd_kvm_target::open): Rename to ... (bsd_kvm_target_open): ... this. Adjust. * bsd-uthread.c (bsd_uthread_target_info): New. (bsd_uthread_target) <shortname, longname, doc>: Delete. <info>: New. * corefile.c (core_file_command): Adjust. * corelow.c (core_target_info): New. (core_target) <shortname, longname, doc>: Delete. <info>: New. (core_target::open): Rename to ... (core_target_open): ... this. Adjust. * ctf.c (ctf_target_info): New. (ctf_target) <shortname, longname, doc>: Delete. <info>: New. (ctf_target::open): Rename to ... (ctf_target_open): ... this. (_initialize_ctf): Adjust. * exec.c (exec_target_info): New. (exec_target) <shortname, longname, doc>: Delete. <info>: New. (exec_target::open): Rename to ... (exec_target_open): ... this. * gdbcore.h (core_target_open): Declare. * go32-nat.c (_initialize_go32_nat): Use add_inf_child_target. * hppa-linux-nat.c (_initialize_hppa_linux_nat): Use add_inf_child_target. * hppa-nbsd-nat.c (_initialize_hppanbsd_nat): Use add_inf_child_target. * hppa-obsd-nat.c (_initialize_hppaobsd_nat): Use add_inf_child_target. * i386-darwin-nat.c (_initialize_i386_darwin_nat): Use add_inf_child_target. * i386-fbsd-nat.c (_initialize_i386fbsd_nat): Use add_inf_child_target. * i386-gnu-nat.c (_initialize_i386gnu_nat): Use add_inf_child_target. * i386-linux-nat.c (_initialize_i386_linux_nat): Use add_inf_child_target. * i386-nbsd-nat.c (_initialize_i386nbsd_nat): Use add_inf_child_target. * i386-obsd-nat.c (_initialize_i386obsd_nat): Use add_inf_child_target. * ia64-linux-nat.c (_initialize_ia64_linux_nat): Use add_inf_child_target. * inf-child.c (inf_child_target_info): New. (inf_child_target::info): New. (inf_child_open_target): Remove 'target' parameter. Use get_native_target instead. (inf_child_target::open): Delete. (add_inf_child_target): New. * inf-child.h (inf_child_target) <shortname, longname, doc, open>: Delete. <info>: New. (add_inf_child_target): Declare. (inf_child_open_target): Declare. * linux-thread-db.c (thread_db_target_info): New. (thread_db_target) <shortname, longname, doc>: Delete. <info>: New. * m32r-linux-nat.c (_initialize_m32r_linux_nat): Use add_inf_child_target. * m68k-bsd-nat.c (_initialize_m68kbsd_nat): Use add_inf_child_target. * m68k-linux-nat.c (_initialize_m68k_linux_nat): Use add_inf_child_target. * m88k-bsd-nat.c (_initialize_m88kbsd_nat): Use add_inf_child_target. * make-target-delegates (print_class): Adjust. * mips-fbsd-nat.c (_initialize_mips_fbsd_nat): Use add_inf_child_target. * mips-linux-nat.c (_initialize_mips_linux_nat): Use add_inf_child_target. * mips-nbsd-nat.c (_initialize_mipsnbsd_nat): Use add_inf_child_target. * mips64-obsd-nat.c (_initialize_mips64obsd_nat): Use add_inf_child_target. * nto-procfs.c (nto_native_target_info): New. (nto_procfs_target_native) <shortname, longname, doc>: Delete. <info>: New. (nto_procfs_target_info): New. (nto_procfs_target_procfs) <shortname, longname, doc>: Delete. <info>: New. (init_procfs_targets): Adjust. * ppc-fbsd-nat.c (_initialize_ppcfbsd_nat): Use add_inf_child_target. * ppc-linux-nat.c (_initialize_ppc_linux_nat): Use add_inf_child_target. * ppc-nbsd-nat.c (_initialize_ppcnbsd_nat): Use add_inf_child_target. * ppc-obsd-nat.c (_initialize_ppcobsd_nat): Use add_inf_child_target. * ravenscar-thread.c (ravenscar_target_info): New. (ravenscar_thread_target) <shortname, longname, doc>: Delete. <info>: New. * record-btrace.c (record_btrace_target_info): (record_btrace_target) <shortname, longname, doc>: Delete. <info>: New. (record_btrace_target::open): Rename to ... (record_btrace_target_open): ... this. Adjust. * record-full.c (record_longname, record_doc): New. (record_full_base_target) <shortname, longname, doc>: Delete. <info>: New. (record_full_target_info): New. (record_full_target): <shortname>: Delete. <info>: New. (record_full_core_open_1, record_full_open_1): Update comments. (record_full_base_target::open): Rename to ... (record_full_open): ... this. (cmd_record_full_restore): Update. (_initialize_record_full): Update. * remote-sim.c (remote_sim_target_info): New. (gdbsim_target) <shortname, longname, doc>: Delete. <info>: New. (gdbsim_target::open): Rename to ... (gdbsim_target_open): ... this. (_initialize_remote_sim): Adjust. * remote.c (remote_doc): New. (remote_target_info): New. (remote_target) <shortname, longname, doc>: Delete. <info>: New. (extended_remote_target_info): New. (extended_remote_target) <shortname, longname, doc>: Delete. <info>: New. (remote_target::open_1): Make static. Adjust. * rs6000-nat.c (_initialize_rs6000_nat): Use add_inf_child_target. * s390-linux-nat.c (_initialize_s390_nat): Use add_inf_child_target. * sh-nbsd-nat.c (_initialize_shnbsd_nat): Use add_inf_child_target. * sol-thread.c (thread_db_target_info): New. (sol_thread_target) <shortname, longname, doc>: Delete. <info>: New. * sparc-linux-nat.c (_initialize_sparc_linux_nat): Use add_inf_child_target. * sparc-nbsd-nat.c (_initialize_sparcnbsd_nat): Use add_inf_child_target. * sparc64-fbsd-nat.c (_initialize_sparc64fbsd_nat): Use add_inf_child_target. * sparc64-linux-nat.c (_initialize_sparc64_linux_nat): Use add_inf_child_target. * sparc64-nbsd-nat.c (_initialize_sparc64nbsd_nat): Use add_inf_child_target. * sparc64-obsd-nat.c (_initialize_sparc64obsd_nat): Use add_inf_child_target. * spu-linux-nat.c (_initialize_spu_nat): Use add_inf_child_target. * spu-multiarch.c (spu_multiarch_target_info): New. (spu_multiarch_target) <shortname, longname, doc>: Delete. <info>: New. * target-delegates.c: Regenerate. * target.c: Include <unordered_map>. (target_ops_p): Delete. (DEF_VEC_P(target_ops_p)): Delete. (target_factories): New. (test_target_info): New. (test_target_ops::info): New. (open_target): Adjust to use target_factories. (add_target_with_completer): Rename to ... (add_target): ... this. Change prototype. Register target_info and open callback in target_factories. Register target_info in command context instead of target_ops. (add_target): Delete old implementation. (add_deprecated_target_alias): Change prototype. Adjust. (the_native_target): New. (set_native_target, get_native_target): New. (find_default_run_target): Use the_native_target. (find_attach_target, find_run_target): Simplify. (target_ops::open): Delete. (dummy_target_info): New. (dummy_target::shortname, dummy_target::longname) (dummy_target::doc): Delete. (dummy_target::info): New. (debug_target::shortname, debug_target::longname) (debug_target::doc): Delete. (debug_target::info): New. * target.h (struct target_info): New. (target_ops::~target_ops): Add comment. (target_ops::info): New. (target_ops::shortname, target_ops::longname, target_ops::doc): No longer virtual. Implement in terms of target_info. (set_native_target, get_native_target): Declare. (target_open_ftype): New. (add_target, add_target_with_completer) (add_deprecated_target_alias): Change prototype. (test_target) <shortname, longname, doc>: Delete. <info>: New. * tilegx-linux-nat.c (_initialize_tile_linux_nat): Use add_inf_child_target. * tracefile-tfile.c (tfile_target_info): New. (tfile_target) <shortname, longname, doc>: Delete. <info>: New. (tfile_target::open): Rename to ... (tfile_target_open): ... this. (_initialize_tracefile_tfile): Adjust. * vax-bsd-nat.c (_initialize_vaxbsd_nat): Use add_inf_child_target. * windows-nat.c (_initialize_windows_nat): Use add_inf_child_target. * xtensa-linux-nat.c (_initialize_xtensa_linux_nat): Use add_inf_child_target.
445 lines
12 KiB
C
445 lines
12 KiB
C
/* Low level interface to i386 running the GNU Hurd.
|
||
|
||
Copyright (C) 1992-2018 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/>. */
|
||
|
||
/* Mach/Hurd headers are not yet ready for C++ compilation. */
|
||
extern "C"
|
||
{
|
||
#include <mach.h>
|
||
#include <mach_error.h>
|
||
#include <mach/message.h>
|
||
#include <mach/exception.h>
|
||
}
|
||
|
||
#include "defs.h"
|
||
#include "x86-nat.h"
|
||
#include "inferior.h"
|
||
#include "floatformat.h"
|
||
#include "regcache.h"
|
||
|
||
#include "i386-tdep.h"
|
||
|
||
#include "gnu-nat.h"
|
||
#include "inf-child.h"
|
||
#include "i387-tdep.h"
|
||
|
||
/* Offset to the thread_state_t location where REG is stored. */
|
||
#define REG_OFFSET(reg) offsetof (struct i386_thread_state, reg)
|
||
|
||
/* At REG_OFFSET[N] is the offset to the thread_state_t location where
|
||
the GDB register N is stored. */
|
||
static int reg_offset[] =
|
||
{
|
||
REG_OFFSET (eax), REG_OFFSET (ecx), REG_OFFSET (edx), REG_OFFSET (ebx),
|
||
REG_OFFSET (uesp), REG_OFFSET (ebp), REG_OFFSET (esi), REG_OFFSET (edi),
|
||
REG_OFFSET (eip), REG_OFFSET (efl), REG_OFFSET (cs), REG_OFFSET (ss),
|
||
REG_OFFSET (ds), REG_OFFSET (es), REG_OFFSET (fs), REG_OFFSET (gs)
|
||
};
|
||
|
||
#define REG_ADDR(state, regnum) ((char *)(state) + reg_offset[regnum])
|
||
|
||
|
||
|
||
/* The i386 GNU Hurd target. */
|
||
|
||
#ifdef i386_DEBUG_STATE
|
||
using gnu_base_target = x86_nat_target<gnu_nat_target>;
|
||
#else
|
||
using gnu_base_target = gnu_nat_target;
|
||
#endif
|
||
|
||
struct i386_gnu_nat_target final : public gnu_base_target
|
||
{
|
||
void fetch_registers (struct regcache *, int) override;
|
||
void store_registers (struct regcache *, int) override;
|
||
};
|
||
|
||
static i386_gnu_nat_target the_i386_gnu_nat_target;
|
||
|
||
/* Get the whole floating-point state of THREAD and record the values
|
||
of the corresponding (pseudo) registers. */
|
||
|
||
static void
|
||
fetch_fpregs (struct regcache *regcache, struct proc *thread)
|
||
{
|
||
mach_msg_type_number_t count = i386_FLOAT_STATE_COUNT;
|
||
struct i386_float_state state;
|
||
kern_return_t err;
|
||
|
||
err = thread_get_state (thread->port, i386_FLOAT_STATE,
|
||
(thread_state_t) &state, &count);
|
||
if (err)
|
||
{
|
||
warning (_("Couldn't fetch floating-point state from %s"),
|
||
proc_string (thread));
|
||
return;
|
||
}
|
||
|
||
if (!state.initialized)
|
||
{
|
||
/* The floating-point state isn't initialized. */
|
||
i387_supply_fsave (regcache, -1, NULL);
|
||
}
|
||
else
|
||
{
|
||
/* Supply the floating-point registers. */
|
||
i387_supply_fsave (regcache, -1, state.hw_state);
|
||
}
|
||
}
|
||
|
||
/* Fetch register REGNO, or all regs if REGNO is -1. */
|
||
static void
|
||
gnu_fetch_registers (struct target_ops *ops,
|
||
struct regcache *regcache, int regno)
|
||
{
|
||
struct proc *thread;
|
||
ptid_t ptid = regcache_get_ptid (regcache);
|
||
|
||
/* Make sure we know about new threads. */
|
||
inf_update_procs (gnu_current_inf);
|
||
|
||
thread = inf_tid_to_thread (gnu_current_inf, ptid_get_lwp (ptid));
|
||
if (!thread)
|
||
error (_("Can't fetch registers from thread %s: No such thread"),
|
||
target_pid_to_str (ptid));
|
||
|
||
if (regno < I386_NUM_GREGS || regno == -1)
|
||
{
|
||
thread_state_t state;
|
||
|
||
/* This does the dirty work for us. */
|
||
state = proc_get_state (thread, 0);
|
||
if (!state)
|
||
{
|
||
warning (_("Couldn't fetch registers from %s"),
|
||
proc_string (thread));
|
||
return;
|
||
}
|
||
|
||
if (regno == -1)
|
||
{
|
||
int i;
|
||
|
||
proc_debug (thread, "fetching all register");
|
||
|
||
for (i = 0; i < I386_NUM_GREGS; i++)
|
||
regcache_raw_supply (regcache, i, REG_ADDR (state, i));
|
||
thread->fetched_regs = ~0;
|
||
}
|
||
else
|
||
{
|
||
proc_debug (thread, "fetching register %s",
|
||
gdbarch_register_name (regcache->arch (),
|
||
regno));
|
||
|
||
regcache_raw_supply (regcache, regno,
|
||
REG_ADDR (state, regno));
|
||
thread->fetched_regs |= (1 << regno);
|
||
}
|
||
}
|
||
|
||
if (regno >= I386_NUM_GREGS || regno == -1)
|
||
{
|
||
proc_debug (thread, "fetching floating-point registers");
|
||
|
||
fetch_fpregs (regcache, thread);
|
||
}
|
||
}
|
||
|
||
|
||
/* Store the whole floating-point state into THREAD using information
|
||
from the corresponding (pseudo) registers. */
|
||
static void
|
||
store_fpregs (const struct regcache *regcache, struct proc *thread, int regno)
|
||
{
|
||
mach_msg_type_number_t count = i386_FLOAT_STATE_COUNT;
|
||
struct i386_float_state state;
|
||
kern_return_t err;
|
||
|
||
err = thread_get_state (thread->port, i386_FLOAT_STATE,
|
||
(thread_state_t) &state, &count);
|
||
if (err)
|
||
{
|
||
warning (_("Couldn't fetch floating-point state from %s"),
|
||
proc_string (thread));
|
||
return;
|
||
}
|
||
|
||
/* FIXME: kettenis/2001-07-15: Is this right? Should we somehow
|
||
take into account DEPRECATED_REGISTER_VALID like the old code did? */
|
||
i387_collect_fsave (regcache, regno, state.hw_state);
|
||
|
||
err = thread_set_state (thread->port, i386_FLOAT_STATE,
|
||
(thread_state_t) &state, i386_FLOAT_STATE_COUNT);
|
||
if (err)
|
||
{
|
||
warning (_("Couldn't store floating-point state into %s"),
|
||
proc_string (thread));
|
||
return;
|
||
}
|
||
}
|
||
|
||
/* Store at least register REGNO, or all regs if REGNO == -1. */
|
||
static void
|
||
gnu_store_registers (struct target_ops *ops,
|
||
struct regcache *regcache, int regno)
|
||
{
|
||
struct proc *thread;
|
||
struct gdbarch *gdbarch = regcache->arch ();
|
||
ptid_t ptid = regcache_get_ptid (regcache);
|
||
|
||
/* Make sure we know about new threads. */
|
||
inf_update_procs (gnu_current_inf);
|
||
|
||
thread = inf_tid_to_thread (gnu_current_inf, ptid_get_lwp (ptid));
|
||
if (!thread)
|
||
error (_("Couldn't store registers into thread %s: No such thread"),
|
||
target_pid_to_str (ptid));
|
||
|
||
if (regno < I386_NUM_GREGS || regno == -1)
|
||
{
|
||
thread_state_t state;
|
||
thread_state_data_t old_state;
|
||
int was_aborted = thread->aborted;
|
||
int was_valid = thread->state_valid;
|
||
int trace;
|
||
|
||
if (!was_aborted && was_valid)
|
||
memcpy (&old_state, &thread->state, sizeof (old_state));
|
||
|
||
state = proc_get_state (thread, 1);
|
||
if (!state)
|
||
{
|
||
warning (_("Couldn't store registers into %s"),
|
||
proc_string (thread));
|
||
return;
|
||
}
|
||
|
||
/* Save the T bit. We might try to restore the %eflags register
|
||
below, but changing the T bit would seriously confuse GDB. */
|
||
trace = ((struct i386_thread_state *)state)->efl & 0x100;
|
||
|
||
if (!was_aborted && was_valid)
|
||
/* See which registers have changed after aborting the thread. */
|
||
{
|
||
int check_regno;
|
||
|
||
for (check_regno = 0; check_regno < I386_NUM_GREGS; check_regno++)
|
||
if ((thread->fetched_regs & (1 << check_regno))
|
||
&& memcpy (REG_ADDR (&old_state, check_regno),
|
||
REG_ADDR (state, check_regno),
|
||
register_size (gdbarch, check_regno)))
|
||
/* Register CHECK_REGNO has changed! Ack! */
|
||
{
|
||
warning (_("Register %s changed after the thread was aborted"),
|
||
gdbarch_register_name (gdbarch, check_regno));
|
||
if (regno >= 0 && regno != check_regno)
|
||
/* Update GDB's copy of the register. */
|
||
regcache_raw_supply (regcache, check_regno,
|
||
REG_ADDR (state, check_regno));
|
||
else
|
||
warning (_("... also writing this register! "
|
||
"Suspicious..."));
|
||
}
|
||
}
|
||
|
||
if (regno == -1)
|
||
{
|
||
int i;
|
||
|
||
proc_debug (thread, "storing all registers");
|
||
|
||
for (i = 0; i < I386_NUM_GREGS; i++)
|
||
if (REG_VALID == regcache_register_status (regcache, i))
|
||
regcache_raw_collect (regcache, i, REG_ADDR (state, i));
|
||
}
|
||
else
|
||
{
|
||
proc_debug (thread, "storing register %s",
|
||
gdbarch_register_name (gdbarch, regno));
|
||
|
||
gdb_assert (REG_VALID == regcache_register_status (regcache, regno));
|
||
regcache_raw_collect (regcache, regno, REG_ADDR (state, regno));
|
||
}
|
||
|
||
/* Restore the T bit. */
|
||
((struct i386_thread_state *)state)->efl &= ~0x100;
|
||
((struct i386_thread_state *)state)->efl |= trace;
|
||
}
|
||
|
||
if (regno >= I386_NUM_GREGS || regno == -1)
|
||
{
|
||
proc_debug (thread, "storing floating-point registers");
|
||
|
||
store_fpregs (regcache, thread, regno);
|
||
}
|
||
}
|
||
|
||
|
||
/* Support for debug registers. */
|
||
|
||
#ifdef i386_DEBUG_STATE
|
||
/* Get debug registers for thread THREAD. */
|
||
|
||
static void
|
||
i386_gnu_dr_get (struct i386_debug_state *regs, struct proc *thread)
|
||
{
|
||
mach_msg_type_number_t count = i386_DEBUG_STATE_COUNT;
|
||
kern_return_t err;
|
||
|
||
err = thread_get_state (thread->port, i386_DEBUG_STATE,
|
||
(thread_state_t) regs, &count);
|
||
if (err != 0 || count != i386_DEBUG_STATE_COUNT)
|
||
warning (_("Couldn't fetch debug state from %s"),
|
||
proc_string (thread));
|
||
}
|
||
|
||
/* Set debug registers for thread THREAD. */
|
||
|
||
static void
|
||
i386_gnu_dr_set (const struct i386_debug_state *regs, struct proc *thread)
|
||
{
|
||
kern_return_t err;
|
||
|
||
err = thread_set_state (thread->port, i386_DEBUG_STATE,
|
||
(thread_state_t) regs, i386_DEBUG_STATE_COUNT);
|
||
if (err != 0)
|
||
warning (_("Couldn't store debug state into %s"),
|
||
proc_string (thread));
|
||
}
|
||
|
||
/* Set DR_CONTROL in THREAD. */
|
||
|
||
static void
|
||
i386_gnu_dr_set_control_one (struct proc *thread, void *arg)
|
||
{
|
||
unsigned long *control = (unsigned long *) arg;
|
||
struct i386_debug_state regs;
|
||
|
||
i386_gnu_dr_get (®s, thread);
|
||
regs.dr[DR_CONTROL] = *control;
|
||
i386_gnu_dr_set (®s, thread);
|
||
}
|
||
|
||
/* Set DR_CONTROL to CONTROL in all threads. */
|
||
|
||
static void
|
||
i386_gnu_dr_set_control (unsigned long control)
|
||
{
|
||
inf_update_procs (gnu_current_inf);
|
||
inf_threads (gnu_current_inf, i386_gnu_dr_set_control_one, &control);
|
||
}
|
||
|
||
/* Parameters to set a debugging address. */
|
||
|
||
struct reg_addr
|
||
{
|
||
int regnum; /* Register number (zero based). */
|
||
CORE_ADDR addr; /* Address. */
|
||
};
|
||
|
||
/* Set address REGNUM (zero based) to ADDR in THREAD. */
|
||
|
||
static void
|
||
i386_gnu_dr_set_addr_one (struct proc *thread, void *arg)
|
||
{
|
||
struct reg_addr *reg_addr = (struct reg_addr *) arg;
|
||
struct i386_debug_state regs;
|
||
|
||
i386_gnu_dr_get (®s, thread);
|
||
regs.dr[reg_addr->regnum] = reg_addr->addr;
|
||
i386_gnu_dr_set (®s, thread);
|
||
}
|
||
|
||
/* Set address REGNUM (zero based) to ADDR in all threads. */
|
||
|
||
static void
|
||
i386_gnu_dr_set_addr (int regnum, CORE_ADDR addr)
|
||
{
|
||
struct reg_addr reg_addr;
|
||
|
||
gdb_assert (DR_FIRSTADDR <= regnum && regnum <= DR_LASTADDR);
|
||
|
||
reg_addr.regnum = regnum;
|
||
reg_addr.addr = addr;
|
||
|
||
inf_update_procs (gnu_current_inf);
|
||
inf_threads (gnu_current_inf, i386_gnu_dr_set_addr_one, ®_addr);
|
||
}
|
||
|
||
/* Get debug register REGNUM value from only the one LWP of PTID. */
|
||
|
||
static unsigned long
|
||
i386_gnu_dr_get_reg (ptid_t ptid, int regnum)
|
||
{
|
||
struct i386_debug_state regs;
|
||
struct proc *thread;
|
||
|
||
/* Make sure we know about new threads. */
|
||
inf_update_procs (gnu_current_inf);
|
||
|
||
thread = inf_tid_to_thread (gnu_current_inf, ptid_get_lwp (ptid));
|
||
i386_gnu_dr_get (®s, thread);
|
||
|
||
return regs.dr[regnum];
|
||
}
|
||
|
||
/* Return the inferior's debug register REGNUM. */
|
||
|
||
static CORE_ADDR
|
||
i386_gnu_dr_get_addr (int regnum)
|
||
{
|
||
gdb_assert (DR_FIRSTADDR <= regnum && regnum <= DR_LASTADDR);
|
||
|
||
return i386_gnu_dr_get_reg (inferior_ptid, regnum);
|
||
}
|
||
|
||
/* Get DR_STATUS from only the one thread of INFERIOR_PTID. */
|
||
|
||
static unsigned long
|
||
i386_gnu_dr_get_status (void)
|
||
{
|
||
return i386_gnu_dr_get_reg (inferior_ptid, DR_STATUS);
|
||
}
|
||
|
||
/* Return the inferior's DR7 debug control register. */
|
||
|
||
static unsigned long
|
||
i386_gnu_dr_get_control (void)
|
||
{
|
||
return i386_gnu_dr_get_reg (inferior_ptid, DR_CONTROL);
|
||
}
|
||
#endif /* i386_DEBUG_STATE */
|
||
|
||
void
|
||
_initialize_i386gnu_nat (void)
|
||
{
|
||
#ifdef i386_DEBUG_STATE
|
||
x86_dr_low.set_control = i386_gnu_dr_set_control;
|
||
gdb_assert (DR_FIRSTADDR == 0 && DR_LASTADDR < i386_DEBUG_STATE_COUNT);
|
||
x86_dr_low.set_addr = i386_gnu_dr_set_addr;
|
||
x86_dr_low.get_addr = i386_gnu_dr_get_addr;
|
||
x86_dr_low.get_status = i386_gnu_dr_get_status;
|
||
x86_dr_low.get_control = i386_gnu_dr_get_control;
|
||
x86_set_debug_register_length (4);
|
||
#endif /* i386_DEBUG_STATE */
|
||
|
||
/* Register the target. */
|
||
add_inf_child_target (&the_i386_gnu_nat_target);
|
||
}
|