1 /* Native-dependent code for GNU/Linux x86-64.
3 Copyright (C) 2001-2024 Free Software Foundation, Inc.
4 Contributed by Jiri Smid, SuSE Labs.
6 This file is part of GDB.
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program. If not, see <http://www.gnu.org/licenses/>. */
23 #include "elf/common.h"
25 #include "nat/gdb_ptrace.h"
26 #include <asm/prctl.h>
29 #include "gdb_proc_service.h"
31 #include "amd64-nat.h"
32 #include "amd64-tdep.h"
33 #include "amd64-linux-tdep.h"
34 #include "i386-linux-tdep.h"
35 #include "gdbsupport/x86-xstate.h"
37 #include "x86-linux-nat.h"
38 #include "nat/linux-ptrace.h"
39 #include "nat/amd64-linux-siginfo.h"
41 /* This definition comes from prctl.h. Kernels older than 2.5.64
43 #ifndef PTRACE_ARCH_PRCTL
44 #define PTRACE_ARCH_PRCTL 30
47 struct amd64_linux_nat_target final
: public x86_linux_nat_target
49 /* Add our register access methods. */
50 void fetch_registers (struct regcache
*, int) override
;
51 void store_registers (struct regcache
*, int) override
;
53 bool low_siginfo_fixup (siginfo_t
*ptrace
, gdb_byte
*inf
, int direction
)
57 static amd64_linux_nat_target the_amd64_linux_nat_target
;
59 /* Mapping between the general-purpose registers in GNU/Linux x86-64
60 `struct user' format and GDB's register cache layout for GNU/Linux
63 Note that most GNU/Linux x86-64 registers are 64-bit, while the
64 GNU/Linux i386 registers are all 32-bit, but since we're
65 little-endian we get away with that. */
67 /* From <sys/reg.h> on GNU/Linux i386. */
68 static int amd64_linux_gregset32_reg_offset
[] =
70 RAX
* 8, RCX
* 8, /* %eax, %ecx */
71 RDX
* 8, RBX
* 8, /* %edx, %ebx */
72 RSP
* 8, RBP
* 8, /* %esp, %ebp */
73 RSI
* 8, RDI
* 8, /* %esi, %edi */
74 RIP
* 8, EFLAGS
* 8, /* %eip, %eflags */
75 CS
* 8, SS
* 8, /* %cs, %ss */
76 DS
* 8, ES
* 8, /* %ds, %es */
77 FS
* 8, GS
* 8, /* %fs, %gs */
78 -1, -1, -1, -1, -1, -1, -1, -1,
79 -1, -1, -1, -1, -1, -1, -1, -1,
80 -1, -1, -1, -1, -1, -1, -1, -1, -1,
81 -1, -1, -1, -1, -1, -1, -1, -1,
82 /* MPX is deprecated. Yet we keep this to not give the registers below
83 a new number. That could break older gdbservers. */
84 -1, -1, -1, -1, /* MPX registers BND0 ... BND3. */
85 -1, -1, /* MPX registers BNDCFGU, BNDSTATUS. */
86 -1, -1, -1, -1, -1, -1, -1, -1, /* k0 ... k7 (AVX512) */
87 -1, -1, -1, -1, -1, -1, -1, -1, /* zmm0 ... zmm7 (AVX512) */
88 -1, /* PKEYS register PKRU */
89 ORIG_RAX
* 8 /* "orig_eax" */
93 /* Transferring the general-purpose registers between GDB, inferiors
96 /* See amd64_collect_native_gregset. This linux specific version handles
97 issues with negative EAX values not being restored correctly upon syscall
98 return when debugging 32-bit targets. It has no effect on 64-bit
102 amd64_linux_collect_native_gregset (const struct regcache
*regcache
,
103 void *gregs
, int regnum
)
105 amd64_collect_native_gregset (regcache
, gregs
, regnum
);
107 struct gdbarch
*gdbarch
= regcache
->arch ();
108 if (gdbarch_bfd_arch_info (gdbarch
)->bits_per_word
== 32)
110 /* Sign extend EAX value to avoid potential syscall restart
113 On Linux, when a syscall is interrupted by a signal, the
114 (kernel function implementing the) syscall may return
115 -ERESTARTSYS when a signal occurs. Doing so indicates that
116 the syscall is restartable. Then, depending on settings
117 associated with the signal handler, and after the signal
118 handler is called, the kernel can then either return -EINTR
119 or it can cause the syscall to be restarted. We are
120 concerned with the latter case here.
122 On (32-bit) i386, the status (-ERESTARTSYS) is placed in the
123 EAX register. When debugging a 32-bit process from a 64-bit
124 (amd64) GDB, the debugger fetches 64-bit registers even
125 though the process being debugged is only 32-bit. The
126 register cache is only 32 bits wide though; GDB discards the
127 high 32 bits when placing 64-bit values in the 32-bit
128 regcache. Normally, this is not a problem since the 32-bit
129 process should only care about the lower 32-bit portions of
130 these registers. That said, it can happen that the 64-bit
131 value being restored will be different from the 64-bit value
132 that was originally retrieved from the kernel. The one place
133 (that we know of) where it does matter is in the kernel's
134 syscall restart code. The kernel's code for restarting a
135 syscall after a signal expects to see a negative value
136 (specifically -ERESTARTSYS) in the 64-bit RAX register in
137 order to correctly cause a syscall to be restarted.
139 The call to amd64_collect_native_gregset, above, is setting
140 the high 32 bits of RAX (and other registers too) to 0. For
141 syscall restart, we need to sign extend EAX so that RAX will
142 appear as a negative value when EAX is set to -ERESTARTSYS.
143 This in turn will cause the signal handling code in the
144 kernel to recognize -ERESTARTSYS which will in turn cause the
145 syscall to be restarted.
147 The test case gdb.base/interrupt.exp tests for this problem.
148 Without this sign extension code in place, it'll show
149 a number of failures when testing against unix/-m32. */
151 if (regnum
== -1 || regnum
== I386_EAX_REGNUM
)
153 void *ptr
= ((gdb_byte
*) gregs
154 + amd64_linux_gregset32_reg_offset
[I386_EAX_REGNUM
]);
156 *(int64_t *) ptr
= *(int32_t *) ptr
;
161 /* Fill GDB's register cache with the general-purpose register values
165 supply_gregset (struct regcache
*regcache
, const elf_gregset_t
*gregsetp
)
167 amd64_supply_native_gregset (regcache
, gregsetp
, -1);
170 /* Fill register REGNUM (if it is a general-purpose register) in
171 *GREGSETP with the value in GDB's register cache. If REGNUM is -1,
172 do this for all registers. */
175 fill_gregset (const struct regcache
*regcache
,
176 elf_gregset_t
*gregsetp
, int regnum
)
178 amd64_linux_collect_native_gregset (regcache
, gregsetp
, regnum
);
181 /* Transferring floating-point registers between GDB, inferiors and cores. */
183 /* Fill GDB's register cache with the floating-point and SSE register
184 values in *FPREGSETP. */
187 supply_fpregset (struct regcache
*regcache
, const elf_fpregset_t
*fpregsetp
)
189 amd64_supply_fxsave (regcache
, -1, fpregsetp
);
192 /* Fill register REGNUM (if it is a floating-point or SSE register) in
193 *FPREGSETP with the value in GDB's register cache. If REGNUM is
194 -1, do this for all registers. */
197 fill_fpregset (const struct regcache
*regcache
,
198 elf_fpregset_t
*fpregsetp
, int regnum
)
200 amd64_collect_fxsave (regcache
, regnum
, fpregsetp
);
204 /* Transferring arbitrary registers between GDB and inferior. */
206 /* Fetch register REGNUM from the child process. If REGNUM is -1, do
207 this for all registers (including the floating point and SSE
211 amd64_linux_nat_target::fetch_registers (struct regcache
*regcache
, int regnum
)
213 struct gdbarch
*gdbarch
= regcache
->arch ();
214 const i386_gdbarch_tdep
*tdep
= gdbarch_tdep
<i386_gdbarch_tdep
> (gdbarch
);
217 /* GNU/Linux LWP ID's are process ID's. */
218 tid
= regcache
->ptid ().lwp ();
220 tid
= regcache
->ptid ().pid (); /* Not a threaded program. */
222 if (regnum
== -1 || amd64_native_gregset_supplies_p (gdbarch
, regnum
))
226 if (ptrace (PTRACE_GETREGS
, tid
, 0, (long) ®s
) < 0)
227 perror_with_name (_("Couldn't get registers"));
229 amd64_supply_native_gregset (regcache
, ®s
, -1);
234 if (regnum
== -1 || !amd64_native_gregset_supplies_p (gdbarch
, regnum
))
236 elf_fpregset_t fpregs
;
238 if (have_ptrace_getregset
== TRIBOOL_TRUE
)
240 /* Pre-4.14 kernels have a bug (fixed by commit 0852b374173b
241 "x86/fpu: Add FPU state copying quirk to handle XRSTOR failure on
242 Intel Skylake CPUs") that sometimes causes the mxcsr location in
243 xstateregs not to be copied by PTRACE_GETREGSET. Make sure that
244 the location is at least initialized with a defined value. */
245 gdb::byte_vector
xstateregs (tdep
->xsave_layout
.sizeof_xsave
, 0);
248 iov
.iov_base
= xstateregs
.data ();
249 iov
.iov_len
= xstateregs
.size ();
250 if (ptrace (PTRACE_GETREGSET
, tid
,
251 (unsigned int) NT_X86_XSTATE
, (long) &iov
) < 0)
252 perror_with_name (_("Couldn't get extended state status"));
254 amd64_supply_xsave (regcache
, -1, xstateregs
.data ());
258 if (ptrace (PTRACE_GETFPREGS
, tid
, 0, (long) &fpregs
) < 0)
259 perror_with_name (_("Couldn't get floating point status"));
261 amd64_supply_fxsave (regcache
, -1, &fpregs
);
266 /* Store register REGNUM back into the child process. If REGNUM is
267 -1, do this for all registers (including the floating-point and SSE
271 amd64_linux_nat_target::store_registers (struct regcache
*regcache
, int regnum
)
273 struct gdbarch
*gdbarch
= regcache
->arch ();
274 const i386_gdbarch_tdep
*tdep
= gdbarch_tdep
<i386_gdbarch_tdep
> (gdbarch
);
277 /* GNU/Linux LWP ID's are process ID's. */
278 tid
= regcache
->ptid ().lwp ();
280 tid
= regcache
->ptid ().pid (); /* Not a threaded program. */
282 if (regnum
== -1 || amd64_native_gregset_supplies_p (gdbarch
, regnum
))
286 if (ptrace (PTRACE_GETREGS
, tid
, 0, (long) ®s
) < 0)
287 perror_with_name (_("Couldn't get registers"));
289 amd64_linux_collect_native_gregset (regcache
, ®s
, regnum
);
291 if (ptrace (PTRACE_SETREGS
, tid
, 0, (long) ®s
) < 0)
292 perror_with_name (_("Couldn't write registers"));
298 if (regnum
== -1 || !amd64_native_gregset_supplies_p (gdbarch
, regnum
))
300 elf_fpregset_t fpregs
;
302 if (have_ptrace_getregset
== TRIBOOL_TRUE
)
304 gdb::byte_vector
xstateregs (tdep
->xsave_layout
.sizeof_xsave
);
307 iov
.iov_base
= xstateregs
.data ();
308 iov
.iov_len
= xstateregs
.size ();
309 if (ptrace (PTRACE_GETREGSET
, tid
,
310 (unsigned int) NT_X86_XSTATE
, (long) &iov
) < 0)
311 perror_with_name (_("Couldn't get extended state status"));
313 amd64_collect_xsave (regcache
, regnum
, xstateregs
.data (), 0);
315 if (ptrace (PTRACE_SETREGSET
, tid
,
316 (unsigned int) NT_X86_XSTATE
, (long) &iov
) < 0)
317 perror_with_name (_("Couldn't write extended state status"));
321 if (ptrace (PTRACE_GETFPREGS
, tid
, 0, (long) &fpregs
) < 0)
322 perror_with_name (_("Couldn't get floating point status"));
324 amd64_collect_fxsave (regcache
, regnum
, &fpregs
);
326 if (ptrace (PTRACE_SETFPREGS
, tid
, 0, (long) &fpregs
) < 0)
327 perror_with_name (_("Couldn't write floating point status"));
333 /* This function is called by libthread_db as part of its handling of
334 a request for a thread's local storage address. */
337 ps_get_thread_area (struct ps_prochandle
*ph
,
338 lwpid_t lwpid
, int idx
, void **base
)
340 if (gdbarch_bfd_arch_info (ph
->thread
->inf
->arch ())->bits_per_word
== 32)
342 unsigned int base_addr
;
345 result
= x86_linux_get_thread_area (lwpid
, (void *) (long) idx
,
349 /* Extend the value to 64 bits. Here it's assumed that
350 a "long" and a "void *" are the same. */
351 (*base
) = (void *) (long) base_addr
;
358 /* FIXME: ezannoni-2003-07-09 see comment above about include
359 file order. We could be getting bogus values for these two. */
360 gdb_assert (FS
< ELF_NGREG
);
361 gdb_assert (GS
< ELF_NGREG
);
368 fs
= ptrace (PTRACE_PEEKUSER
, lwpid
,
369 offsetof (struct user_regs_struct
, fs_base
), 0);
383 gs
= ptrace (PTRACE_PEEKUSER
, lwpid
,
384 offsetof (struct user_regs_struct
, gs_base
), 0);
393 default: /* Should not happen. */
397 return PS_ERR
; /* ptrace failed. */
401 /* Convert a ptrace/host siginfo object, into/from the siginfo in the
402 layout of the inferiors' architecture. Returns true if any
403 conversion was done; false otherwise. If DIRECTION is 1, then copy
404 from INF to PTRACE. If DIRECTION is 0, copy from PTRACE to
408 amd64_linux_nat_target::low_siginfo_fixup (siginfo_t
*ptrace
,
412 struct gdbarch
*gdbarch
= get_frame_arch (get_current_frame ());
414 /* Is the inferior 32-bit? If so, then do fixup the siginfo
416 if (gdbarch_bfd_arch_info (gdbarch
)->bits_per_word
== 32)
417 return amd64_linux_siginfo_fixup_common (ptrace
, inf
, direction
,
419 /* No fixup for native x32 GDB. */
420 else if (gdbarch_addr_bit (gdbarch
) == 32 && sizeof (void *) == 8)
421 return amd64_linux_siginfo_fixup_common (ptrace
, inf
, direction
,
427 void _initialize_amd64_linux_nat ();
429 _initialize_amd64_linux_nat ()
431 amd64_native_gregset32_reg_offset
= amd64_linux_gregset32_reg_offset
;
432 amd64_native_gregset32_num_regs
= I386_LINUX_NUM_REGS
;
433 amd64_native_gregset64_reg_offset
= amd64_linux_gregset_reg_offset
;
434 amd64_native_gregset64_num_regs
= AMD64_LINUX_NUM_REGS
;
436 gdb_assert (ARRAY_SIZE (amd64_linux_gregset32_reg_offset
)
437 == amd64_native_gregset32_num_regs
);
439 linux_target
= &the_amd64_linux_nat_target
;
441 /* Add the target. */
442 add_inf_child_target (linux_target
);