mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-02-05 12:53:16 +08:00
Many spots incorrectly use only spaces for indentation (for example, there are a lot of spots in ada-lang.c). I've always found it awkward when I needed to edit one of these spots: do I keep the original wrong indentation, or do I fix it? What if the lines around it are also wrong, do I fix them too? I probably don't want to fix them in the same patch, to avoid adding noise to my patch. So I propose to fix as much as possible once and for all (hopefully). One typical counter argument for this is that it makes code archeology more difficult, because git-blame will show this commit as the last change for these lines. My counter counter argument is: when git-blaming, you often need to do "blame the file at the parent commit" anyway, to go past some other refactor that touched the line you are interested in, but is not the change you are looking for. So you already need a somewhat efficient way to do this. Using some interactive tool, rather than plain git-blame, makes this trivial. For example, I use "tig blame <file>", where going back past the commit that changed the currently selected line is one keystroke. It looks like Magit in Emacs does it too (though I've never used it). Web viewers of Github and Gitlab do it too. My point is that it won't really make archeology more difficult. The other typical counter argument is that it will cause conflicts with existing patches. That's true... but it's a one time cost, and those are not conflicts that are difficult to resolve. I have also tried "git rebase --ignore-whitespace", it seems to work well. Although that will re-introduce the faulty indentation, so one needs to take care of fixing the indentation in the patch after that (which is easy). gdb/ChangeLog: * aarch64-linux-tdep.c: Fix indentation. * aarch64-ravenscar-thread.c: Fix indentation. * aarch64-tdep.c: Fix indentation. * aarch64-tdep.h: Fix indentation. * ada-lang.c: Fix indentation. * ada-lang.h: Fix indentation. * ada-tasks.c: Fix indentation. * ada-typeprint.c: Fix indentation. * ada-valprint.c: Fix indentation. * ada-varobj.c: Fix indentation. * addrmap.c: Fix indentation. * addrmap.h: Fix indentation. * agent.c: Fix indentation. * aix-thread.c: Fix indentation. * alpha-bsd-nat.c: Fix indentation. * alpha-linux-tdep.c: Fix indentation. * alpha-mdebug-tdep.c: Fix indentation. * alpha-nbsd-tdep.c: Fix indentation. * alpha-obsd-tdep.c: Fix indentation. * alpha-tdep.c: Fix indentation. * amd64-bsd-nat.c: Fix indentation. * amd64-darwin-tdep.c: Fix indentation. * amd64-linux-nat.c: Fix indentation. * amd64-linux-tdep.c: Fix indentation. * amd64-nat.c: Fix indentation. * amd64-obsd-tdep.c: Fix indentation. * amd64-tdep.c: Fix indentation. * amd64-windows-tdep.c: Fix indentation. * annotate.c: Fix indentation. * arc-tdep.c: Fix indentation. * arch-utils.c: Fix indentation. * arch/arm-get-next-pcs.c: Fix indentation. * arch/arm.c: Fix indentation. * arm-linux-nat.c: Fix indentation. * arm-linux-tdep.c: Fix indentation. * arm-nbsd-tdep.c: Fix indentation. * arm-pikeos-tdep.c: Fix indentation. * arm-tdep.c: Fix indentation. * arm-tdep.h: Fix indentation. * arm-wince-tdep.c: Fix indentation. * auto-load.c: Fix indentation. * auxv.c: Fix indentation. * avr-tdep.c: Fix indentation. * ax-gdb.c: Fix indentation. * ax-general.c: Fix indentation. * bfin-linux-tdep.c: Fix indentation. * block.c: Fix indentation. * block.h: Fix indentation. * blockframe.c: Fix indentation. * bpf-tdep.c: Fix indentation. * break-catch-sig.c: Fix indentation. * break-catch-syscall.c: Fix indentation. * break-catch-throw.c: Fix indentation. * breakpoint.c: Fix indentation. * breakpoint.h: Fix indentation. * bsd-uthread.c: Fix indentation. * btrace.c: Fix indentation. * build-id.c: Fix indentation. * buildsym-legacy.h: Fix indentation. * buildsym.c: Fix indentation. * c-typeprint.c: Fix indentation. * c-valprint.c: Fix indentation. * c-varobj.c: Fix indentation. * charset.c: Fix indentation. * cli/cli-cmds.c: Fix indentation. * cli/cli-decode.c: Fix indentation. * cli/cli-decode.h: Fix indentation. * cli/cli-script.c: Fix indentation. * cli/cli-setshow.c: Fix indentation. * coff-pe-read.c: Fix indentation. * coffread.c: Fix indentation. * compile/compile-cplus-types.c: Fix indentation. * compile/compile-object-load.c: Fix indentation. * compile/compile-object-run.c: Fix indentation. * completer.c: Fix indentation. * corefile.c: Fix indentation. * corelow.c: Fix indentation. * cp-abi.h: Fix indentation. * cp-namespace.c: Fix indentation. * cp-support.c: Fix indentation. * cp-valprint.c: Fix indentation. * cris-linux-tdep.c: Fix indentation. * cris-tdep.c: Fix indentation. * darwin-nat-info.c: Fix indentation. * darwin-nat.c: Fix indentation. * darwin-nat.h: Fix indentation. * dbxread.c: Fix indentation. * dcache.c: Fix indentation. * disasm.c: Fix indentation. * dtrace-probe.c: Fix indentation. * dwarf2/abbrev.c: Fix indentation. * dwarf2/attribute.c: Fix indentation. * dwarf2/expr.c: Fix indentation. * dwarf2/frame.c: Fix indentation. * dwarf2/index-cache.c: Fix indentation. * dwarf2/index-write.c: Fix indentation. * dwarf2/line-header.c: Fix indentation. * dwarf2/loc.c: Fix indentation. * dwarf2/macro.c: Fix indentation. * dwarf2/read.c: Fix indentation. * dwarf2/read.h: Fix indentation. * elfread.c: Fix indentation. * eval.c: Fix indentation. * event-top.c: Fix indentation. * exec.c: Fix indentation. * exec.h: Fix indentation. * expprint.c: Fix indentation. * f-lang.c: Fix indentation. * f-typeprint.c: Fix indentation. * f-valprint.c: Fix indentation. * fbsd-nat.c: Fix indentation. * fbsd-tdep.c: Fix indentation. * findvar.c: Fix indentation. * fork-child.c: Fix indentation. * frame-unwind.c: Fix indentation. * frame-unwind.h: Fix indentation. * frame.c: Fix indentation. * frv-linux-tdep.c: Fix indentation. * frv-tdep.c: Fix indentation. * frv-tdep.h: Fix indentation. * ft32-tdep.c: Fix indentation. * gcore.c: Fix indentation. * gdb_bfd.c: Fix indentation. * gdbarch.sh: Fix indentation. * gdbarch.c: Re-generate * gdbarch.h: Re-generate. * gdbcore.h: Fix indentation. * gdbthread.h: Fix indentation. * gdbtypes.c: Fix indentation. * gdbtypes.h: Fix indentation. * glibc-tdep.c: Fix indentation. * gnu-nat.c: Fix indentation. * gnu-nat.h: Fix indentation. * gnu-v2-abi.c: Fix indentation. * gnu-v3-abi.c: Fix indentation. * go32-nat.c: Fix indentation. * guile/guile-internal.h: Fix indentation. * guile/scm-cmd.c: Fix indentation. * guile/scm-frame.c: Fix indentation. * guile/scm-iterator.c: Fix indentation. * guile/scm-math.c: Fix indentation. * guile/scm-ports.c: Fix indentation. * guile/scm-pretty-print.c: Fix indentation. * guile/scm-value.c: Fix indentation. * h8300-tdep.c: Fix indentation. * hppa-linux-nat.c: Fix indentation. * hppa-linux-tdep.c: Fix indentation. * hppa-nbsd-nat.c: Fix indentation. * hppa-nbsd-tdep.c: Fix indentation. * hppa-obsd-nat.c: Fix indentation. * hppa-tdep.c: Fix indentation. * hppa-tdep.h: Fix indentation. * i386-bsd-nat.c: Fix indentation. * i386-darwin-nat.c: Fix indentation. * i386-darwin-tdep.c: Fix indentation. * i386-dicos-tdep.c: Fix indentation. * i386-gnu-nat.c: Fix indentation. * i386-linux-nat.c: Fix indentation. * i386-linux-tdep.c: Fix indentation. * i386-nto-tdep.c: Fix indentation. * i386-obsd-tdep.c: Fix indentation. * i386-sol2-nat.c: Fix indentation. * i386-tdep.c: Fix indentation. * i386-tdep.h: Fix indentation. * i386-windows-tdep.c: Fix indentation. * i387-tdep.c: Fix indentation. * i387-tdep.h: Fix indentation. * ia64-libunwind-tdep.c: Fix indentation. * ia64-libunwind-tdep.h: Fix indentation. * ia64-linux-nat.c: Fix indentation. * ia64-linux-tdep.c: Fix indentation. * ia64-tdep.c: Fix indentation. * ia64-tdep.h: Fix indentation. * ia64-vms-tdep.c: Fix indentation. * infcall.c: Fix indentation. * infcmd.c: Fix indentation. * inferior.c: Fix indentation. * infrun.c: Fix indentation. * iq2000-tdep.c: Fix indentation. * language.c: Fix indentation. * linespec.c: Fix indentation. * linux-fork.c: Fix indentation. * linux-nat.c: Fix indentation. * linux-tdep.c: Fix indentation. * linux-thread-db.c: Fix indentation. * lm32-tdep.c: Fix indentation. * m2-lang.c: Fix indentation. * m2-typeprint.c: Fix indentation. * m2-valprint.c: Fix indentation. * m32c-tdep.c: Fix indentation. * m32r-linux-tdep.c: Fix indentation. * m32r-tdep.c: Fix indentation. * m68hc11-tdep.c: Fix indentation. * m68k-bsd-nat.c: Fix indentation. * m68k-linux-nat.c: Fix indentation. * m68k-linux-tdep.c: Fix indentation. * m68k-tdep.c: Fix indentation. * machoread.c: Fix indentation. * macrocmd.c: Fix indentation. * macroexp.c: Fix indentation. * macroscope.c: Fix indentation. * macrotab.c: Fix indentation. * macrotab.h: Fix indentation. * main.c: Fix indentation. * mdebugread.c: Fix indentation. * mep-tdep.c: Fix indentation. * mi/mi-cmd-catch.c: Fix indentation. * mi/mi-cmd-disas.c: Fix indentation. * mi/mi-cmd-env.c: Fix indentation. * mi/mi-cmd-stack.c: Fix indentation. * mi/mi-cmd-var.c: Fix indentation. * mi/mi-cmds.c: Fix indentation. * mi/mi-main.c: Fix indentation. * mi/mi-parse.c: Fix indentation. * microblaze-tdep.c: Fix indentation. * minidebug.c: Fix indentation. * minsyms.c: Fix indentation. * mips-linux-nat.c: Fix indentation. * mips-linux-tdep.c: Fix indentation. * mips-nbsd-tdep.c: Fix indentation. * mips-tdep.c: Fix indentation. * mn10300-linux-tdep.c: Fix indentation. * mn10300-tdep.c: Fix indentation. * moxie-tdep.c: Fix indentation. * msp430-tdep.c: Fix indentation. * namespace.h: Fix indentation. * nat/fork-inferior.c: Fix indentation. * nat/gdb_ptrace.h: Fix indentation. * nat/linux-namespaces.c: Fix indentation. * nat/linux-osdata.c: Fix indentation. * nat/netbsd-nat.c: Fix indentation. * nat/x86-dregs.c: Fix indentation. * nbsd-nat.c: Fix indentation. * nbsd-tdep.c: Fix indentation. * nios2-linux-tdep.c: Fix indentation. * nios2-tdep.c: Fix indentation. * nto-procfs.c: Fix indentation. * nto-tdep.c: Fix indentation. * objfiles.c: Fix indentation. * objfiles.h: Fix indentation. * opencl-lang.c: Fix indentation. * or1k-tdep.c: Fix indentation. * osabi.c: Fix indentation. * osabi.h: Fix indentation. * osdata.c: Fix indentation. * p-lang.c: Fix indentation. * p-typeprint.c: Fix indentation. * p-valprint.c: Fix indentation. * parse.c: Fix indentation. * ppc-linux-nat.c: Fix indentation. * ppc-linux-tdep.c: Fix indentation. * ppc-nbsd-nat.c: Fix indentation. * ppc-nbsd-tdep.c: Fix indentation. * ppc-obsd-nat.c: Fix indentation. * ppc-ravenscar-thread.c: Fix indentation. * ppc-sysv-tdep.c: Fix indentation. * ppc64-tdep.c: Fix indentation. * printcmd.c: Fix indentation. * proc-api.c: Fix indentation. * producer.c: Fix indentation. * producer.h: Fix indentation. * prologue-value.c: Fix indentation. * prologue-value.h: Fix indentation. * psymtab.c: Fix indentation. * python/py-arch.c: Fix indentation. * python/py-bpevent.c: Fix indentation. * python/py-event.c: Fix indentation. * python/py-event.h: Fix indentation. * python/py-finishbreakpoint.c: Fix indentation. * python/py-frame.c: Fix indentation. * python/py-framefilter.c: Fix indentation. * python/py-inferior.c: Fix indentation. * python/py-infthread.c: Fix indentation. * python/py-objfile.c: Fix indentation. * python/py-prettyprint.c: Fix indentation. * python/py-registers.c: Fix indentation. * python/py-signalevent.c: Fix indentation. * python/py-stopevent.c: Fix indentation. * python/py-stopevent.h: Fix indentation. * python/py-threadevent.c: Fix indentation. * python/py-tui.c: Fix indentation. * python/py-unwind.c: Fix indentation. * python/py-value.c: Fix indentation. * python/py-xmethods.c: Fix indentation. * python/python-internal.h: Fix indentation. * python/python.c: Fix indentation. * ravenscar-thread.c: Fix indentation. * record-btrace.c: Fix indentation. * record-full.c: Fix indentation. * record.c: Fix indentation. * reggroups.c: Fix indentation. * regset.h: Fix indentation. * remote-fileio.c: Fix indentation. * remote.c: Fix indentation. * reverse.c: Fix indentation. * riscv-linux-tdep.c: Fix indentation. * riscv-ravenscar-thread.c: Fix indentation. * riscv-tdep.c: Fix indentation. * rl78-tdep.c: Fix indentation. * rs6000-aix-tdep.c: Fix indentation. * rs6000-lynx178-tdep.c: Fix indentation. * rs6000-nat.c: Fix indentation. * rs6000-tdep.c: Fix indentation. * rust-lang.c: Fix indentation. * rx-tdep.c: Fix indentation. * s12z-tdep.c: Fix indentation. * s390-linux-tdep.c: Fix indentation. * score-tdep.c: Fix indentation. * ser-base.c: Fix indentation. * ser-mingw.c: Fix indentation. * ser-uds.c: Fix indentation. * ser-unix.c: Fix indentation. * serial.c: Fix indentation. * sh-linux-tdep.c: Fix indentation. * sh-nbsd-tdep.c: Fix indentation. * sh-tdep.c: Fix indentation. * skip.c: Fix indentation. * sol-thread.c: Fix indentation. * solib-aix.c: Fix indentation. * solib-darwin.c: Fix indentation. * solib-frv.c: Fix indentation. * solib-svr4.c: Fix indentation. * solib.c: Fix indentation. * source.c: Fix indentation. * sparc-linux-tdep.c: Fix indentation. * sparc-nbsd-tdep.c: Fix indentation. * sparc-obsd-tdep.c: Fix indentation. * sparc-ravenscar-thread.c: Fix indentation. * sparc-tdep.c: Fix indentation. * sparc64-linux-tdep.c: Fix indentation. * sparc64-nbsd-tdep.c: Fix indentation. * sparc64-obsd-tdep.c: Fix indentation. * sparc64-tdep.c: Fix indentation. * stabsread.c: Fix indentation. * stack.c: Fix indentation. * stap-probe.c: Fix indentation. * stubs/ia64vms-stub.c: Fix indentation. * stubs/m32r-stub.c: Fix indentation. * stubs/m68k-stub.c: Fix indentation. * stubs/sh-stub.c: Fix indentation. * stubs/sparc-stub.c: Fix indentation. * symfile-mem.c: Fix indentation. * symfile.c: Fix indentation. * symfile.h: Fix indentation. * symmisc.c: Fix indentation. * symtab.c: Fix indentation. * symtab.h: Fix indentation. * target-float.c: Fix indentation. * target.c: Fix indentation. * target.h: Fix indentation. * tic6x-tdep.c: Fix indentation. * tilegx-linux-tdep.c: Fix indentation. * tilegx-tdep.c: Fix indentation. * top.c: Fix indentation. * tracefile-tfile.c: Fix indentation. * tracepoint.c: Fix indentation. * tui/tui-disasm.c: Fix indentation. * tui/tui-io.c: Fix indentation. * tui/tui-regs.c: Fix indentation. * tui/tui-stack.c: Fix indentation. * tui/tui-win.c: Fix indentation. * tui/tui-winsource.c: Fix indentation. * tui/tui.c: Fix indentation. * typeprint.c: Fix indentation. * ui-out.h: Fix indentation. * unittests/copy_bitwise-selftests.c: Fix indentation. * unittests/memory-map-selftests.c: Fix indentation. * utils.c: Fix indentation. * v850-tdep.c: Fix indentation. * valarith.c: Fix indentation. * valops.c: Fix indentation. * valprint.c: Fix indentation. * valprint.h: Fix indentation. * value.c: Fix indentation. * value.h: Fix indentation. * varobj.c: Fix indentation. * vax-tdep.c: Fix indentation. * windows-nat.c: Fix indentation. * windows-tdep.c: Fix indentation. * xcoffread.c: Fix indentation. * xml-syscall.c: Fix indentation. * xml-tdesc.c: Fix indentation. * xstormy16-tdep.c: Fix indentation. * xtensa-config.c: Fix indentation. * xtensa-linux-nat.c: Fix indentation. * xtensa-linux-tdep.c: Fix indentation. * xtensa-tdep.c: Fix indentation. gdbserver/ChangeLog: * ax.cc: Fix indentation. * dll.cc: Fix indentation. * inferiors.h: Fix indentation. * linux-low.cc: Fix indentation. * linux-nios2-low.cc: Fix indentation. * linux-ppc-ipa.cc: Fix indentation. * linux-ppc-low.cc: Fix indentation. * linux-x86-low.cc: Fix indentation. * linux-xtensa-low.cc: Fix indentation. * regcache.cc: Fix indentation. * server.cc: Fix indentation. * tracepoint.cc: Fix indentation. gdbsupport/ChangeLog: * common-exceptions.h: Fix indentation. * event-loop.cc: Fix indentation. * fileio.cc: Fix indentation. * filestuff.cc: Fix indentation. * gdb-dlfcn.cc: Fix indentation. * gdb_string_view.h: Fix indentation. * job-control.cc: Fix indentation. * signals.cc: Fix indentation. Change-Id: I4bad7ae6be0fbe14168b8ebafb98ffe14964a695
662 lines
17 KiB
C
662 lines
17 KiB
C
/* Darwin support for GDB, the GNU debugger.
|
|
Copyright (C) 1997-2020 Free Software Foundation, Inc.
|
|
|
|
Contributed by Apple Computer, 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 "frame.h"
|
|
#include "inferior.h"
|
|
#include "target.h"
|
|
#include "symfile.h"
|
|
#include "symtab.h"
|
|
#include "objfiles.h"
|
|
#include "gdbcmd.h"
|
|
#include "regcache.h"
|
|
#include "i386-tdep.h"
|
|
#include "i387-tdep.h"
|
|
#include "gdbarch.h"
|
|
#include "arch-utils.h"
|
|
#include "gdbcore.h"
|
|
|
|
#include "x86-nat.h"
|
|
#include "darwin-nat.h"
|
|
#include "i386-darwin-tdep.h"
|
|
|
|
#ifdef BFD64
|
|
#include "amd64-nat.h"
|
|
#include "amd64-tdep.h"
|
|
#include "amd64-darwin-tdep.h"
|
|
#endif
|
|
|
|
struct i386_darwin_nat_target final : public x86_nat_target<darwin_nat_target>
|
|
{
|
|
/* Add our register access methods. */
|
|
void fetch_registers (struct regcache *, int) override;
|
|
void store_registers (struct regcache *, int) override;
|
|
};
|
|
|
|
static struct i386_darwin_nat_target darwin_target;
|
|
|
|
/* Read register values from the inferior process.
|
|
If REGNO is -1, do this for all registers.
|
|
Otherwise, REGNO specifies which register (so we can save time). */
|
|
|
|
void
|
|
i386_darwin_nat_target::fetch_registers (struct regcache *regcache, int regno)
|
|
{
|
|
thread_t current_thread = regcache->ptid ().tid ();
|
|
int fetched = 0;
|
|
struct gdbarch *gdbarch = regcache->arch ();
|
|
|
|
#ifdef BFD64
|
|
if (gdbarch_ptr_bit (gdbarch) == 64)
|
|
{
|
|
if (regno == -1 || amd64_native_gregset_supplies_p (gdbarch, regno))
|
|
{
|
|
x86_thread_state_t gp_regs;
|
|
unsigned int gp_count = x86_THREAD_STATE_COUNT;
|
|
kern_return_t ret;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_THREAD_STATE, (thread_state_t) & gp_regs,
|
|
&gp_count);
|
|
if (ret != KERN_SUCCESS)
|
|
{
|
|
printf_unfiltered (_("Error calling thread_get_state for "
|
|
"GP registers for thread 0x%lx\n"),
|
|
(unsigned long) current_thread);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
|
|
/* Some kernels don't sanitize the values. */
|
|
gp_regs.uts.ts64.__fs &= 0xffff;
|
|
gp_regs.uts.ts64.__gs &= 0xffff;
|
|
|
|
amd64_supply_native_gregset (regcache, &gp_regs.uts, -1);
|
|
fetched++;
|
|
}
|
|
|
|
if (regno == -1 || !amd64_native_gregset_supplies_p (gdbarch, regno))
|
|
{
|
|
x86_float_state_t fp_regs;
|
|
unsigned int fp_count = x86_FLOAT_STATE_COUNT;
|
|
kern_return_t ret;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_FLOAT_STATE, (thread_state_t) & fp_regs,
|
|
&fp_count);
|
|
if (ret != KERN_SUCCESS)
|
|
{
|
|
printf_unfiltered (_("Error calling thread_get_state for "
|
|
"float registers for thread 0x%lx\n"),
|
|
(unsigned long) current_thread);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
amd64_supply_fxsave (regcache, -1, &fp_regs.ufs.fs64.__fpu_fcw);
|
|
fetched++;
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
if (regno == -1 || regno < I386_NUM_GREGS)
|
|
{
|
|
x86_thread_state32_t gp_regs;
|
|
unsigned int gp_count = x86_THREAD_STATE32_COUNT;
|
|
kern_return_t ret;
|
|
int i;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_THREAD_STATE32, (thread_state_t) &gp_regs,
|
|
&gp_count);
|
|
if (ret != KERN_SUCCESS)
|
|
{
|
|
printf_unfiltered (_("Error calling thread_get_state for "
|
|
"GP registers for thread 0x%lx\n"),
|
|
(unsigned long) current_thread);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
for (i = 0; i < I386_NUM_GREGS; i++)
|
|
regcache->raw_supply
|
|
(i, (char *) &gp_regs + i386_darwin_thread_state_reg_offset[i]);
|
|
|
|
fetched++;
|
|
}
|
|
|
|
if (regno == -1
|
|
|| (regno >= I386_ST0_REGNUM && regno < I386_SSE_NUM_REGS))
|
|
{
|
|
x86_float_state32_t fp_regs;
|
|
unsigned int fp_count = x86_FLOAT_STATE32_COUNT;
|
|
kern_return_t ret;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_FLOAT_STATE32, (thread_state_t) &fp_regs,
|
|
&fp_count);
|
|
if (ret != KERN_SUCCESS)
|
|
{
|
|
printf_unfiltered (_("Error calling thread_get_state for "
|
|
"float registers for thread 0x%lx\n"),
|
|
(unsigned long) current_thread);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
i387_supply_fxsave (regcache, -1, &fp_regs.__fpu_fcw);
|
|
fetched++;
|
|
}
|
|
}
|
|
|
|
if (! fetched)
|
|
{
|
|
warning (_("unknown register %d"), regno);
|
|
regcache->raw_supply (regno, NULL);
|
|
}
|
|
}
|
|
|
|
/* Store our register values back into the inferior.
|
|
If REGNO is -1, do this for all registers.
|
|
Otherwise, REGNO specifies which register (so we can save time). */
|
|
|
|
void
|
|
i386_darwin_nat_target::store_registers (struct regcache *regcache,
|
|
int regno)
|
|
{
|
|
thread_t current_thread = regcache->ptid ().tid ();
|
|
struct gdbarch *gdbarch = regcache->arch ();
|
|
|
|
#ifdef BFD64
|
|
if (gdbarch_ptr_bit (gdbarch) == 64)
|
|
{
|
|
if (regno == -1 || amd64_native_gregset_supplies_p (gdbarch, regno))
|
|
{
|
|
x86_thread_state_t gp_regs;
|
|
kern_return_t ret;
|
|
unsigned int gp_count = x86_THREAD_STATE_COUNT;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_THREAD_STATE, (thread_state_t) &gp_regs,
|
|
&gp_count);
|
|
MACH_CHECK_ERROR (ret);
|
|
gdb_assert (gp_regs.tsh.flavor == x86_THREAD_STATE64);
|
|
gdb_assert (gp_regs.tsh.count == x86_THREAD_STATE64_COUNT);
|
|
|
|
amd64_collect_native_gregset (regcache, &gp_regs.uts, regno);
|
|
|
|
/* Some kernels don't sanitize the values. */
|
|
gp_regs.uts.ts64.__fs &= 0xffff;
|
|
gp_regs.uts.ts64.__gs &= 0xffff;
|
|
|
|
ret = thread_set_state (current_thread, x86_THREAD_STATE,
|
|
(thread_state_t) &gp_regs,
|
|
x86_THREAD_STATE_COUNT);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
|
|
if (regno == -1 || !amd64_native_gregset_supplies_p (gdbarch, regno))
|
|
{
|
|
x86_float_state_t fp_regs;
|
|
kern_return_t ret;
|
|
unsigned int fp_count = x86_FLOAT_STATE_COUNT;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_FLOAT_STATE, (thread_state_t) & fp_regs,
|
|
&fp_count);
|
|
MACH_CHECK_ERROR (ret);
|
|
gdb_assert (fp_regs.fsh.flavor == x86_FLOAT_STATE64);
|
|
gdb_assert (fp_regs.fsh.count == x86_FLOAT_STATE64_COUNT);
|
|
|
|
amd64_collect_fxsave (regcache, regno, &fp_regs.ufs.fs64.__fpu_fcw);
|
|
|
|
ret = thread_set_state (current_thread, x86_FLOAT_STATE,
|
|
(thread_state_t) & fp_regs,
|
|
x86_FLOAT_STATE_COUNT);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
if (regno == -1 || regno < I386_NUM_GREGS)
|
|
{
|
|
x86_thread_state32_t gp_regs;
|
|
kern_return_t ret;
|
|
unsigned int gp_count = x86_THREAD_STATE32_COUNT;
|
|
int i;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_THREAD_STATE32, (thread_state_t) &gp_regs,
|
|
&gp_count);
|
|
MACH_CHECK_ERROR (ret);
|
|
|
|
for (i = 0; i < I386_NUM_GREGS; i++)
|
|
if (regno == -1 || regno == i)
|
|
regcache->raw_collect
|
|
(i, (char *) &gp_regs + i386_darwin_thread_state_reg_offset[i]);
|
|
|
|
ret = thread_set_state (current_thread, x86_THREAD_STATE32,
|
|
(thread_state_t) &gp_regs,
|
|
x86_THREAD_STATE32_COUNT);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
|
|
if (regno == -1
|
|
|| (regno >= I386_ST0_REGNUM && regno < I386_SSE_NUM_REGS))
|
|
{
|
|
x86_float_state32_t fp_regs;
|
|
unsigned int fp_count = x86_FLOAT_STATE32_COUNT;
|
|
kern_return_t ret;
|
|
|
|
ret = thread_get_state
|
|
(current_thread, x86_FLOAT_STATE32, (thread_state_t) & fp_regs,
|
|
&fp_count);
|
|
MACH_CHECK_ERROR (ret);
|
|
|
|
i387_collect_fxsave (regcache, regno, &fp_regs.__fpu_fcw);
|
|
|
|
ret = thread_set_state (current_thread, x86_FLOAT_STATE32,
|
|
(thread_state_t) &fp_regs,
|
|
x86_FLOAT_STATE32_COUNT);
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Support for debug registers, boosted mostly from i386-linux-nat.c. */
|
|
|
|
static void
|
|
i386_darwin_dr_set (int regnum, CORE_ADDR value)
|
|
{
|
|
thread_t current_thread;
|
|
x86_debug_state_t dr_regs;
|
|
kern_return_t ret;
|
|
unsigned int dr_count;
|
|
|
|
gdb_assert (regnum >= 0 && regnum <= DR_CONTROL);
|
|
|
|
current_thread = inferior_ptid.tid ();
|
|
|
|
dr_regs.dsh.flavor = x86_DEBUG_STATE;
|
|
dr_regs.dsh.count = x86_DEBUG_STATE_COUNT;
|
|
dr_count = x86_DEBUG_STATE_COUNT;
|
|
ret = thread_get_state (current_thread, x86_DEBUG_STATE,
|
|
(thread_state_t) &dr_regs, &dr_count);
|
|
MACH_CHECK_ERROR (ret);
|
|
|
|
switch (dr_regs.dsh.flavor)
|
|
{
|
|
case x86_DEBUG_STATE32:
|
|
switch (regnum)
|
|
{
|
|
case 0:
|
|
dr_regs.uds.ds32.__dr0 = value;
|
|
break;
|
|
case 1:
|
|
dr_regs.uds.ds32.__dr1 = value;
|
|
break;
|
|
case 2:
|
|
dr_regs.uds.ds32.__dr2 = value;
|
|
break;
|
|
case 3:
|
|
dr_regs.uds.ds32.__dr3 = value;
|
|
break;
|
|
case 4:
|
|
dr_regs.uds.ds32.__dr4 = value;
|
|
break;
|
|
case 5:
|
|
dr_regs.uds.ds32.__dr5 = value;
|
|
break;
|
|
case 6:
|
|
dr_regs.uds.ds32.__dr6 = value;
|
|
break;
|
|
case 7:
|
|
dr_regs.uds.ds32.__dr7 = value;
|
|
break;
|
|
}
|
|
break;
|
|
#ifdef BFD64
|
|
case x86_DEBUG_STATE64:
|
|
switch (regnum)
|
|
{
|
|
case 0:
|
|
dr_regs.uds.ds64.__dr0 = value;
|
|
break;
|
|
case 1:
|
|
dr_regs.uds.ds64.__dr1 = value;
|
|
break;
|
|
case 2:
|
|
dr_regs.uds.ds64.__dr2 = value;
|
|
break;
|
|
case 3:
|
|
dr_regs.uds.ds64.__dr3 = value;
|
|
break;
|
|
case 4:
|
|
dr_regs.uds.ds64.__dr4 = value;
|
|
break;
|
|
case 5:
|
|
dr_regs.uds.ds64.__dr5 = value;
|
|
break;
|
|
case 6:
|
|
dr_regs.uds.ds64.__dr6 = value;
|
|
break;
|
|
case 7:
|
|
dr_regs.uds.ds64.__dr7 = value;
|
|
break;
|
|
}
|
|
break;
|
|
#endif
|
|
}
|
|
|
|
ret = thread_set_state (current_thread, dr_regs.dsh.flavor,
|
|
(thread_state_t) &dr_regs.uds, dr_count);
|
|
|
|
MACH_CHECK_ERROR (ret);
|
|
}
|
|
|
|
static CORE_ADDR
|
|
i386_darwin_dr_get (int regnum)
|
|
{
|
|
thread_t current_thread;
|
|
x86_debug_state_t dr_regs;
|
|
kern_return_t ret;
|
|
unsigned int dr_count;
|
|
|
|
gdb_assert (regnum >= 0 && regnum <= DR_CONTROL);
|
|
|
|
current_thread = inferior_ptid.tid ();
|
|
|
|
dr_regs.dsh.flavor = x86_DEBUG_STATE;
|
|
dr_regs.dsh.count = x86_DEBUG_STATE_COUNT;
|
|
dr_count = x86_DEBUG_STATE_COUNT;
|
|
ret = thread_get_state (current_thread, x86_DEBUG_STATE,
|
|
(thread_state_t) &dr_regs, &dr_count);
|
|
MACH_CHECK_ERROR (ret);
|
|
|
|
switch (dr_regs.dsh.flavor)
|
|
{
|
|
case x86_DEBUG_STATE32:
|
|
switch (regnum)
|
|
{
|
|
case 0:
|
|
return dr_regs.uds.ds32.__dr0;
|
|
case 1:
|
|
return dr_regs.uds.ds32.__dr1;
|
|
case 2:
|
|
return dr_regs.uds.ds32.__dr2;
|
|
case 3:
|
|
return dr_regs.uds.ds32.__dr3;
|
|
case 4:
|
|
return dr_regs.uds.ds32.__dr4;
|
|
case 5:
|
|
return dr_regs.uds.ds32.__dr5;
|
|
case 6:
|
|
return dr_regs.uds.ds32.__dr6;
|
|
case 7:
|
|
return dr_regs.uds.ds32.__dr7;
|
|
default:
|
|
return -1;
|
|
}
|
|
break;
|
|
#ifdef BFD64
|
|
case x86_DEBUG_STATE64:
|
|
switch (regnum)
|
|
{
|
|
case 0:
|
|
return dr_regs.uds.ds64.__dr0;
|
|
case 1:
|
|
return dr_regs.uds.ds64.__dr1;
|
|
case 2:
|
|
return dr_regs.uds.ds64.__dr2;
|
|
case 3:
|
|
return dr_regs.uds.ds64.__dr3;
|
|
case 4:
|
|
return dr_regs.uds.ds64.__dr4;
|
|
case 5:
|
|
return dr_regs.uds.ds64.__dr5;
|
|
case 6:
|
|
return dr_regs.uds.ds64.__dr6;
|
|
case 7:
|
|
return dr_regs.uds.ds64.__dr7;
|
|
default:
|
|
return -1;
|
|
}
|
|
break;
|
|
#endif
|
|
default:
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
static void
|
|
i386_darwin_dr_set_control (unsigned long control)
|
|
{
|
|
i386_darwin_dr_set (DR_CONTROL, control);
|
|
}
|
|
|
|
static void
|
|
i386_darwin_dr_set_addr (int regnum, CORE_ADDR addr)
|
|
{
|
|
gdb_assert (regnum >= 0 && regnum <= DR_LASTADDR - DR_FIRSTADDR);
|
|
|
|
i386_darwin_dr_set (DR_FIRSTADDR + regnum, addr);
|
|
}
|
|
|
|
static CORE_ADDR
|
|
i386_darwin_dr_get_addr (int regnum)
|
|
{
|
|
return i386_darwin_dr_get (regnum);
|
|
}
|
|
|
|
static unsigned long
|
|
i386_darwin_dr_get_status (void)
|
|
{
|
|
return i386_darwin_dr_get (DR_STATUS);
|
|
}
|
|
|
|
static unsigned long
|
|
i386_darwin_dr_get_control (void)
|
|
{
|
|
return i386_darwin_dr_get (DR_CONTROL);
|
|
}
|
|
|
|
void
|
|
darwin_check_osabi (darwin_inferior *inf, thread_t thread)
|
|
{
|
|
if (gdbarch_osabi (target_gdbarch ()) == GDB_OSABI_UNKNOWN)
|
|
{
|
|
/* Attaching to a process. Let's figure out what kind it is. */
|
|
x86_thread_state_t gp_regs;
|
|
struct gdbarch_info info;
|
|
unsigned int gp_count = x86_THREAD_STATE_COUNT;
|
|
kern_return_t ret;
|
|
|
|
ret = thread_get_state (thread, x86_THREAD_STATE,
|
|
(thread_state_t) &gp_regs, &gp_count);
|
|
if (ret != KERN_SUCCESS)
|
|
{
|
|
MACH_CHECK_ERROR (ret);
|
|
return;
|
|
}
|
|
|
|
gdbarch_info_init (&info);
|
|
gdbarch_info_fill (&info);
|
|
info.byte_order = gdbarch_byte_order (target_gdbarch ());
|
|
info.osabi = GDB_OSABI_DARWIN;
|
|
if (gp_regs.tsh.flavor == x86_THREAD_STATE64)
|
|
info.bfd_arch_info = bfd_lookup_arch (bfd_arch_i386,
|
|
bfd_mach_x86_64);
|
|
else
|
|
info.bfd_arch_info = bfd_lookup_arch (bfd_arch_i386,
|
|
bfd_mach_i386_i386);
|
|
gdbarch_update_p (info);
|
|
}
|
|
}
|
|
|
|
#define X86_EFLAGS_T 0x100UL
|
|
|
|
/* Returning from a signal trampoline is done by calling a
|
|
special system call (sigreturn). This system call
|
|
restores the registers that were saved when the signal was
|
|
raised, including %eflags/%rflags. That means that single-stepping
|
|
won't work. Instead, we'll have to modify the signal context
|
|
that's about to be restored, and set the trace flag there. */
|
|
|
|
static int
|
|
i386_darwin_sstep_at_sigreturn (x86_thread_state_t *regs)
|
|
{
|
|
enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ());
|
|
static const gdb_byte darwin_syscall[] = { 0xcd, 0x80 }; /* int 0x80 */
|
|
gdb_byte buf[sizeof (darwin_syscall)];
|
|
|
|
/* Check if PC is at a sigreturn system call. */
|
|
if (target_read_memory (regs->uts.ts32.__eip, buf, sizeof (buf)) == 0
|
|
&& memcmp (buf, darwin_syscall, sizeof (darwin_syscall)) == 0
|
|
&& regs->uts.ts32.__eax == 0xb8 /* SYS_sigreturn */)
|
|
{
|
|
ULONGEST uctx_addr;
|
|
ULONGEST mctx_addr;
|
|
ULONGEST flags_addr;
|
|
unsigned int eflags;
|
|
|
|
uctx_addr = read_memory_unsigned_integer
|
|
(regs->uts.ts32.__esp + 4, 4, byte_order);
|
|
mctx_addr = read_memory_unsigned_integer
|
|
(uctx_addr + 28, 4, byte_order);
|
|
|
|
flags_addr = mctx_addr + 12 + 9 * 4;
|
|
read_memory (flags_addr, (gdb_byte *) &eflags, 4);
|
|
eflags |= X86_EFLAGS_T;
|
|
write_memory (flags_addr, (gdb_byte *) &eflags, 4);
|
|
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#ifdef BFD64
|
|
static int
|
|
amd64_darwin_sstep_at_sigreturn (x86_thread_state_t *regs)
|
|
{
|
|
enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ());
|
|
static const gdb_byte darwin_syscall[] = { 0x0f, 0x05 }; /* syscall */
|
|
gdb_byte buf[sizeof (darwin_syscall)];
|
|
|
|
/* Check if PC is at a sigreturn system call. */
|
|
if (target_read_memory (regs->uts.ts64.__rip, buf, sizeof (buf)) == 0
|
|
&& memcmp (buf, darwin_syscall, sizeof (darwin_syscall)) == 0
|
|
&& (regs->uts.ts64.__rax & 0xffffffff) == 0x20000b8 /* SYS_sigreturn */)
|
|
{
|
|
ULONGEST mctx_addr;
|
|
ULONGEST flags_addr;
|
|
unsigned int rflags;
|
|
|
|
mctx_addr = read_memory_unsigned_integer
|
|
(regs->uts.ts64.__rdi + 48, 8, byte_order);
|
|
flags_addr = mctx_addr + 16 + 17 * 8;
|
|
|
|
/* AMD64 is little endian. */
|
|
read_memory (flags_addr, (gdb_byte *) &rflags, 4);
|
|
rflags |= X86_EFLAGS_T;
|
|
write_memory (flags_addr, (gdb_byte *) &rflags, 4);
|
|
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
void
|
|
darwin_set_sstep (thread_t thread, int enable)
|
|
{
|
|
x86_thread_state_t regs;
|
|
unsigned int count = x86_THREAD_STATE_COUNT;
|
|
kern_return_t kret;
|
|
|
|
kret = thread_get_state (thread, x86_THREAD_STATE,
|
|
(thread_state_t) ®s, &count);
|
|
if (kret != KERN_SUCCESS)
|
|
{
|
|
printf_unfiltered (_("darwin_set_sstep: error %x, thread=%x\n"),
|
|
kret, thread);
|
|
return;
|
|
}
|
|
|
|
switch (regs.tsh.flavor)
|
|
{
|
|
case x86_THREAD_STATE32:
|
|
{
|
|
__uint32_t bit = enable ? X86_EFLAGS_T : 0;
|
|
|
|
if (enable && i386_darwin_sstep_at_sigreturn (®s))
|
|
return;
|
|
if ((regs.uts.ts32.__eflags & X86_EFLAGS_T) == bit)
|
|
return;
|
|
regs.uts.ts32.__eflags
|
|
= (regs.uts.ts32.__eflags & ~X86_EFLAGS_T) | bit;
|
|
kret = thread_set_state (thread, x86_THREAD_STATE,
|
|
(thread_state_t) ®s, count);
|
|
MACH_CHECK_ERROR (kret);
|
|
}
|
|
break;
|
|
#ifdef BFD64
|
|
case x86_THREAD_STATE64:
|
|
{
|
|
__uint64_t bit = enable ? X86_EFLAGS_T : 0;
|
|
|
|
if (enable && amd64_darwin_sstep_at_sigreturn (®s))
|
|
return;
|
|
if ((regs.uts.ts64.__rflags & X86_EFLAGS_T) == bit)
|
|
return;
|
|
regs.uts.ts64.__rflags
|
|
= (regs.uts.ts64.__rflags & ~X86_EFLAGS_T) | bit;
|
|
kret = thread_set_state (thread, x86_THREAD_STATE,
|
|
(thread_state_t) ®s, count);
|
|
MACH_CHECK_ERROR (kret);
|
|
}
|
|
break;
|
|
#endif
|
|
default:
|
|
error (_("darwin_set_sstep: unknown flavour: %d"), regs.tsh.flavor);
|
|
}
|
|
}
|
|
|
|
void _initialize_i386_darwin_nat ();
|
|
void
|
|
_initialize_i386_darwin_nat ()
|
|
{
|
|
#ifdef BFD64
|
|
amd64_native_gregset64_reg_offset = amd64_darwin_thread_state_reg_offset;
|
|
amd64_native_gregset64_num_regs = amd64_darwin_thread_state_num_regs;
|
|
amd64_native_gregset32_reg_offset = i386_darwin_thread_state_reg_offset;
|
|
amd64_native_gregset32_num_regs = i386_darwin_thread_state_num_regs;
|
|
#endif
|
|
|
|
x86_dr_low.set_control = i386_darwin_dr_set_control;
|
|
x86_dr_low.set_addr = i386_darwin_dr_set_addr;
|
|
x86_dr_low.get_addr = i386_darwin_dr_get_addr;
|
|
x86_dr_low.get_status = i386_darwin_dr_get_status;
|
|
x86_dr_low.get_control = i386_darwin_dr_get_control;
|
|
|
|
/* Let's assume that the kernel is 64 bits iff the executable is. */
|
|
#ifdef __x86_64__
|
|
x86_set_debug_register_length (8);
|
|
#else
|
|
x86_set_debug_register_length (4);
|
|
#endif
|
|
|
|
add_inf_child_target (&darwin_target);
|
|
}
|