forked from Imagelibrary/binutils-gdb
This updates the copyright headers to include 2025. I did this by running gdb/copyright.py and then manually modifying a few files as noted by the script. Approved-By: Eli Zaretskii <eliz@gnu.org>
129 lines
3.1 KiB
C
129 lines
3.1 KiB
C
/* Native-dependent code for GNU/Linux x86 (i386 and x86-64).
|
|
|
|
Copyright (C) 1999-2025 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 "x86-linux.h"
|
|
#include "x86-linux-dregs.h"
|
|
#include "nat/gdb_ptrace.h"
|
|
#include <sys/user.h>
|
|
|
|
/* Per-thread arch-specific data we want to keep. */
|
|
|
|
struct arch_lwp_info
|
|
{
|
|
/* Non-zero if our copy differs from what's recorded in the
|
|
thread. */
|
|
int debug_registers_changed;
|
|
};
|
|
|
|
/* See nat/x86-linux.h. */
|
|
|
|
void
|
|
lwp_set_debug_registers_changed (struct lwp_info *lwp, int value)
|
|
{
|
|
if (lwp_arch_private_info (lwp) == NULL)
|
|
lwp_set_arch_private_info (lwp, XCNEW (struct arch_lwp_info));
|
|
|
|
lwp_arch_private_info (lwp)->debug_registers_changed = value;
|
|
}
|
|
|
|
/* See nat/x86-linux.h. */
|
|
|
|
int
|
|
lwp_debug_registers_changed (struct lwp_info *lwp)
|
|
{
|
|
struct arch_lwp_info *info = lwp_arch_private_info (lwp);
|
|
|
|
/* NULL means either that this is the main thread still going
|
|
through the shell, or that no watchpoint has been set yet.
|
|
The debug registers are unchanged in either case. */
|
|
if (info == NULL)
|
|
return 0;
|
|
|
|
return info->debug_registers_changed;
|
|
}
|
|
|
|
/* See nat/x86-linux.h. */
|
|
|
|
void
|
|
x86_linux_new_thread (struct lwp_info *lwp)
|
|
{
|
|
lwp_set_debug_registers_changed (lwp, 1);
|
|
}
|
|
|
|
/* See nat/x86-linux.h. */
|
|
|
|
void
|
|
x86_linux_delete_thread (struct arch_lwp_info *arch_lwp)
|
|
{
|
|
xfree (arch_lwp);
|
|
}
|
|
|
|
/* See nat/x86-linux.h. */
|
|
|
|
void
|
|
x86_linux_prepare_to_resume (struct lwp_info *lwp)
|
|
{
|
|
x86_linux_update_debug_registers (lwp);
|
|
}
|
|
|
|
#ifdef __x86_64__
|
|
/* Value of CS segment register:
|
|
64bit process: 0x33
|
|
32bit process: 0x23 */
|
|
#define AMD64_LINUX_USER64_CS 0x33
|
|
|
|
/* Value of DS segment register:
|
|
LP64 process: 0x0
|
|
X32 process: 0x2b */
|
|
#define AMD64_LINUX_X32_DS 0x2b
|
|
#endif
|
|
|
|
/* See nat/x86-linux.h. */
|
|
|
|
x86_linux_arch_size
|
|
x86_linux_ptrace_get_arch_size (int tid)
|
|
{
|
|
#ifdef __x86_64__
|
|
unsigned long cs;
|
|
unsigned long ds;
|
|
|
|
/* Get CS register. */
|
|
errno = 0;
|
|
cs = ptrace (PTRACE_PEEKUSER, tid,
|
|
offsetof (struct user_regs_struct, cs), 0);
|
|
if (errno != 0)
|
|
perror_with_name (_("Couldn't get CS register"));
|
|
|
|
bool is_64bit = cs == AMD64_LINUX_USER64_CS;
|
|
|
|
/* Get DS register. */
|
|
errno = 0;
|
|
ds = ptrace (PTRACE_PEEKUSER, tid,
|
|
offsetof (struct user_regs_struct, ds), 0);
|
|
if (errno != 0)
|
|
perror_with_name (_("Couldn't get DS register"));
|
|
|
|
bool is_x32 = ds == AMD64_LINUX_X32_DS;
|
|
|
|
return x86_linux_arch_size (is_64bit, is_x32);
|
|
#else
|
|
return x86_linux_arch_size (false, false);
|
|
#endif
|
|
}
|