| /* Native-dependent code for GNU/Linux ARC. | 
 |  | 
 |    Copyright 2020-2024 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 "frame.h" | 
 | #include "inferior.h" | 
 | #include "gdbcore.h" | 
 | #include "regcache.h" | 
 | #include "gdbsupport/gdb_assert.h" | 
 | #include "target.h" | 
 | #include "linux-nat.h" | 
 | #include "nat/gdb_ptrace.h" | 
 |  | 
 | #include <stdint.h> | 
 | #include <sys/types.h> | 
 | #include <sys/param.h> | 
 | #include <signal.h> | 
 | #include <sys/user.h> | 
 | #include <sys/ioctl.h> | 
 | #include "gdbsupport/gdb_wait.h" | 
 | #include <fcntl.h> | 
 | #include <sys/procfs.h> | 
 | #include <linux/elf.h> | 
 |  | 
 | #include "gregset.h" | 
 | #include "arc-tdep.h" | 
 | #include "arc-linux-tdep.h" | 
 | #include "arch/arc.h" | 
 |  | 
 | /* Defines ps_err_e, struct ps_prochandle.  */ | 
 | #include "gdb_proc_service.h" | 
 |  | 
 | /* Print an "arc-linux-nat" debug statement.  */ | 
 |  | 
 | #define arc_linux_nat_debug_printf(fmt, ...) \ | 
 |   debug_prefixed_printf_cond (arc_debug, "arc-linux-nat", fmt, ##__VA_ARGS__) | 
 |  | 
 | /* Linux starting with 4.12 supports NT_ARC_V2 note type, which adds R30, | 
 |    R58 and R59 registers, which are specific to ARC HS and aren't | 
 |    available in ARC 700.  */ | 
 | #if defined (NT_ARC_V2) && defined (__ARCHS__) | 
 | #define ARC_HAS_V2_REGSET | 
 | #endif | 
 |  | 
 | class arc_linux_nat_target final : public linux_nat_target | 
 | { | 
 | public: | 
 |   /* Add ARC register access methods.  */ | 
 |   void fetch_registers (struct regcache *, int) override; | 
 |   void store_registers (struct regcache *, int) override; | 
 |  | 
 |   const struct target_desc *read_description () override; | 
 |  | 
 |   /* Handle threads  */ | 
 |   void low_prepare_to_resume (struct lwp_info *lp) override; | 
 | }; | 
 |  | 
 | static arc_linux_nat_target the_arc_linux_nat_target; | 
 |  | 
 | /* Read general registers from target process/thread (via ptrace) | 
 |    into REGCACHE.  */ | 
 |  | 
 | static void | 
 | fetch_gregs (struct regcache *regcache, int regnum) | 
 | { | 
 |   const int tid = get_ptrace_pid (regcache->ptid ()); | 
 |   struct iovec iov; | 
 |   gdb_gregset_t regs; | 
 |  | 
 |   iov.iov_base = ®s; | 
 |   iov.iov_len = sizeof (gdb_gregset_t); | 
 |  | 
 |   if (ptrace (PTRACE_GETREGSET, tid, NT_PRSTATUS, (void *) &iov) < 0) | 
 |     perror_with_name (_("Couldn't get general registers")); | 
 |   else | 
 |     arc_linux_supply_gregset (NULL, regcache, regnum, ®s, 0); | 
 | } | 
 |  | 
 | #ifdef ARC_HAS_V2_REGSET | 
 | /* Read ARC v2 registers from target process/thread (via ptrace) | 
 |    into REGCACHE.  */ | 
 |  | 
 | static void | 
 | fetch_v2_regs (struct regcache *regcache, int regnum) | 
 | { | 
 |   const int tid = get_ptrace_pid (regcache->ptid ()); | 
 |   struct iovec iov; | 
 |   bfd_byte v2_buffer[ARC_LINUX_SIZEOF_V2_REGSET]; | 
 |  | 
 |   iov.iov_base = &v2_buffer; | 
 |   iov.iov_len = ARC_LINUX_SIZEOF_V2_REGSET; | 
 |  | 
 |   if (ptrace (PTRACE_GETREGSET, tid, NT_ARC_V2, (void *) &iov) < 0) | 
 |     perror_with_name (_("Couldn't get ARC HS registers")); | 
 |   else | 
 |     arc_linux_supply_v2_regset (NULL, regcache, regnum, v2_buffer, 0); | 
 | } | 
 | #endif | 
 |  | 
 | /* Store general registers from REGCACHE into the target process/thread.  */ | 
 |  | 
 | static void | 
 | store_gregs (const struct regcache *regcache, int regnum) | 
 | { | 
 |   const int tid = get_ptrace_pid (regcache->ptid ()); | 
 |   struct iovec iov; | 
 |   gdb_gregset_t regs; | 
 |  | 
 |   iov.iov_base = ®s; | 
 |   iov.iov_len = sizeof (gdb_gregset_t); | 
 |  | 
 |   if (ptrace (PTRACE_GETREGSET, tid, NT_PRSTATUS, (void *) &iov) < 0) | 
 |     perror_with_name (_("Couldn't get general registers")); | 
 |   else | 
 |     { | 
 |       arc_linux_collect_gregset (NULL, regcache, regnum, regs, 0); | 
 |  | 
 |       if (ptrace (PTRACE_SETREGSET, tid, NT_PRSTATUS, (void *) &iov) < 0) | 
 | 	perror_with_name (_("Couldn't write general registers")); | 
 |     } | 
 | } | 
 |  | 
 | #ifdef ARC_HAS_V2_REGSET | 
 | /* Store ARC v2 registers from REGCACHE into the target process/thread.  */ | 
 |  | 
 | static void | 
 | store_v2_regs (const struct regcache *regcache, int regnum) | 
 | { | 
 |   const int tid = get_ptrace_pid (regcache->ptid ()); | 
 |   struct iovec iov; | 
 |   bfd_byte v2_buffer[ARC_LINUX_SIZEOF_V2_REGSET]; | 
 |  | 
 |   iov.iov_base = &v2_buffer; | 
 |   iov.iov_len = ARC_LINUX_SIZEOF_V2_REGSET; | 
 |  | 
 |   if (ptrace (PTRACE_GETREGSET, tid, NT_ARC_V2, (void *) &iov) < 0) | 
 |     perror_with_name (_("Couldn't get ARC HS registers")); | 
 |   else | 
 |     { | 
 |       arc_linux_collect_v2_regset (NULL, regcache, regnum, v2_buffer, 0); | 
 |  | 
 |       if (ptrace (PTRACE_SETREGSET, tid, NT_ARC_V2, (void *) &iov) < 0) | 
 | 	perror_with_name (_("Couldn't write ARC HS registers")); | 
 |     } | 
 | } | 
 | #endif | 
 |  | 
 | /* Target operation: Read REGNUM register (all registers if REGNUM == -1) | 
 |    from target process into REGCACHE.  */ | 
 |  | 
 | void | 
 | arc_linux_nat_target::fetch_registers (struct regcache *regcache, int regnum) | 
 | { | 
 |  | 
 |   if (regnum == -1 || regnum <= ARC_LAST_REGNUM) | 
 |     fetch_gregs (regcache, regnum); | 
 |  | 
 | #ifdef ARC_HAS_V2_REGSET | 
 |   if (regnum == -1 | 
 |       || regnum == ARC_R30_REGNUM | 
 |       || regnum == ARC_R58_REGNUM | 
 |       || regnum == ARC_R59_REGNUM) | 
 |       fetch_v2_regs (regcache, regnum); | 
 | #endif | 
 | } | 
 |  | 
 | /* Target operation: Store REGNUM register (all registers if REGNUM == -1) | 
 |    to the target process from REGCACHE.  */ | 
 |  | 
 | void | 
 | arc_linux_nat_target::store_registers (struct regcache *regcache, int regnum) | 
 | { | 
 |   if (regnum == -1 || regnum <= ARC_LAST_REGNUM) | 
 |     store_gregs (regcache, regnum); | 
 |  | 
 | #ifdef ARC_HAS_V2_REGSET | 
 |   if (regnum == -1 | 
 |       || regnum == ARC_R30_REGNUM | 
 |       || regnum == ARC_R58_REGNUM | 
 |       || regnum == ARC_R59_REGNUM) | 
 |     store_v2_regs (regcache, regnum); | 
 | #endif | 
 | } | 
 |  | 
 | /* Copy general purpose register(s) from REGCACHE into regset GREGS. | 
 |    This function is exported to proc-service.c  */ | 
 |  | 
 | void | 
 | fill_gregset (const struct regcache *regcache, | 
 | 	      gdb_gregset_t *gregs, int regnum) | 
 | { | 
 |   arc_linux_collect_gregset (NULL, regcache, regnum, gregs, 0); | 
 | } | 
 |  | 
 | /* Copy all the general purpose registers from regset GREGS into REGCACHE. | 
 |    This function is exported to proc-service.c.  */ | 
 |  | 
 | void | 
 | supply_gregset (struct regcache *regcache, const gdb_gregset_t *gregs) | 
 | { | 
 |   arc_linux_supply_gregset (NULL, regcache, -1, gregs, 0); | 
 | } | 
 |  | 
 | /* ARC doesn't have separate FP registers.  This function is exported | 
 |    to proc-service.c.  */ | 
 |  | 
 | void | 
 | fill_fpregset (const struct regcache *regcache, | 
 | 	       gdb_fpregset_t *fpregsetp, int regnum) | 
 | { | 
 |   arc_linux_nat_debug_printf ("called"); | 
 | } | 
 |  | 
 | /* ARC doesn't have separate FP registers.  This function is exported | 
 |    to proc-service.c.  */ | 
 |  | 
 | void | 
 | supply_fpregset (struct regcache *regcache, const gdb_fpregset_t *fpregsetp) | 
 | { | 
 |   arc_linux_nat_debug_printf ("called"); | 
 | } | 
 |  | 
 | /* Implement the "read_description" method of linux_nat_target.  */ | 
 |  | 
 | const struct target_desc * | 
 | arc_linux_nat_target::read_description () | 
 | { | 
 |   /* This is a native target, hence description is hardcoded.  */ | 
 | #ifdef __ARCHS__ | 
 |   arc_arch_features features (4, ARC_ISA_ARCV2); | 
 | #else | 
 |   arc_arch_features features (4, ARC_ISA_ARCV1); | 
 | #endif | 
 |   return arc_lookup_target_description (features); | 
 | } | 
 |  | 
 | /* As described in arc_linux_collect_gregset(), we need to write resume-PC | 
 |    to ERET.  However by default GDB for native targets doesn't write | 
 |    registers if they haven't been changed.  This is a callback called by | 
 |    generic GDB, and in this callback we have to rewrite PC value so it | 
 |    would force rewrite of register on target.  It seems that the only | 
 |    other arch that utilizes this hook is x86/x86-64 for HW breakpoint | 
 |    support.  But then, AFAIK no other arch has this stop_pc/eret | 
 |    complexity. | 
 |  | 
 |    No better way was found, other than this fake write of register value, | 
 |    to force GDB into writing register to target.  Is there any?  */ | 
 |  | 
 | void | 
 | arc_linux_nat_target::low_prepare_to_resume (struct lwp_info *lwp) | 
 | { | 
 |   /* When new processes and threads are created we do not have the address | 
 |      space for them and calling get_thread_regcache will cause an internal | 
 |      error in GDB.  It looks like that checking for last_resume_kind is the | 
 |      sensible way to determine processes for which we cannot get regcache. | 
 |      Ultimately, a better way would be removing the need for | 
 |      low_prepare_to_resume in the first place.  */ | 
 |   if (lwp->last_resume_kind == resume_stop) | 
 |     return; | 
 |  | 
 |   struct regcache *regcache = get_thread_regcache (this, lwp->ptid); | 
 |   struct gdbarch *gdbarch = regcache->arch (); | 
 |  | 
 |   /* Read current PC value, then write it back.  It is required to call | 
 |      invalidate(), otherwise GDB will note that new value is equal to old | 
 |      value and will skip write.  */ | 
 |   ULONGEST new_pc; | 
 |   regcache_cooked_read_unsigned (regcache, gdbarch_pc_regnum (gdbarch), | 
 | 				 &new_pc); | 
 |   regcache->invalidate (gdbarch_pc_regnum (gdbarch)); | 
 |   regcache_cooked_write_unsigned (regcache, gdbarch_pc_regnum (gdbarch), | 
 | 				  new_pc); | 
 | } | 
 |  | 
 | /* Fetch the thread-local storage pointer for libthread_db.  Note that | 
 |    this function is not called from GDB, but is called from libthread_db. | 
 |    This is required to debug multithreaded applications with NPTL.  */ | 
 |  | 
 | ps_err_e | 
 | ps_get_thread_area (struct ps_prochandle *ph, lwpid_t lwpid, int idx, | 
 | 		    void **base) | 
 | { | 
 |   arc_linux_nat_debug_printf ("called"); | 
 |  | 
 |   if (ptrace (PTRACE_GET_THREAD_AREA, lwpid, NULL, base) != 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.  */ | 
 |   *base = (void *) ((char *) *base - idx); | 
 |  | 
 |   return PS_OK; | 
 | } | 
 |  | 
 | /* Suppress warning from -Wmissing-prototypes.  */ | 
 | void _initialize_arc_linux_nat (); | 
 | void | 
 | _initialize_arc_linux_nat () | 
 | { | 
 |   /* Register the target.  */ | 
 |   linux_target = &the_arc_linux_nat_target; | 
 |   add_inf_child_target (&the_arc_linux_nat_target); | 
 | } |