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.
597 lines
14 KiB
C
597 lines
14 KiB
C
/* Low-level siginfo manipulation for amd64.
|
|
|
|
Copyright (C) 2002-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 <signal.h>
|
|
#include "gdbsupport/common-defs.h"
|
|
#include "amd64-linux-siginfo.h"
|
|
|
|
#define GDB_SI_SIZE 128
|
|
|
|
/* The types below define the most complete kernel siginfo types known
|
|
for the architecture, independent of the system/libc headers. They
|
|
are named from a 64-bit kernel's perspective:
|
|
|
|
| layout | type |
|
|
|--------+----------------------|
|
|
| 64-bit | nat_siginfo_t |
|
|
| 32-bit | compat_siginfo_t |
|
|
| x32 | compat_x32_siginfo_t |
|
|
*/
|
|
|
|
#ifndef __ILP32__
|
|
|
|
typedef int nat_int_t;
|
|
typedef unsigned long nat_uptr_t;
|
|
|
|
typedef int nat_time_t;
|
|
typedef int nat_timer_t;
|
|
|
|
/* For native 64-bit, clock_t in _sigchld is 64-bit. */
|
|
typedef long nat_clock_t;
|
|
|
|
typedef union nat_sigval
|
|
{
|
|
nat_int_t sival_int;
|
|
nat_uptr_t sival_ptr;
|
|
} nat_sigval_t;
|
|
|
|
typedef struct nat_siginfo
|
|
{
|
|
int si_signo;
|
|
int si_errno;
|
|
int si_code;
|
|
|
|
union
|
|
{
|
|
int _pad[((128 / sizeof (int)) - 4)];
|
|
/* kill() */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
} _kill;
|
|
|
|
/* POSIX.1b timers */
|
|
struct
|
|
{
|
|
nat_timer_t _tid;
|
|
int _overrun;
|
|
nat_sigval_t _sigval;
|
|
} _timer;
|
|
|
|
/* POSIX.1b signals */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
nat_sigval_t _sigval;
|
|
} _rt;
|
|
|
|
/* SIGCHLD */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
int _status;
|
|
nat_clock_t _utime;
|
|
nat_clock_t _stime;
|
|
} _sigchld;
|
|
|
|
/* SIGILL, SIGFPE, SIGSEGV, SIGBUS */
|
|
struct
|
|
{
|
|
nat_uptr_t _addr;
|
|
short int _addr_lsb;
|
|
struct
|
|
{
|
|
nat_uptr_t _lower;
|
|
nat_uptr_t _upper;
|
|
} si_addr_bnd;
|
|
} _sigfault;
|
|
|
|
/* SIGPOLL */
|
|
struct
|
|
{
|
|
int _band;
|
|
int _fd;
|
|
} _sigpoll;
|
|
} _sifields;
|
|
} nat_siginfo_t;
|
|
|
|
#endif /* __ILP32__ */
|
|
|
|
/* These types below (compat_*) define a siginfo type that is layout
|
|
compatible with the siginfo type exported by the 32-bit userspace
|
|
support. */
|
|
|
|
typedef int compat_int_t;
|
|
typedef unsigned int compat_uptr_t;
|
|
|
|
typedef int compat_time_t;
|
|
typedef int compat_timer_t;
|
|
typedef int compat_clock_t;
|
|
|
|
struct compat_timeval
|
|
{
|
|
compat_time_t tv_sec;
|
|
int tv_usec;
|
|
};
|
|
|
|
typedef union compat_sigval
|
|
{
|
|
compat_int_t sival_int;
|
|
compat_uptr_t sival_ptr;
|
|
} compat_sigval_t;
|
|
|
|
typedef struct compat_siginfo
|
|
{
|
|
int si_signo;
|
|
int si_errno;
|
|
int si_code;
|
|
|
|
union
|
|
{
|
|
int _pad[((128 / sizeof (int)) - 3)];
|
|
|
|
/* kill() */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
} _kill;
|
|
|
|
/* POSIX.1b timers */
|
|
struct
|
|
{
|
|
compat_timer_t _tid;
|
|
int _overrun;
|
|
compat_sigval_t _sigval;
|
|
} _timer;
|
|
|
|
/* POSIX.1b signals */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
compat_sigval_t _sigval;
|
|
} _rt;
|
|
|
|
/* SIGCHLD */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
int _status;
|
|
compat_clock_t _utime;
|
|
compat_clock_t _stime;
|
|
} _sigchld;
|
|
|
|
/* SIGILL, SIGFPE, SIGSEGV, SIGBUS */
|
|
struct
|
|
{
|
|
unsigned int _addr;
|
|
short int _addr_lsb;
|
|
struct
|
|
{
|
|
unsigned int _lower;
|
|
unsigned int _upper;
|
|
} si_addr_bnd;
|
|
} _sigfault;
|
|
|
|
/* SIGPOLL */
|
|
struct
|
|
{
|
|
int _band;
|
|
int _fd;
|
|
} _sigpoll;
|
|
} _sifields;
|
|
} compat_siginfo_t;
|
|
|
|
/* For x32, clock_t in _sigchld is 64bit aligned at 4 bytes. */
|
|
typedef long __attribute__ ((__aligned__ (4))) compat_x32_clock_t;
|
|
|
|
typedef struct compat_x32_siginfo
|
|
{
|
|
int si_signo;
|
|
int si_errno;
|
|
int si_code;
|
|
|
|
union
|
|
{
|
|
int _pad[((128 / sizeof (int)) - 3)];
|
|
|
|
/* kill() */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
} _kill;
|
|
|
|
/* POSIX.1b timers */
|
|
struct
|
|
{
|
|
compat_timer_t _tid;
|
|
int _overrun;
|
|
compat_sigval_t _sigval;
|
|
} _timer;
|
|
|
|
/* POSIX.1b signals */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
compat_sigval_t _sigval;
|
|
} _rt;
|
|
|
|
/* SIGCHLD */
|
|
struct
|
|
{
|
|
unsigned int _pid;
|
|
unsigned int _uid;
|
|
int _status;
|
|
compat_x32_clock_t _utime;
|
|
compat_x32_clock_t _stime;
|
|
} _sigchld;
|
|
|
|
/* SIGILL, SIGFPE, SIGSEGV, SIGBUS */
|
|
struct
|
|
{
|
|
unsigned int _addr;
|
|
unsigned int _addr_lsb;
|
|
} _sigfault;
|
|
|
|
/* SIGPOLL */
|
|
struct
|
|
{
|
|
int _band;
|
|
int _fd;
|
|
} _sigpoll;
|
|
} _sifields;
|
|
} compat_x32_siginfo_t __attribute__ ((__aligned__ (8)));
|
|
|
|
/* To simplify usage of siginfo fields. */
|
|
|
|
#define cpt_si_pid _sifields._kill._pid
|
|
#define cpt_si_uid _sifields._kill._uid
|
|
#define cpt_si_timerid _sifields._timer._tid
|
|
#define cpt_si_overrun _sifields._timer._overrun
|
|
#define cpt_si_status _sifields._sigchld._status
|
|
#define cpt_si_utime _sifields._sigchld._utime
|
|
#define cpt_si_stime _sifields._sigchld._stime
|
|
#define cpt_si_ptr _sifields._rt._sigval.sival_ptr
|
|
#define cpt_si_addr _sifields._sigfault._addr
|
|
#define cpt_si_addr_lsb _sifields._sigfault._addr_lsb
|
|
#define cpt_si_band _sifields._sigpoll._band
|
|
#define cpt_si_fd _sifields._sigpoll._fd
|
|
|
|
/* glibc at least up to 2.3.2 doesn't have si_timerid, si_overrun.
|
|
In their place is si_timer1,si_timer2. */
|
|
|
|
#ifndef si_timerid
|
|
#define si_timerid si_timer1
|
|
#endif
|
|
#ifndef si_overrun
|
|
#define si_overrun si_timer2
|
|
#endif
|
|
|
|
/* The type of the siginfo object the kernel returns in
|
|
PTRACE_GETSIGINFO. If gdb is built as a x32 program, we get a x32
|
|
siginfo. */
|
|
#ifdef __ILP32__
|
|
typedef compat_x32_siginfo_t ptrace_siginfo_t;
|
|
#else
|
|
typedef nat_siginfo_t ptrace_siginfo_t;
|
|
#endif
|
|
|
|
/* Convert the system provided siginfo into compatible siginfo. */
|
|
|
|
static void
|
|
compat_siginfo_from_siginfo (compat_siginfo_t *to, const siginfo_t *from)
|
|
{
|
|
ptrace_siginfo_t from_ptrace;
|
|
|
|
memcpy (&from_ptrace, from, sizeof (from_ptrace));
|
|
memset (to, 0, sizeof (*to));
|
|
|
|
to->si_signo = from_ptrace.si_signo;
|
|
to->si_errno = from_ptrace.si_errno;
|
|
to->si_code = from_ptrace.si_code;
|
|
|
|
if (to->si_code == SI_TIMER)
|
|
{
|
|
to->cpt_si_timerid = from_ptrace.cpt_si_timerid;
|
|
to->cpt_si_overrun = from_ptrace.cpt_si_overrun;
|
|
to->cpt_si_ptr = from_ptrace.cpt_si_ptr;
|
|
}
|
|
else if (to->si_code == SI_USER)
|
|
{
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
}
|
|
else if (to->si_code < 0)
|
|
{
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
to->cpt_si_ptr = from_ptrace.cpt_si_ptr;
|
|
}
|
|
else
|
|
{
|
|
switch (to->si_signo)
|
|
{
|
|
case SIGCHLD:
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
to->cpt_si_status = from_ptrace.cpt_si_status;
|
|
to->cpt_si_utime = from_ptrace.cpt_si_utime;
|
|
to->cpt_si_stime = from_ptrace.cpt_si_stime;
|
|
break;
|
|
case SIGILL:
|
|
case SIGFPE:
|
|
case SIGSEGV:
|
|
case SIGBUS:
|
|
to->cpt_si_addr = from_ptrace.cpt_si_addr;
|
|
break;
|
|
case SIGPOLL:
|
|
to->cpt_si_band = from_ptrace.cpt_si_band;
|
|
to->cpt_si_fd = from_ptrace.cpt_si_fd;
|
|
break;
|
|
default:
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
to->cpt_si_ptr = from_ptrace.cpt_si_ptr;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Convert the compatible siginfo into system siginfo. */
|
|
|
|
static void
|
|
siginfo_from_compat_siginfo (siginfo_t *to, const compat_siginfo_t *from)
|
|
{
|
|
ptrace_siginfo_t to_ptrace;
|
|
|
|
memset (&to_ptrace, 0, sizeof (to_ptrace));
|
|
|
|
to_ptrace.si_signo = from->si_signo;
|
|
to_ptrace.si_errno = from->si_errno;
|
|
to_ptrace.si_code = from->si_code;
|
|
|
|
if (to_ptrace.si_code == SI_TIMER)
|
|
{
|
|
to_ptrace.cpt_si_timerid = from->cpt_si_timerid;
|
|
to_ptrace.cpt_si_overrun = from->cpt_si_overrun;
|
|
to_ptrace.cpt_si_ptr = from->cpt_si_ptr;
|
|
}
|
|
else if (to_ptrace.si_code == SI_USER)
|
|
{
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
}
|
|
if (to_ptrace.si_code < 0)
|
|
{
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
to_ptrace.cpt_si_ptr = from->cpt_si_ptr;
|
|
}
|
|
else
|
|
{
|
|
switch (to_ptrace.si_signo)
|
|
{
|
|
case SIGCHLD:
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
to_ptrace.cpt_si_status = from->cpt_si_status;
|
|
to_ptrace.cpt_si_utime = from->cpt_si_utime;
|
|
to_ptrace.cpt_si_stime = from->cpt_si_stime;
|
|
break;
|
|
case SIGILL:
|
|
case SIGFPE:
|
|
case SIGSEGV:
|
|
case SIGBUS:
|
|
to_ptrace.cpt_si_addr = from->cpt_si_addr;
|
|
to_ptrace.cpt_si_addr_lsb = from->cpt_si_addr_lsb;
|
|
break;
|
|
case SIGPOLL:
|
|
to_ptrace.cpt_si_band = from->cpt_si_band;
|
|
to_ptrace.cpt_si_fd = from->cpt_si_fd;
|
|
break;
|
|
default:
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
to_ptrace.cpt_si_ptr = from->cpt_si_ptr;
|
|
break;
|
|
}
|
|
}
|
|
memcpy (to, &to_ptrace, sizeof (to_ptrace));
|
|
}
|
|
|
|
/* Convert the system provided siginfo into compatible x32 siginfo. */
|
|
|
|
static void
|
|
compat_x32_siginfo_from_siginfo (compat_x32_siginfo_t *to,
|
|
const siginfo_t *from)
|
|
{
|
|
ptrace_siginfo_t from_ptrace;
|
|
|
|
memcpy (&from_ptrace, from, sizeof (from_ptrace));
|
|
memset (to, 0, sizeof (*to));
|
|
|
|
to->si_signo = from_ptrace.si_signo;
|
|
to->si_errno = from_ptrace.si_errno;
|
|
to->si_code = from_ptrace.si_code;
|
|
|
|
if (to->si_code == SI_TIMER)
|
|
{
|
|
to->cpt_si_timerid = from_ptrace.cpt_si_timerid;
|
|
to->cpt_si_overrun = from_ptrace.cpt_si_overrun;
|
|
to->cpt_si_ptr = from_ptrace.cpt_si_ptr;
|
|
}
|
|
else if (to->si_code == SI_USER)
|
|
{
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
}
|
|
else if (to->si_code < 0)
|
|
{
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
to->cpt_si_ptr = from_ptrace.cpt_si_ptr;
|
|
}
|
|
else
|
|
{
|
|
switch (to->si_signo)
|
|
{
|
|
case SIGCHLD:
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
to->cpt_si_status = from_ptrace.cpt_si_status;
|
|
memcpy (&to->cpt_si_utime, &from_ptrace.cpt_si_utime,
|
|
sizeof (to->cpt_si_utime));
|
|
memcpy (&to->cpt_si_stime, &from_ptrace.cpt_si_stime,
|
|
sizeof (to->cpt_si_stime));
|
|
break;
|
|
case SIGILL:
|
|
case SIGFPE:
|
|
case SIGSEGV:
|
|
case SIGBUS:
|
|
to->cpt_si_addr = from_ptrace.cpt_si_addr;
|
|
break;
|
|
case SIGPOLL:
|
|
to->cpt_si_band = from_ptrace.cpt_si_band;
|
|
to->cpt_si_fd = from_ptrace.cpt_si_fd;
|
|
break;
|
|
default:
|
|
to->cpt_si_pid = from_ptrace.cpt_si_pid;
|
|
to->cpt_si_uid = from_ptrace.cpt_si_uid;
|
|
to->cpt_si_ptr = from_ptrace.cpt_si_ptr;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Convert the compatible x32 siginfo into system siginfo. */
|
|
static void
|
|
siginfo_from_compat_x32_siginfo (siginfo_t *to,
|
|
const compat_x32_siginfo_t *from)
|
|
{
|
|
ptrace_siginfo_t to_ptrace;
|
|
|
|
memset (&to_ptrace, 0, sizeof (to_ptrace));
|
|
to_ptrace.si_signo = from->si_signo;
|
|
to_ptrace.si_errno = from->si_errno;
|
|
to_ptrace.si_code = from->si_code;
|
|
|
|
if (to_ptrace.si_code == SI_TIMER)
|
|
{
|
|
to_ptrace.cpt_si_timerid = from->cpt_si_timerid;
|
|
to_ptrace.cpt_si_overrun = from->cpt_si_overrun;
|
|
to_ptrace.cpt_si_ptr = from->cpt_si_ptr;
|
|
}
|
|
else if (to_ptrace.si_code == SI_USER)
|
|
{
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
}
|
|
if (to_ptrace.si_code < 0)
|
|
{
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
to_ptrace.cpt_si_ptr = from->cpt_si_ptr;
|
|
}
|
|
else
|
|
{
|
|
switch (to_ptrace.si_signo)
|
|
{
|
|
case SIGCHLD:
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
to_ptrace.cpt_si_status = from->cpt_si_status;
|
|
memcpy (&to_ptrace.cpt_si_utime, &from->cpt_si_utime,
|
|
sizeof (to_ptrace.cpt_si_utime));
|
|
memcpy (&to_ptrace.cpt_si_stime, &from->cpt_si_stime,
|
|
sizeof (to_ptrace.cpt_si_stime));
|
|
break;
|
|
case SIGILL:
|
|
case SIGFPE:
|
|
case SIGSEGV:
|
|
case SIGBUS:
|
|
to_ptrace.cpt_si_addr = from->cpt_si_addr;
|
|
break;
|
|
case SIGPOLL:
|
|
to_ptrace.cpt_si_band = from->cpt_si_band;
|
|
to_ptrace.cpt_si_fd = from->cpt_si_fd;
|
|
break;
|
|
default:
|
|
to_ptrace.cpt_si_pid = from->cpt_si_pid;
|
|
to_ptrace.cpt_si_uid = from->cpt_si_uid;
|
|
to_ptrace.cpt_si_ptr = from->cpt_si_ptr;
|
|
break;
|
|
}
|
|
}
|
|
memcpy (to, &to_ptrace, sizeof (to_ptrace));
|
|
}
|
|
|
|
/* Convert a ptrace siginfo object, into/from the siginfo in the
|
|
layout of the inferiors' architecture. Returns true if any
|
|
conversion was done; false otherwise. If DIRECTION is 1, then copy
|
|
from INF to PTRACE. If DIRECTION is 0, then copy from NATIVE to
|
|
INF. */
|
|
|
|
int
|
|
amd64_linux_siginfo_fixup_common (siginfo_t *ptrace, gdb_byte *inf,
|
|
int direction,
|
|
enum amd64_siginfo_fixup_mode mode)
|
|
{
|
|
if (mode == FIXUP_32)
|
|
{
|
|
if (direction == 0)
|
|
compat_siginfo_from_siginfo ((struct compat_siginfo *) inf, ptrace);
|
|
else
|
|
siginfo_from_compat_siginfo (ptrace, (struct compat_siginfo *) inf);
|
|
|
|
return 1;
|
|
}
|
|
else if (mode == FIXUP_X32)
|
|
{
|
|
if (direction == 0)
|
|
compat_x32_siginfo_from_siginfo ((struct compat_x32_siginfo *) inf,
|
|
ptrace);
|
|
else
|
|
siginfo_from_compat_x32_siginfo (ptrace,
|
|
(struct compat_x32_siginfo *) inf);
|
|
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Sanity check for the siginfo structure sizes. */
|
|
|
|
gdb_static_assert (sizeof (siginfo_t) == GDB_SI_SIZE);
|
|
#ifndef __ILP32__
|
|
gdb_static_assert (sizeof (nat_siginfo_t) == GDB_SI_SIZE);
|
|
#endif
|
|
gdb_static_assert (sizeof (compat_x32_siginfo_t) == GDB_SI_SIZE);
|
|
gdb_static_assert (sizeof (compat_siginfo_t) == GDB_SI_SIZE);
|
|
gdb_static_assert (sizeof (ptrace_siginfo_t) == GDB_SI_SIZE);
|