|  | /* Copyright (C) 2009-2024 Free Software Foundation, Inc. | 
|  | Contributed by ARM Ltd. | 
|  |  | 
|  | 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 "gdbsupport/break-common.h" | 
|  | #include "nat/linux-nat.h" | 
|  | #include "nat/aarch64-linux-hw-point.h" | 
|  | #include "nat/aarch64-linux.h" | 
|  |  | 
|  | #include "elf/common.h" | 
|  | #include "nat/gdb_ptrace.h" | 
|  | #include <asm/ptrace.h> | 
|  | #include <sys/uio.h> | 
|  |  | 
|  | /* Called when resuming a thread LWP. | 
|  | The hardware debug registers are updated when there is any change.  */ | 
|  |  | 
|  | void | 
|  | aarch64_linux_prepare_to_resume (struct lwp_info *lwp) | 
|  | { | 
|  | struct arch_lwp_info *info = lwp_arch_private_info (lwp); | 
|  |  | 
|  | /* NULL means this is the main thread still going through the shell, | 
|  | or, no watchpoint has been set yet.  In that case, there's | 
|  | nothing to do.  */ | 
|  | if (info == NULL) | 
|  | return; | 
|  |  | 
|  | if (DR_HAS_CHANGED (info->dr_changed_bp) | 
|  | || DR_HAS_CHANGED (info->dr_changed_wp)) | 
|  | { | 
|  | ptid_t ptid = ptid_of_lwp (lwp); | 
|  | int tid = ptid.lwp (); | 
|  | struct aarch64_debug_reg_state *state | 
|  | = aarch64_get_debug_reg_state (ptid.pid ()); | 
|  |  | 
|  | if (show_debug_regs) | 
|  | debug_printf ("prepare_to_resume thread %d\n", tid); | 
|  |  | 
|  | /* Watchpoints.  */ | 
|  | if (DR_HAS_CHANGED (info->dr_changed_wp)) | 
|  | { | 
|  | aarch64_linux_set_debug_regs (state, tid, 1); | 
|  | DR_CLEAR_CHANGED (info->dr_changed_wp); | 
|  | } | 
|  |  | 
|  | /* Breakpoints.  */ | 
|  | if (DR_HAS_CHANGED (info->dr_changed_bp)) | 
|  | { | 
|  | aarch64_linux_set_debug_regs (state, tid, 0); | 
|  | DR_CLEAR_CHANGED (info->dr_changed_bp); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Function to call when a new thread is detected.  */ | 
|  |  | 
|  | void | 
|  | aarch64_linux_new_thread (struct lwp_info *lwp) | 
|  | { | 
|  | ptid_t ptid = ptid_of_lwp (lwp); | 
|  | struct aarch64_debug_reg_state *state | 
|  | = aarch64_get_debug_reg_state (ptid.pid ()); | 
|  | struct arch_lwp_info *info = XCNEW (struct arch_lwp_info); | 
|  |  | 
|  | /* If there are hardware breakpoints/watchpoints in the process then mark that | 
|  | all the hardware breakpoint/watchpoint register pairs for this thread need | 
|  | to be initialized (with data from aarch_process_info.debug_reg_state).  */ | 
|  | if (aarch64_any_set_debug_regs_state (state, false)) | 
|  | DR_MARK_ALL_CHANGED (info->dr_changed_bp, aarch64_num_bp_regs); | 
|  | if (aarch64_any_set_debug_regs_state (state, true)) | 
|  | DR_MARK_ALL_CHANGED (info->dr_changed_wp, aarch64_num_wp_regs); | 
|  |  | 
|  | lwp_set_arch_private_info (lwp, info); | 
|  | } | 
|  |  | 
|  | /* See nat/aarch64-linux.h.  */ | 
|  |  | 
|  | void | 
|  | aarch64_linux_delete_thread (struct arch_lwp_info *arch_lwp) | 
|  | { | 
|  | xfree (arch_lwp); | 
|  | } | 
|  |  | 
|  | /* Convert native siginfo FROM to the siginfo in the layout of the | 
|  | inferior's architecture TO.  */ | 
|  |  | 
|  | void | 
|  | aarch64_compat_siginfo_from_siginfo (compat_siginfo_t *to, siginfo_t *from) | 
|  | { | 
|  | memset (to, 0, sizeof (*to)); | 
|  |  | 
|  | to->si_signo = from->si_signo; | 
|  | to->si_errno = from->si_errno; | 
|  | to->si_code = from->si_code; | 
|  |  | 
|  | if (to->si_code == SI_TIMER) | 
|  | { | 
|  | to->cpt_si_timerid = from->si_timerid; | 
|  | to->cpt_si_overrun = from->si_overrun; | 
|  | to->cpt_si_ptr = (intptr_t) from->si_ptr; | 
|  | } | 
|  | else if (to->si_code == SI_USER) | 
|  | { | 
|  | to->cpt_si_pid = from->si_pid; | 
|  | to->cpt_si_uid = from->si_uid; | 
|  | } | 
|  | else if (to->si_code < 0) | 
|  | { | 
|  | to->cpt_si_pid = from->si_pid; | 
|  | to->cpt_si_uid = from->si_uid; | 
|  | to->cpt_si_ptr = (intptr_t) from->si_ptr; | 
|  | } | 
|  | else | 
|  | { | 
|  | switch (to->si_signo) | 
|  | { | 
|  | case SIGCHLD: | 
|  | to->cpt_si_pid = from->si_pid; | 
|  | to->cpt_si_uid = from->si_uid; | 
|  | to->cpt_si_status = from->si_status; | 
|  | to->cpt_si_utime = from->si_utime; | 
|  | to->cpt_si_stime = from->si_stime; | 
|  | break; | 
|  | case SIGILL: | 
|  | case SIGFPE: | 
|  | case SIGSEGV: | 
|  | case SIGBUS: | 
|  | to->cpt_si_addr = (intptr_t) from->si_addr; | 
|  | break; | 
|  | case SIGPOLL: | 
|  | to->cpt_si_band = from->si_band; | 
|  | to->cpt_si_fd = from->si_fd; | 
|  | break; | 
|  | default: | 
|  | to->cpt_si_pid = from->si_pid; | 
|  | to->cpt_si_uid = from->si_uid; | 
|  | to->cpt_si_ptr = (intptr_t) from->si_ptr; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Convert inferior's architecture siginfo FROM to native siginfo TO.  */ | 
|  |  | 
|  | void | 
|  | aarch64_siginfo_from_compat_siginfo (siginfo_t *to, compat_siginfo_t *from) | 
|  | { | 
|  | memset (to, 0, sizeof (*to)); | 
|  |  | 
|  | to->si_signo = from->si_signo; | 
|  | to->si_errno = from->si_errno; | 
|  | to->si_code = from->si_code; | 
|  |  | 
|  | if (to->si_code == SI_TIMER) | 
|  | { | 
|  | to->si_timerid = from->cpt_si_timerid; | 
|  | to->si_overrun = from->cpt_si_overrun; | 
|  | to->si_ptr = (void *) (intptr_t) from->cpt_si_ptr; | 
|  | } | 
|  | else if (to->si_code == SI_USER) | 
|  | { | 
|  | to->si_pid = from->cpt_si_pid; | 
|  | to->si_uid = from->cpt_si_uid; | 
|  | } | 
|  | if (to->si_code < 0) | 
|  | { | 
|  | to->si_pid = from->cpt_si_pid; | 
|  | to->si_uid = from->cpt_si_uid; | 
|  | to->si_ptr = (void *) (intptr_t) from->cpt_si_ptr; | 
|  | } | 
|  | else | 
|  | { | 
|  | switch (to->si_signo) | 
|  | { | 
|  | case SIGCHLD: | 
|  | to->si_pid = from->cpt_si_pid; | 
|  | to->si_uid = from->cpt_si_uid; | 
|  | to->si_status = from->cpt_si_status; | 
|  | to->si_utime = from->cpt_si_utime; | 
|  | to->si_stime = from->cpt_si_stime; | 
|  | break; | 
|  | case SIGILL: | 
|  | case SIGFPE: | 
|  | case SIGSEGV: | 
|  | case SIGBUS: | 
|  | to->si_addr = (void *) (intptr_t) from->cpt_si_addr; | 
|  | break; | 
|  | case SIGPOLL: | 
|  | to->si_band = from->cpt_si_band; | 
|  | to->si_fd = from->cpt_si_fd; | 
|  | break; | 
|  | default: | 
|  | to->si_pid = from->cpt_si_pid; | 
|  | to->si_uid = from->cpt_si_uid; | 
|  | to->si_ptr = (void* ) (intptr_t) from->cpt_si_ptr; | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Called by libthread_db.  Returns a pointer to the thread local | 
|  | storage (or its descriptor).  */ | 
|  |  | 
|  | ps_err_e | 
|  | aarch64_ps_get_thread_area (struct ps_prochandle *ph, | 
|  | lwpid_t lwpid, int idx, void **base, | 
|  | int is_64bit_p) | 
|  | { | 
|  | struct iovec iovec; | 
|  | uint64_t reg64; | 
|  | uint32_t reg32; | 
|  |  | 
|  | if (is_64bit_p) | 
|  | { | 
|  | iovec.iov_base = ®64; | 
|  | iovec.iov_len = sizeof (reg64); | 
|  | } | 
|  | else | 
|  | { | 
|  | iovec.iov_base = ®32; | 
|  | iovec.iov_len = sizeof (reg32); | 
|  | } | 
|  |  | 
|  | if (ptrace (PTRACE_GETREGSET, lwpid, NT_ARM_TLS, &iovec) != 0) | 
|  | return PS_ERR; | 
|  |  | 
|  | /* IDX is the bias from the thread pointer to the beginning of the | 
|  | thread descriptor.  It has to be subtracted due to implementation | 
|  | quirks in libthread_db.  */ | 
|  | if (is_64bit_p) | 
|  | *base = (void *) (reg64 - idx); | 
|  | else | 
|  | *base = (void *) (uintptr_t) (reg32 - idx); | 
|  |  | 
|  | return PS_OK; | 
|  | } | 
|  |  | 
|  | /* See nat/aarch64-linux.h.  */ | 
|  |  | 
|  | int | 
|  | aarch64_tls_register_count (int tid) | 
|  | { | 
|  | uint64_t tls_regs[2]; | 
|  | struct iovec iovec; | 
|  | iovec.iov_base = tls_regs; | 
|  | iovec.iov_len = sizeof (tls_regs); | 
|  |  | 
|  | /* Attempt to read both TPIDR and TPIDR2.  If ptrace returns less data than | 
|  | we are expecting, that means it doesn't support all the registers.  From | 
|  | the iovec length, figure out how many TPIDR registers the target actually | 
|  | supports.  */ | 
|  | if (ptrace (PTRACE_GETREGSET, tid, NT_ARM_TLS, &iovec) != 0) | 
|  | return 0; | 
|  |  | 
|  | /* Calculate how many TPIDR registers we have.  */ | 
|  | return iovec.iov_len / sizeof (uint64_t); | 
|  | } |