/* Native-dependent code for BSD Unix running on ARM's, for GDB. Copyright (C) 1988-2016 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 . */ #ifndef _KERNTYPES #define _KERNTYPES #endif #include "defs.h" #include "gdbcore.h" #include "inferior.h" #include "regcache.h" #include "target.h" #include "nbsd-nat.h" #include #include #include #include #include /* Support for debugging kernel virtual memory images. */ #include #include "arm-tdep.h" #include "inf-ptrace.h" #include "bsd-kvm.h" #ifndef HAVE_GREGSET_T typedef struct reg gregset_t; #endif #ifndef HAVE_FPREGSET_T typedef struct fpreg fpregset_t; #endif #include "gregset.h" extern int arm_apcs_32; static int armnbsd_supply_pcb (struct regcache *regcache, struct pcb *pcb) { struct switchframe sf; /* The following is true for NetBSD/arm32 in 5.0 and after: The pcb contains r8-r13 (sp) at the point of context switch in cpu_switchto() or call of dumpsys(). At that point we have a stack frame as described by `struct switchframe', which for NetBSD/arm32 has the following layout: r4 ascending. r5 | r6 | r7 \|/ old sp pc we reconstruct the register state as it would look when we just returned from cpu_switchto() or dumpsys(). */ if (!arm_apcs_32) return 0; /* The stack pointer shouldn't be zero. */ if (pcb->pcb_un.un_32.pcb32_sp == 0) return 0; read_memory (pcb->pcb_un.un_32.pcb32_sp, (gdb_byte *) &sf, sizeof sf); regcache_raw_supply (regcache, ARM_PC_REGNUM, &sf.sf_pc); regcache_raw_supply (regcache, ARM_SP_REGNUM, &pcb->pcb_un.un_32.pcb32_sp); regcache_raw_supply (regcache, 12, &pcb->pcb_un.un_32.pcb32_r12); regcache_raw_supply (regcache, 11, &pcb->pcb_un.un_32.pcb32_r11); regcache_raw_supply (regcache, 10, &pcb->pcb_un.un_32.pcb32_r10); regcache_raw_supply (regcache, 9, &pcb->pcb_un.un_32.pcb32_r9); regcache_raw_supply (regcache, 8, &pcb->pcb_un.un_32.pcb32_r8); regcache_raw_supply (regcache, 7, &sf.sf_r7); regcache_raw_supply (regcache, 6, &sf.sf_r6); regcache_raw_supply (regcache, 5, &sf.sf_r5); regcache_raw_supply (regcache, 4, &sf.sf_r4); return 1; } static void arm_supply_gregset (struct regcache *regcache, struct reg *gregset) { int regno; CORE_ADDR r_pc; /* Integer registers. */ for (regno = ARM_A1_REGNUM; regno < ARM_SP_REGNUM; regno++) regcache_raw_supply (regcache, regno, (char *) &gregset->r[regno]); regcache_raw_supply (regcache, ARM_SP_REGNUM, (char *) &gregset->r_sp); regcache_raw_supply (regcache, ARM_LR_REGNUM, (char *) &gregset->r_lr); /* This is ok: we're running native... */ r_pc = gdbarch_addr_bits_remove (get_regcache_arch (regcache), gregset->r_pc); regcache_raw_supply (regcache, ARM_PC_REGNUM, (char *) &r_pc); if (arm_apcs_32) regcache_raw_supply (regcache, ARM_PS_REGNUM, (char *) &gregset->r_cpsr); else regcache_raw_supply (regcache, ARM_PS_REGNUM, (char *) &gregset->r_pc); } static void arm_supply_vfpregset (struct regcache *regcache, struct fpreg *vfpregset) { int regno; for (regno = 0; regno < 16; regno++) regcache_raw_supply (regcache, regno + ARM_D0_REGNUM, (char *) vfpregset->fpr_vfp.vfp_regs + 8*regno); regcache_raw_supply (regcache, ARM_FPSCR_REGNUM, (char *) &vfpregset->fpr_vfp.vfp_fpscr); } void fill_gregset (const struct regcache *regcache, gregset_t *gregsetp, int regno) { if (-1 == regno) { int regnum; for (regnum = ARM_A1_REGNUM; regnum < ARM_SP_REGNUM; regnum++) regcache_raw_collect (regcache, regnum, (char *) &gregsetp->r[regnum]); } else if (regno >= ARM_A1_REGNUM && regno < ARM_SP_REGNUM) regcache_raw_collect (regcache, regno, (char *) &gregsetp->r[regno]); if (ARM_SP_REGNUM == regno || -1 == regno) regcache_raw_collect (regcache, ARM_SP_REGNUM, (char *) &gregsetp->r_sp); if (ARM_LR_REGNUM == regno || -1 == regno) regcache_raw_collect (regcache, ARM_LR_REGNUM, (char *) &gregsetp->r_lr); if (ARM_PC_REGNUM == regno || -1 == regno) regcache_raw_collect (regcache, ARM_PC_REGNUM, (char *) &gregsetp->r_pc); if (ARM_PS_REGNUM == regno || -1 == regno) { if (arm_apcs_32) regcache_raw_collect (regcache, ARM_PS_REGNUM, (char *) &gregsetp->r_cpsr); else regcache_raw_collect (regcache, ARM_PS_REGNUM, (char *) &gregsetp->r_pc); } } void fill_fpregset (const struct regcache *regcache, fpregset_t *vfpregsetp, int regno) { if (-1 == regno) { int regnum; for (regnum = 0; regnum <= 15; regnum++) regcache_raw_collect(regcache, regnum + ARM_D0_REGNUM, (char *) vfpregsetp->fpr_vfp.vfp_regs + 8*regnum); } else if (regno >= ARM_D0_REGNUM && regno <= ARM_D0_REGNUM + 15) regcache_raw_collect(regcache, regno, (char *) vfpregsetp->fpr_vfp.vfp_regs + 8 * (regno - ARM_D0_REGNUM)); if (ARM_FPSCR_REGNUM == regno || -1 == regno) regcache_raw_collect (regcache, ARM_FPSCR_REGNUM, (char *) &vfpregsetp->fpr_vfp.vfp_fpscr); } void supply_gregset (struct regcache *regcache, const gdb_gregset_t *gregsetp) { arm_supply_gregset (regcache, (struct reg *)gregsetp); } void supply_fpregset (struct regcache *regcache, const gdb_fpregset_t *fpregsetp) { arm_supply_vfpregset (regcache, (struct fpreg *)fpregsetp); } static void fetch_register (struct regcache *regcache, int regno) { struct reg inferior_registers; int ret; ret = ptrace (PT_GETREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) { warning (_("unable to fetch general register")); return; } switch (regno) { case ARM_SP_REGNUM: regcache_raw_supply (regcache, ARM_SP_REGNUM, (char *) &inferior_registers.r_sp); break; case ARM_LR_REGNUM: regcache_raw_supply (regcache, ARM_LR_REGNUM, (char *) &inferior_registers.r_lr); break; case ARM_PC_REGNUM: /* This is ok: we're running native... */ inferior_registers.r_pc = gdbarch_addr_bits_remove (get_regcache_arch (regcache), inferior_registers.r_pc); regcache_raw_supply (regcache, ARM_PC_REGNUM, (char *) &inferior_registers.r_pc); break; case ARM_PS_REGNUM: if (arm_apcs_32) regcache_raw_supply (regcache, ARM_PS_REGNUM, (char *) &inferior_registers.r_cpsr); else regcache_raw_supply (regcache, ARM_PS_REGNUM, (char *) &inferior_registers.r_pc); break; default: regcache_raw_supply (regcache, regno, (char *) &inferior_registers.r[regno]); break; } } static void fetch_regs (struct regcache *regcache) { struct reg inferior_registers; int ret; int regno; ret = ptrace (PT_GETREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) { warning (_("unable to fetch general registers")); return; } arm_supply_gregset (regcache, &inferior_registers); } static void fetch_fp_register (struct regcache *regcache, int regno) { struct fpreg inferior_fp_registers; int ret; ret = ptrace (PT_GETFPREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_fp_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) { warning (_("unable to fetch floating-point register")); return; } switch (regno) { case ARM_FPSCR_REGNUM: regcache_raw_supply (regcache, ARM_FPSCR_REGNUM, (char *) &inferior_fp_registers.fpr_vfp.vfp_fpscr); break; default: regcache_raw_supply (regcache, regno, (char *) inferior_fp_registers.fpr_vfp.vfp_regs + 8 * (regno - ARM_D0_REGNUM)); break; } } static void fetch_fp_regs (struct regcache *regcache) { struct fpreg inferior_fp_registers; int ret; int regno; ret = ptrace (PT_GETFPREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_fp_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) { warning (_("unable to fetch floating-point registers")); return; } arm_supply_vfpregset (regcache, &inferior_fp_registers); } static void armnbsd_fetch_registers (struct target_ops *ops, struct regcache *regcache, int regno) { if (regno >= 0) { if (regno >= ARM_D0_REGNUM && regno <= ARM_FPSCR_REGNUM) fetch_fp_register (regcache, regno); else fetch_register (regcache, regno); } else { fetch_regs (regcache); fetch_fp_regs (regcache); } } static void store_register (const struct regcache *regcache, int regno) { struct gdbarch *gdbarch = get_regcache_arch (regcache); struct reg inferior_registers; int ret; ret = ptrace (PT_GETREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) { warning (_("unable to fetch general registers")); return; } switch (regno) { case ARM_SP_REGNUM: regcache_raw_collect (regcache, ARM_SP_REGNUM, (char *) &inferior_registers.r_sp); break; case ARM_LR_REGNUM: regcache_raw_collect (regcache, ARM_LR_REGNUM, (char *) &inferior_registers.r_lr); break; case ARM_PC_REGNUM: if (arm_apcs_32) regcache_raw_collect (regcache, ARM_PC_REGNUM, (char *) &inferior_registers.r_pc); else { unsigned pc_val; regcache_raw_collect (regcache, ARM_PC_REGNUM, (char *) &pc_val); pc_val = gdbarch_addr_bits_remove (gdbarch, pc_val); inferior_registers.r_pc ^= gdbarch_addr_bits_remove (gdbarch, inferior_registers.r_pc); inferior_registers.r_pc |= pc_val; } break; case ARM_PS_REGNUM: if (arm_apcs_32) regcache_raw_collect (regcache, ARM_PS_REGNUM, (char *) &inferior_registers.r_cpsr); else { unsigned psr_val; regcache_raw_collect (regcache, ARM_PS_REGNUM, (char *) &psr_val); psr_val ^= gdbarch_addr_bits_remove (gdbarch, psr_val); inferior_registers.r_pc = gdbarch_addr_bits_remove (gdbarch, inferior_registers.r_pc); inferior_registers.r_pc |= psr_val; } break; default: regcache_raw_collect (regcache, regno, (char *) &inferior_registers.r[regno]); break; } ret = ptrace (PT_SETREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) warning (_("unable to write register %d to inferior"), regno); } static void store_regs (const struct regcache *regcache) { struct gdbarch *gdbarch = get_regcache_arch (regcache); struct reg inferior_registers; int ret; int regno; for (regno = ARM_A1_REGNUM; regno < ARM_SP_REGNUM; regno++) regcache_raw_collect (regcache, regno, (char *) &inferior_registers.r[regno]); regcache_raw_collect (regcache, ARM_SP_REGNUM, (char *) &inferior_registers.r_sp); regcache_raw_collect (regcache, ARM_LR_REGNUM, (char *) &inferior_registers.r_lr); if (arm_apcs_32) { regcache_raw_collect (regcache, ARM_PC_REGNUM, (char *) &inferior_registers.r_pc); regcache_raw_collect (regcache, ARM_PS_REGNUM, (char *) &inferior_registers.r_cpsr); } else { unsigned pc_val; unsigned psr_val; regcache_raw_collect (regcache, ARM_PC_REGNUM, (char *) &pc_val); regcache_raw_collect (regcache, ARM_PS_REGNUM, (char *) &psr_val); pc_val = gdbarch_addr_bits_remove (gdbarch, pc_val); psr_val ^= gdbarch_addr_bits_remove (gdbarch, psr_val); inferior_registers.r_pc = pc_val | psr_val; } ret = ptrace (PT_SETREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) warning (_("unable to store general registers")); } static void store_fp_register (const struct regcache *regcache, int regno) { struct fpreg inferior_fp_registers; int ret; ret = ptrace (PT_GETFPREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_fp_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) { warning (_("unable to fetch floating-point registers")); return; } switch (regno) { case ARM_FPSCR_REGNUM: regcache_raw_collect (regcache, ARM_FPS_REGNUM, (char *) &inferior_fp_registers.fpr_vfp.vfp_fpscr); break; default: regcache_raw_collect (regcache, regno, (char *) inferior_fp_registers.fpr_vfp.vfp_regs + 8 * (regno - ARM_D0_REGNUM)); break; } ret = ptrace (PT_SETFPREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_fp_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) warning (_("unable to write register %d to inferior"), regno); } static void store_fp_regs (const struct regcache *regcache) { struct fpreg inferior_fp_registers; int ret; int regno; for (regno = 0; regno <= 15; regno++) regcache_raw_collect (regcache, regno + ARM_D0_REGNUM, (char *) inferior_fp_registers.fpr_vfp.vfp_regs + 8 * regno); regcache_raw_collect (regcache, ARM_FPSCR_REGNUM, (char *) &inferior_fp_registers.fpr_vfp.vfp_fpscr); ret = ptrace (PT_SETFPREGS, ptid_get_pid (inferior_ptid), (PTRACE_TYPE_ARG3) &inferior_fp_registers, ptid_get_lwp(inferior_ptid)); if (ret < 0) warning (_("unable to store floating-point registers")); } static void armnbsd_store_registers (struct target_ops *ops, struct regcache *regcache, int regno) { if (regno >= 0) { if (regno >= ARM_D0_REGNUM && regno <= ARM_FPSCR_REGNUM) store_fp_register (regcache, regno); else store_register (regcache, regno); } else { store_regs (regcache); store_fp_regs (regcache); } } struct md_core { struct reg intreg; struct fpreg freg; }; static void fetch_core_registers (struct regcache *regcache, char *core_reg_sect, unsigned core_reg_size, int which, CORE_ADDR ignore) { struct md_core *core_reg = (struct md_core *) core_reg_sect; int regno; CORE_ADDR r_pc; arm_supply_gregset (regcache, &core_reg->intreg); arm_supply_vfpregset (regcache, &core_reg->freg); } static void fetch_elfcore_registers (struct regcache *regcache, char *core_reg_sect, unsigned core_reg_size, int which, CORE_ADDR ignore) { struct reg gregset; struct fpreg vfpregset; switch (which) { case 0: /* Integer registers. */ if (core_reg_size != sizeof (struct reg)) warning (_("wrong size of register set in core file")); else { /* The memcpy may be unnecessary, but we can't really be sure of the alignment of the data in the core file. */ memcpy (&gregset, core_reg_sect, sizeof (gregset)); arm_supply_gregset (regcache, &gregset); } break; case 2: if (core_reg_size != sizeof (struct fpreg)) warning (_("wrong size of FPA register set in core file")); else { /* The memcpy may be unnecessary, but we can't really be sure of the alignment of the data in the core file. */ memcpy (&vfpregset, core_reg_sect, sizeof (vfpregset)); arm_supply_vfpregset (regcache, &vfpregset); } break; default: /* Don't know what kind of register request this is; just ignore it. */ break; } } static struct core_fns arm_netbsd_core_fns = { bfd_target_unknown_flavour, /* core_flovour. */ default_check_format, /* check_format. */ default_core_sniffer, /* core_sniffer. */ fetch_core_registers, /* core_read_registers. */ NULL }; static struct core_fns arm_netbsd_elfcore_fns = { bfd_target_elf_flavour, /* core_flovour. */ default_check_format, /* check_format. */ default_core_sniffer, /* core_sniffer. */ fetch_elfcore_registers, /* core_read_registers. */ NULL }; void _initialize_arm_netbsd_nat (void) { struct target_ops *t; t = inf_ptrace_target (); t->to_fetch_registers = armnbsd_fetch_registers; t->to_store_registers = armnbsd_store_registers; t->to_pid_to_exec_file = nbsd_pid_to_exec_file; add_target (t); /* Support debugging kernel virtual memory images. */ bsd_kvm_add_target (armnbsd_supply_pcb); deprecated_add_core_fns (&arm_netbsd_core_fns); deprecated_add_core_fns (&arm_netbsd_elfcore_fns); }