mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2024-12-15 04:31:49 +08:00
268a13a5a3
This is the next patch in the ongoing series to move gdbsever to the top level. This patch just renames the "common" directory. The idea is to do this move in two parts: first rename the directory (this patch), then move the directory to the top. This approach makes the patches a bit more tractable. I chose the name "gdbsupport" for the directory. However, as this patch was largely written by sed, we could pick a new name without too much difficulty. Tested by the buildbot. gdb/ChangeLog 2019-07-09 Tom Tromey <tom@tromey.com> * contrib/ari/gdb_ari.sh: Change common to gdbsupport. * configure: Rebuild. * configure.ac: Change common to gdbsupport. * gdbsupport: Rename from common. * acinclude.m4: Change common to gdbsupport. * Makefile.in (CONFIG_SRC_SUBDIR, COMMON_SFILES) (HFILES_NO_SRCDIR, stamp-version, ALLDEPFILES): Change common to gdbsupport. * aarch64-tdep.c, ada-lang.c, ada-lang.h, agent.c, alloc.c, amd64-darwin-tdep.c, amd64-dicos-tdep.c, amd64-fbsd-nat.c, amd64-fbsd-tdep.c, amd64-linux-nat.c, amd64-linux-tdep.c, amd64-nbsd-tdep.c, amd64-obsd-tdep.c, amd64-sol2-tdep.c, amd64-tdep.c, amd64-windows-tdep.c, arch-utils.c, arch/aarch64-insn.c, arch/aarch64.c, arch/aarch64.h, arch/amd64.c, arch/amd64.h, arch/arm-get-next-pcs.c, arch/arm-linux.c, arch/arm.c, arch/i386.c, arch/i386.h, arch/ppc-linux-common.c, arch/riscv.c, arch/riscv.h, arch/tic6x.c, arm-tdep.c, auto-load.c, auxv.c, ax-gdb.c, ax-general.c, ax.h, breakpoint.c, breakpoint.h, btrace.c, btrace.h, build-id.c, build-id.h, c-lang.h, charset.c, charset.h, cli/cli-cmds.c, cli/cli-cmds.h, cli/cli-decode.c, cli/cli-dump.c, cli/cli-option.h, cli/cli-script.c, coff-pe-read.c, command.h, compile/compile-c-support.c, compile/compile-c.h, compile/compile-cplus-symbols.c, compile/compile-cplus-types.c, compile/compile-cplus.h, compile/compile-loc2c.c, compile/compile.c, completer.c, completer.h, contrib/ari/gdb_ari.sh, corefile.c, corelow.c, cp-support.c, cp-support.h, cp-valprint.c, csky-tdep.c, ctf.c, darwin-nat.c, debug.c, defs.h, disasm-selftests.c, disasm.c, disasm.h, dtrace-probe.c, dwarf-index-cache.c, dwarf-index-cache.h, dwarf-index-write.c, dwarf2-frame.c, dwarf2expr.c, dwarf2loc.c, dwarf2read.c, event-loop.c, event-top.c, exceptions.c, exec.c, extension.h, fbsd-nat.c, features/aarch64-core.c, features/aarch64-fpu.c, features/aarch64-pauth.c, features/aarch64-sve.c, features/i386/32bit-avx.c, features/i386/32bit-avx512.c, features/i386/32bit-core.c, features/i386/32bit-linux.c, features/i386/32bit-mpx.c, features/i386/32bit-pkeys.c, features/i386/32bit-segments.c, features/i386/32bit-sse.c, features/i386/64bit-avx.c, features/i386/64bit-avx512.c, features/i386/64bit-core.c, features/i386/64bit-linux.c, features/i386/64bit-mpx.c, features/i386/64bit-pkeys.c, features/i386/64bit-segments.c, features/i386/64bit-sse.c, features/i386/x32-core.c, features/riscv/32bit-cpu.c, features/riscv/32bit-csr.c, features/riscv/32bit-fpu.c, features/riscv/64bit-cpu.c, features/riscv/64bit-csr.c, features/riscv/64bit-fpu.c, features/tic6x-c6xp.c, features/tic6x-core.c, features/tic6x-gp.c, filename-seen-cache.h, findcmd.c, findvar.c, fork-child.c, gcore.c, gdb_bfd.c, gdb_bfd.h, gdb_proc_service.h, gdb_regex.c, gdb_select.h, gdb_usleep.c, gdbarch-selftests.c, gdbthread.h, gdbtypes.h, gnu-nat.c, go32-nat.c, guile/guile.c, guile/scm-ports.c, guile/scm-safe-call.c, guile/scm-type.c, i386-fbsd-nat.c, i386-fbsd-tdep.c, i386-go32-tdep.c, i386-linux-nat.c, i386-linux-tdep.c, i386-tdep.c, i387-tdep.c, ia64-libunwind-tdep.c, ia64-linux-nat.c, inf-child.c, inf-ptrace.c, infcall.c, infcall.h, infcmd.c, inferior-iter.h, inferior.c, inferior.h, inflow.c, inflow.h, infrun.c, infrun.h, inline-frame.c, language.h, linespec.c, linux-fork.c, linux-nat.c, linux-tdep.c, linux-thread-db.c, location.c, machoread.c, macrotab.h, main.c, maint.c, maint.h, memattr.c, memrange.h, mi/mi-cmd-break.h, mi/mi-cmd-env.c, mi/mi-cmd-stack.c, mi/mi-cmd-var.c, mi/mi-interp.c, mi/mi-main.c, mi/mi-parse.h, minsyms.c, mips-linux-tdep.c, namespace.h, nat/aarch64-linux-hw-point.c, nat/aarch64-linux-hw-point.h, nat/aarch64-linux.c, nat/aarch64-sve-linux-ptrace.c, nat/amd64-linux-siginfo.c, nat/fork-inferior.c, nat/linux-btrace.c, nat/linux-btrace.h, nat/linux-namespaces.c, nat/linux-nat.h, nat/linux-osdata.c, nat/linux-personality.c, nat/linux-procfs.c, nat/linux-ptrace.c, nat/linux-ptrace.h, nat/linux-waitpid.c, nat/mips-linux-watch.c, nat/mips-linux-watch.h, nat/ppc-linux.c, nat/x86-dregs.c, nat/x86-dregs.h, nat/x86-linux-dregs.c, nat/x86-linux.c, nto-procfs.c, nto-tdep.c, objfile-flags.h, objfiles.c, objfiles.h, obsd-nat.c, observable.h, osdata.c, p-valprint.c, parse.c, parser-defs.h, ppc-linux-nat.c, printcmd.c, probe.c, proc-api.c, procfs.c, producer.c, progspace.h, psymtab.h, python/py-framefilter.c, python/py-inferior.c, python/py-ref.h, python/py-type.c, python/python.c, record-btrace.c, record-full.c, record.c, record.h, regcache-dump.c, regcache.c, regcache.h, remote-fileio.c, remote-fileio.h, remote-sim.c, remote.c, riscv-tdep.c, rs6000-aix-tdep.c, rust-exp.y, s12z-tdep.c, selftest-arch.c, ser-base.c, ser-event.c, ser-pipe.c, ser-tcp.c, ser-unix.c, skip.c, solib-aix.c, solib-target.c, solib.c, source-cache.c, source.c, source.h, sparc-nat.c, spu-linux-nat.c, stack.c, stap-probe.c, symfile-add-flags.h, symfile.c, symfile.h, symtab.c, symtab.h, target-descriptions.c, target-descriptions.h, target-memory.c, target.c, target.h, target/waitstatus.c, target/waitstatus.h, thread-iter.h, thread.c, tilegx-tdep.c, top.c, top.h, tracefile-tfile.c, tracefile.c, tracepoint.c, tracepoint.h, tui/tui-io.c, ui-file.c, ui-out.h, unittests/array-view-selftests.c, unittests/child-path-selftests.c, unittests/cli-utils-selftests.c, unittests/common-utils-selftests.c, unittests/copy_bitwise-selftests.c, unittests/environ-selftests.c, unittests/format_pieces-selftests.c, unittests/function-view-selftests.c, unittests/lookup_name_info-selftests.c, unittests/memory-map-selftests.c, unittests/memrange-selftests.c, unittests/mkdir-recursive-selftests.c, unittests/observable-selftests.c, unittests/offset-type-selftests.c, unittests/optional-selftests.c, unittests/parse-connection-spec-selftests.c, unittests/ptid-selftests.c, unittests/rsp-low-selftests.c, unittests/scoped_fd-selftests.c, unittests/scoped_mmap-selftests.c, unittests/scoped_restore-selftests.c, unittests/string_view-selftests.c, unittests/style-selftests.c, unittests/tracepoint-selftests.c, unittests/unpack-selftests.c, unittests/utils-selftests.c, unittests/xml-utils-selftests.c, utils.c, utils.h, valarith.c, valops.c, valprint.c, value.c, value.h, varobj.c, varobj.h, windows-nat.c, x86-linux-nat.c, xml-support.c, xml-support.h, xml-tdesc.h, xstormy16-tdep.c, xtensa-linux-nat.c, dwarf2read.h: Change common to gdbsupport. gdb/gdbserver/ChangeLog 2019-07-09 Tom Tromey <tom@tromey.com> * configure: Rebuild. * configure.ac: Change common to gdbsupport. * acinclude.m4: Change common to gdbsupport. * Makefile.in (SFILES, OBS, GDBREPLAY_OBS, IPA_OBJS) (version-generated.c, gdbsupport/%-ipa.o, gdbsupport/%.o): Change common to gdbsupport. * ax.c, event-loop.c, fork-child.c, gdb_proc_service.h, gdbreplay.c, gdbthread.h, hostio-errno.c, hostio.c, i387-fp.c, inferiors.c, inferiors.h, linux-aarch64-tdesc-selftest.c, linux-amd64-ipa.c, linux-i386-ipa.c, linux-low.c, linux-tic6x-low.c, linux-x86-low.c, linux-x86-tdesc-selftest.c, linux-x86-tdesc.c, lynx-i386-low.c, lynx-low.c, mem-break.h, nto-x86-low.c, regcache.c, regcache.h, remote-utils.c, server.c, server.h, spu-low.c, symbol.c, target.h, tdesc.c, tdesc.h, thread-db.c, tracepoint.c, win32-i386-low.c, win32-low.c: Change common to gdbsupport.
460 lines
13 KiB
C
460 lines
13 KiB
C
/* Target-dependent code for OpenBSD/amd64.
|
||
|
||
Copyright (C) 2003-2019 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 "frame.h"
|
||
#include "frame-unwind.h"
|
||
#include "gdbcore.h"
|
||
#include "symtab.h"
|
||
#include "objfiles.h"
|
||
#include "osabi.h"
|
||
#include "regcache.h"
|
||
#include "regset.h"
|
||
#include "target.h"
|
||
#include "trad-frame.h"
|
||
|
||
#include "obsd-tdep.h"
|
||
#include "amd64-tdep.h"
|
||
#include "i387-tdep.h"
|
||
#include "gdbsupport/x86-xstate.h"
|
||
#include "solib-svr4.h"
|
||
#include "bsd-uthread.h"
|
||
|
||
/* Support for signal handlers. */
|
||
|
||
/* Default page size. */
|
||
static const int amd64obsd_page_size = 4096;
|
||
|
||
/* Return whether THIS_FRAME corresponds to an OpenBSD sigtramp
|
||
routine. */
|
||
|
||
static int
|
||
amd64obsd_sigtramp_p (struct frame_info *this_frame)
|
||
{
|
||
CORE_ADDR pc = get_frame_pc (this_frame);
|
||
CORE_ADDR start_pc = (pc & ~(amd64obsd_page_size - 1));
|
||
const gdb_byte osigreturn[] =
|
||
{
|
||
0x48, 0xc7, 0xc0,
|
||
0x67, 0x00, 0x00, 0x00, /* movq $SYS_sigreturn, %rax */
|
||
0xcd, 0x80 /* int $0x80 */
|
||
};
|
||
const gdb_byte sigreturn[] =
|
||
{
|
||
0x48, 0xc7, 0xc0,
|
||
0x67, 0x00, 0x00, 0x00, /* movq $SYS_sigreturn, %rax */
|
||
0x0f, 0x05 /* syscall */
|
||
};
|
||
size_t buflen = (sizeof sigreturn) + 1;
|
||
gdb_byte *buf;
|
||
const char *name;
|
||
|
||
/* If the function has a valid symbol name, it isn't a
|
||
trampoline. */
|
||
find_pc_partial_function (pc, &name, NULL, NULL);
|
||
if (name != NULL)
|
||
return 0;
|
||
|
||
/* If the function lives in a valid section (even without a starting
|
||
point) it isn't a trampoline. */
|
||
if (find_pc_section (pc) != NULL)
|
||
return 0;
|
||
|
||
/* If we can't read the instructions at START_PC, return zero. */
|
||
buf = (gdb_byte *) alloca ((sizeof sigreturn) + 1);
|
||
if (!safe_frame_unwind_memory (this_frame, start_pc + 6, buf, buflen))
|
||
return 0;
|
||
|
||
/* Check for sigreturn(2). Depending on how the assembler encoded
|
||
the `movq %rsp, %rdi' instruction, the code starts at offset 6 or
|
||
7. OpenBSD 5.0 and later use the `syscall' instruction. Older
|
||
versions use `int $0x80'. Check for both. */
|
||
if (memcmp (buf, sigreturn, sizeof sigreturn)
|
||
&& memcmp (buf + 1, sigreturn, sizeof sigreturn)
|
||
&& memcmp (buf, osigreturn, sizeof osigreturn)
|
||
&& memcmp (buf + 1, osigreturn, sizeof osigreturn))
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* Assuming THIS_FRAME is for a BSD sigtramp routine, return the
|
||
address of the associated sigcontext structure. */
|
||
|
||
static CORE_ADDR
|
||
amd64obsd_sigcontext_addr (struct frame_info *this_frame)
|
||
{
|
||
CORE_ADDR pc = get_frame_pc (this_frame);
|
||
ULONGEST offset = (pc & (amd64obsd_page_size - 1));
|
||
|
||
/* The %rsp register points at `struct sigcontext' upon entry of a
|
||
signal trampoline. The relevant part of the trampoline is
|
||
|
||
call *%rax
|
||
movq %rsp, %rdi
|
||
pushq %rdi
|
||
movq $SYS_sigreturn,%rax
|
||
int $0x80
|
||
|
||
(see /usr/src/sys/arch/amd64/amd64/locore.S). The `pushq'
|
||
instruction clobbers %rsp, but its value is saved in `%rdi'. */
|
||
|
||
if (offset > 5)
|
||
return get_frame_register_unsigned (this_frame, AMD64_RDI_REGNUM);
|
||
else
|
||
return get_frame_register_unsigned (this_frame, AMD64_RSP_REGNUM);
|
||
}
|
||
|
||
/* OpenBSD 3.5 or later. */
|
||
|
||
/* Mapping between the general-purpose registers in `struct reg'
|
||
format and GDB's register cache layout. */
|
||
|
||
/* From <machine/reg.h>. */
|
||
int amd64obsd_r_reg_offset[] =
|
||
{
|
||
14 * 8, /* %rax */
|
||
13 * 8, /* %rbx */
|
||
3 * 8, /* %rcx */
|
||
2 * 8, /* %rdx */
|
||
1 * 8, /* %rsi */
|
||
0 * 8, /* %rdi */
|
||
12 * 8, /* %rbp */
|
||
15 * 8, /* %rsp */
|
||
4 * 8, /* %r8 .. */
|
||
5 * 8,
|
||
6 * 8,
|
||
7 * 8,
|
||
8 * 8,
|
||
9 * 8,
|
||
10 * 8,
|
||
11 * 8, /* ... %r15 */
|
||
16 * 8, /* %rip */
|
||
17 * 8, /* %eflags */
|
||
18 * 8, /* %cs */
|
||
19 * 8, /* %ss */
|
||
20 * 8, /* %ds */
|
||
21 * 8, /* %es */
|
||
22 * 8, /* %fs */
|
||
23 * 8 /* %gs */
|
||
};
|
||
|
||
/* From <machine/signal.h>. */
|
||
static int amd64obsd_sc_reg_offset[] =
|
||
{
|
||
14 * 8, /* %rax */
|
||
13 * 8, /* %rbx */
|
||
3 * 8, /* %rcx */
|
||
2 * 8, /* %rdx */
|
||
1 * 8, /* %rsi */
|
||
0 * 8, /* %rdi */
|
||
12 * 8, /* %rbp */
|
||
24 * 8, /* %rsp */
|
||
4 * 8, /* %r8 ... */
|
||
5 * 8,
|
||
6 * 8,
|
||
7 * 8,
|
||
8 * 8,
|
||
9 * 8,
|
||
10 * 8,
|
||
11 * 8, /* ... %r15 */
|
||
21 * 8, /* %rip */
|
||
23 * 8, /* %eflags */
|
||
22 * 8, /* %cs */
|
||
25 * 8, /* %ss */
|
||
18 * 8, /* %ds */
|
||
17 * 8, /* %es */
|
||
16 * 8, /* %fs */
|
||
15 * 8 /* %gs */
|
||
};
|
||
|
||
/* From /usr/src/lib/libpthread/arch/amd64/uthread_machdep.c. */
|
||
static int amd64obsd_uthread_reg_offset[] =
|
||
{
|
||
19 * 8, /* %rax */
|
||
16 * 8, /* %rbx */
|
||
18 * 8, /* %rcx */
|
||
17 * 8, /* %rdx */
|
||
14 * 8, /* %rsi */
|
||
13 * 8, /* %rdi */
|
||
15 * 8, /* %rbp */
|
||
-1, /* %rsp */
|
||
12 * 8, /* %r8 ... */
|
||
11 * 8,
|
||
10 * 8,
|
||
9 * 8,
|
||
8 * 8,
|
||
7 * 8,
|
||
6 * 8,
|
||
5 * 8, /* ... %r15 */
|
||
20 * 8, /* %rip */
|
||
4 * 8, /* %eflags */
|
||
21 * 8, /* %cs */
|
||
-1, /* %ss */
|
||
3 * 8, /* %ds */
|
||
2 * 8, /* %es */
|
||
1 * 8, /* %fs */
|
||
0 * 8 /* %gs */
|
||
};
|
||
|
||
/* Offset within the thread structure where we can find the saved
|
||
stack pointer (%esp). */
|
||
#define AMD64OBSD_UTHREAD_RSP_OFFSET 400
|
||
|
||
static void
|
||
amd64obsd_supply_uthread (struct regcache *regcache,
|
||
int regnum, CORE_ADDR addr)
|
||
{
|
||
struct gdbarch *gdbarch = regcache->arch ();
|
||
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
||
CORE_ADDR sp_addr = addr + AMD64OBSD_UTHREAD_RSP_OFFSET;
|
||
CORE_ADDR sp = 0;
|
||
gdb_byte buf[8];
|
||
int i;
|
||
|
||
gdb_assert (regnum >= -1);
|
||
|
||
if (regnum == -1 || regnum == AMD64_RSP_REGNUM)
|
||
{
|
||
int offset;
|
||
|
||
/* Fetch stack pointer from thread structure. */
|
||
sp = read_memory_unsigned_integer (sp_addr, 8, byte_order);
|
||
|
||
/* Adjust the stack pointer such that it looks as if we just
|
||
returned from _thread_machdep_switch. */
|
||
offset = amd64obsd_uthread_reg_offset[AMD64_RIP_REGNUM] + 8;
|
||
store_unsigned_integer (buf, 8, byte_order, sp + offset);
|
||
regcache->raw_supply (AMD64_RSP_REGNUM, buf);
|
||
}
|
||
|
||
for (i = 0; i < ARRAY_SIZE (amd64obsd_uthread_reg_offset); i++)
|
||
{
|
||
if (amd64obsd_uthread_reg_offset[i] != -1
|
||
&& (regnum == -1 || regnum == i))
|
||
{
|
||
/* Fetch stack pointer from thread structure (if we didn't
|
||
do so already). */
|
||
if (sp == 0)
|
||
sp = read_memory_unsigned_integer (sp_addr, 8, byte_order);
|
||
|
||
/* Read the saved register from the stack frame. */
|
||
read_memory (sp + amd64obsd_uthread_reg_offset[i], buf, 8);
|
||
regcache->raw_supply (i, buf);
|
||
}
|
||
}
|
||
}
|
||
|
||
static void
|
||
amd64obsd_collect_uthread (const struct regcache *regcache,
|
||
int regnum, CORE_ADDR addr)
|
||
{
|
||
struct gdbarch *gdbarch = regcache->arch ();
|
||
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
||
CORE_ADDR sp_addr = addr + AMD64OBSD_UTHREAD_RSP_OFFSET;
|
||
CORE_ADDR sp = 0;
|
||
gdb_byte buf[8];
|
||
int i;
|
||
|
||
gdb_assert (regnum >= -1);
|
||
|
||
if (regnum == -1 || regnum == AMD64_RSP_REGNUM)
|
||
{
|
||
int offset;
|
||
|
||
/* Calculate the stack pointer (frame pointer) that will be
|
||
stored into the thread structure. */
|
||
offset = amd64obsd_uthread_reg_offset[AMD64_RIP_REGNUM] + 8;
|
||
regcache->raw_collect (AMD64_RSP_REGNUM, buf);
|
||
sp = extract_unsigned_integer (buf, 8, byte_order) - offset;
|
||
|
||
/* Store the stack pointer. */
|
||
write_memory_unsigned_integer (sp_addr, 8, byte_order, sp);
|
||
|
||
/* The stack pointer was (potentially) modified. Make sure we
|
||
build a proper stack frame. */
|
||
regnum = -1;
|
||
}
|
||
|
||
for (i = 0; i < ARRAY_SIZE (amd64obsd_uthread_reg_offset); i++)
|
||
{
|
||
if (amd64obsd_uthread_reg_offset[i] != -1
|
||
&& (regnum == -1 || regnum == i))
|
||
{
|
||
/* Fetch stack pointer from thread structure (if we didn't
|
||
calculate it already). */
|
||
if (sp == 0)
|
||
sp = read_memory_unsigned_integer (sp_addr, 8, byte_order);
|
||
|
||
/* Write the register into the stack frame. */
|
||
regcache->raw_collect (i, buf);
|
||
write_memory (sp + amd64obsd_uthread_reg_offset[i], buf, 8);
|
||
}
|
||
}
|
||
}
|
||
/* Kernel debugging support. */
|
||
|
||
/* From <machine/frame.h>. Easy since `struct trapframe' matches
|
||
`struct sigcontext'. */
|
||
#define amd64obsd_tf_reg_offset amd64obsd_sc_reg_offset
|
||
|
||
static struct trad_frame_cache *
|
||
amd64obsd_trapframe_cache (struct frame_info *this_frame, void **this_cache)
|
||
{
|
||
struct gdbarch *gdbarch = get_frame_arch (this_frame);
|
||
enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
|
||
struct trad_frame_cache *cache;
|
||
CORE_ADDR func, sp, addr;
|
||
ULONGEST cs;
|
||
const char *name;
|
||
int i;
|
||
|
||
if (*this_cache)
|
||
return (struct trad_frame_cache *) *this_cache;
|
||
|
||
cache = trad_frame_cache_zalloc (this_frame);
|
||
*this_cache = cache;
|
||
|
||
func = get_frame_func (this_frame);
|
||
sp = get_frame_register_unsigned (this_frame, AMD64_RSP_REGNUM);
|
||
|
||
find_pc_partial_function (func, &name, NULL, NULL);
|
||
if (name && startswith (name, "Xintr"))
|
||
addr = sp + 8; /* It's an interrupt frame. */
|
||
else
|
||
addr = sp;
|
||
|
||
for (i = 0; i < ARRAY_SIZE (amd64obsd_tf_reg_offset); i++)
|
||
if (amd64obsd_tf_reg_offset[i] != -1)
|
||
trad_frame_set_reg_addr (cache, i, addr + amd64obsd_tf_reg_offset[i]);
|
||
|
||
/* Read %cs from trap frame. */
|
||
addr += amd64obsd_tf_reg_offset[AMD64_CS_REGNUM];
|
||
cs = read_memory_unsigned_integer (addr, 8, byte_order);
|
||
if ((cs & I386_SEL_RPL) == I386_SEL_UPL)
|
||
{
|
||
/* Trap from user space; terminate backtrace. */
|
||
trad_frame_set_id (cache, outer_frame_id);
|
||
}
|
||
else
|
||
{
|
||
/* Construct the frame ID using the function start. */
|
||
trad_frame_set_id (cache, frame_id_build (sp + 16, func));
|
||
}
|
||
|
||
return cache;
|
||
}
|
||
|
||
static void
|
||
amd64obsd_trapframe_this_id (struct frame_info *this_frame,
|
||
void **this_cache, struct frame_id *this_id)
|
||
{
|
||
struct trad_frame_cache *cache =
|
||
amd64obsd_trapframe_cache (this_frame, this_cache);
|
||
|
||
trad_frame_get_id (cache, this_id);
|
||
}
|
||
|
||
static struct value *
|
||
amd64obsd_trapframe_prev_register (struct frame_info *this_frame,
|
||
void **this_cache, int regnum)
|
||
{
|
||
struct trad_frame_cache *cache =
|
||
amd64obsd_trapframe_cache (this_frame, this_cache);
|
||
|
||
return trad_frame_get_register (cache, this_frame, regnum);
|
||
}
|
||
|
||
static int
|
||
amd64obsd_trapframe_sniffer (const struct frame_unwind *self,
|
||
struct frame_info *this_frame,
|
||
void **this_prologue_cache)
|
||
{
|
||
ULONGEST cs;
|
||
const char *name;
|
||
|
||
/* Check Current Privilege Level and bail out if we're not executing
|
||
in kernel space. */
|
||
cs = get_frame_register_unsigned (this_frame, AMD64_CS_REGNUM);
|
||
if ((cs & I386_SEL_RPL) == I386_SEL_UPL)
|
||
return 0;
|
||
|
||
find_pc_partial_function (get_frame_pc (this_frame), &name, NULL, NULL);
|
||
return (name && ((strcmp (name, "calltrap") == 0)
|
||
|| (strcmp (name, "osyscall1") == 0)
|
||
|| (strcmp (name, "Xsyscall") == 0)
|
||
|| (startswith (name, "Xintr"))));
|
||
}
|
||
|
||
static const struct frame_unwind amd64obsd_trapframe_unwind = {
|
||
/* FIXME: kettenis/20051219: This really is more like an interrupt
|
||
frame, but SIGTRAMP_FRAME would print <signal handler called>,
|
||
which really is not what we want here. */
|
||
NORMAL_FRAME,
|
||
default_frame_unwind_stop_reason,
|
||
amd64obsd_trapframe_this_id,
|
||
amd64obsd_trapframe_prev_register,
|
||
NULL,
|
||
amd64obsd_trapframe_sniffer
|
||
};
|
||
|
||
|
||
static void
|
||
amd64obsd_init_abi (struct gdbarch_info info, struct gdbarch *gdbarch)
|
||
{
|
||
struct gdbarch_tdep *tdep = gdbarch_tdep (gdbarch);
|
||
|
||
amd64_init_abi (info, gdbarch,
|
||
amd64_target_description (X86_XSTATE_SSE_MASK, true));
|
||
obsd_init_abi (info, gdbarch);
|
||
|
||
/* Initialize general-purpose register set details. */
|
||
tdep->gregset_reg_offset = amd64obsd_r_reg_offset;
|
||
tdep->gregset_num_regs = ARRAY_SIZE (amd64obsd_r_reg_offset);
|
||
tdep->sizeof_gregset = 24 * 8;
|
||
|
||
tdep->jb_pc_offset = 7 * 8;
|
||
|
||
tdep->sigtramp_p = amd64obsd_sigtramp_p;
|
||
tdep->sigcontext_addr = amd64obsd_sigcontext_addr;
|
||
tdep->sc_reg_offset = amd64obsd_sc_reg_offset;
|
||
tdep->sc_num_regs = ARRAY_SIZE (amd64obsd_sc_reg_offset);
|
||
|
||
/* OpenBSD provides a user-level threads implementation. */
|
||
bsd_uthread_set_supply_uthread (gdbarch, amd64obsd_supply_uthread);
|
||
bsd_uthread_set_collect_uthread (gdbarch, amd64obsd_collect_uthread);
|
||
|
||
/* OpenBSD uses SVR4-style shared libraries. */
|
||
set_solib_svr4_fetch_link_map_offsets
|
||
(gdbarch, svr4_lp64_fetch_link_map_offsets);
|
||
|
||
/* Unwind kernel trap frames correctly. */
|
||
frame_unwind_prepend_unwinder (gdbarch, &amd64obsd_trapframe_unwind);
|
||
}
|
||
|
||
void
|
||
_initialize_amd64obsd_tdep (void)
|
||
{
|
||
/* The OpenBSD/amd64 native dependent code makes this assumption. */
|
||
gdb_assert (ARRAY_SIZE (amd64obsd_r_reg_offset) == AMD64_NUM_GREGS);
|
||
|
||
gdbarch_register_osabi (bfd_arch_i386, bfd_mach_x86_64,
|
||
GDB_OSABI_OPENBSD, amd64obsd_init_abi);
|
||
}
|