gdb/s390-linux-nat.c - gdb
Global variables defined
Data types defined
Functions defined
Macros defined
Source code
- #include "defs.h"
- #include "regcache.h"
- #include "inferior.h"
- #include "target.h"
- #include "linux-nat.h"
- #include "auxv.h"
- #include "gregset.h"
- #include "regset.h"
- #include "s390-linux-tdep.h"
- #include "elf/common.h"
- #include <asm/ptrace.h>
- #include <sys/ptrace.h>
- #include <asm/types.h>
- #include <sys/procfs.h>
- #include <sys/ucontext.h>
- #include <elf.h>
- #ifndef PTRACE_GETREGSET
- #define PTRACE_GETREGSET 0x4204
- #endif
- #ifndef PTRACE_SETREGSET
- #define PTRACE_SETREGSET 0x4205
- #endif
- static int have_regset_last_break = 0;
- static int have_regset_system_call = 0;
- static int have_regset_tdb = 0;
- #ifdef __s390x__
- static const struct regcache_map_entry s390_64_regmap_gregset[] =
- {
-
- { 2, REGCACHE_MAP_SKIP, 8 },
- { 1, S390_R0_UPPER_REGNUM, 4 }, { 1, S390_R0_REGNUM, 4 },
- { 1, S390_R1_UPPER_REGNUM, 4 }, { 1, S390_R1_REGNUM, 4 },
- { 1, S390_R2_UPPER_REGNUM, 4 }, { 1, S390_R2_REGNUM, 4 },
- { 1, S390_R3_UPPER_REGNUM, 4 }, { 1, S390_R3_REGNUM, 4 },
- { 1, S390_R4_UPPER_REGNUM, 4 }, { 1, S390_R4_REGNUM, 4 },
- { 1, S390_R5_UPPER_REGNUM, 4 }, { 1, S390_R5_REGNUM, 4 },
- { 1, S390_R6_UPPER_REGNUM, 4 }, { 1, S390_R6_REGNUM, 4 },
- { 1, S390_R7_UPPER_REGNUM, 4 }, { 1, S390_R7_REGNUM, 4 },
- { 1, S390_R8_UPPER_REGNUM, 4 }, { 1, S390_R8_REGNUM, 4 },
- { 1, S390_R9_UPPER_REGNUM, 4 }, { 1, S390_R9_REGNUM, 4 },
- { 1, S390_R10_UPPER_REGNUM, 4 }, { 1, S390_R10_REGNUM, 4 },
- { 1, S390_R11_UPPER_REGNUM, 4 }, { 1, S390_R11_REGNUM, 4 },
- { 1, S390_R12_UPPER_REGNUM, 4 }, { 1, S390_R12_REGNUM, 4 },
- { 1, S390_R13_UPPER_REGNUM, 4 }, { 1, S390_R13_REGNUM, 4 },
- { 1, S390_R14_UPPER_REGNUM, 4 }, { 1, S390_R14_REGNUM, 4 },
- { 1, S390_R15_UPPER_REGNUM, 4 }, { 1, S390_R15_REGNUM, 4 },
- { 16, S390_A0_REGNUM, 4 },
- { 1, REGCACHE_MAP_SKIP, 4 }, { 1, S390_ORIG_R2_REGNUM, 4 },
- { 0 }
- };
- static const struct regset s390_64_gregset =
- {
- s390_64_regmap_gregset,
- regcache_supply_regset,
- regcache_collect_regset
- };
- #define S390_PSWM_OFFSET 0
- #define S390_PSWA_OFFSET 8
- #endif
- void
- supply_gregset (struct regcache *regcache, const gregset_t *regp)
- {
- #ifdef __s390x__
- struct gdbarch *gdbarch = get_regcache_arch (regcache);
- if (gdbarch_ptr_bit (gdbarch) == 32)
- {
- enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
- ULONGEST pswm, pswa;
- gdb_byte buf[4];
- regcache_supply_regset (&s390_64_gregset, regcache, -1,
- regp, sizeof (gregset_t));
- pswm = extract_unsigned_integer ((const gdb_byte *) regp
- + S390_PSWM_OFFSET, 8, byte_order);
- pswa = extract_unsigned_integer ((const gdb_byte *) regp
- + S390_PSWA_OFFSET, 8, byte_order);
- store_unsigned_integer (buf, 4, byte_order, (pswm >> 32) | 0x80000);
- regcache_raw_supply (regcache, S390_PSWM_REGNUM, buf);
- store_unsigned_integer (buf, 4, byte_order,
- (pswa & 0x7fffffff) | (pswm & 0x80000000));
- regcache_raw_supply (regcache, S390_PSWA_REGNUM, buf);
- return;
- }
- #endif
- regcache_supply_regset (&s390_gregset, regcache, -1, regp,
- sizeof (gregset_t));
- }
- void
- fill_gregset (const struct regcache *regcache, gregset_t *regp, int regno)
- {
- #ifdef __s390x__
- struct gdbarch *gdbarch = get_regcache_arch (regcache);
- if (gdbarch_ptr_bit (gdbarch) == 32)
- {
- regcache_collect_regset (&s390_64_gregset, regcache, regno,
- regp, sizeof (gregset_t));
- if (regno == -1
- || regno == S390_PSWM_REGNUM || regno == S390_PSWA_REGNUM)
- {
- enum bfd_endian byte_order = gdbarch_byte_order (gdbarch);
- ULONGEST pswa, pswm;
- gdb_byte buf[4];
- regcache_raw_collect (regcache, S390_PSWM_REGNUM, buf);
- pswm = extract_unsigned_integer (buf, 4, byte_order);
- regcache_raw_collect (regcache, S390_PSWA_REGNUM, buf);
- pswa = extract_unsigned_integer (buf, 4, byte_order);
- if (regno == -1 || regno == S390_PSWM_REGNUM)
- store_unsigned_integer ((gdb_byte *) regp + S390_PSWM_OFFSET, 8,
- byte_order, ((pswm & 0xfff7ffff) << 32) |
- (pswa & 0x80000000));
- if (regno == -1 || regno == S390_PSWA_REGNUM)
- store_unsigned_integer ((gdb_byte *) regp + S390_PSWA_OFFSET, 8,
- byte_order, pswa & 0x7fffffff);
- }
- return;
- }
- #endif
- regcache_collect_regset (&s390_gregset, regcache, regno, regp,
- sizeof (gregset_t));
- }
- void
- supply_fpregset (struct regcache *regcache, const fpregset_t *regp)
- {
- regcache_supply_regset (&s390_fpregset, regcache, -1, regp,
- sizeof (fpregset_t));
- }
- void
- fill_fpregset (const struct regcache *regcache, fpregset_t *regp, int regno)
- {
- regcache_collect_regset (&s390_fpregset, regcache, regno, regp,
- sizeof (fpregset_t));
- }
- static int
- s390_inferior_tid (void)
- {
-
- int tid = ptid_get_lwp (inferior_ptid);
- if (tid == 0)
- tid = ptid_get_pid (inferior_ptid);
- return tid;
- }
- static void
- fetch_regs (struct regcache *regcache, int tid)
- {
- gregset_t regs;
- ptrace_area parea;
- parea.len = sizeof (regs);
- parea.process_addr = (addr_t) ®s;
- parea.kernel_addr = offsetof (struct user_regs_struct, psw);
- if (ptrace (PTRACE_PEEKUSR_AREA, tid, (long) &parea) < 0)
- perror_with_name (_("Couldn't get registers"));
- supply_gregset (regcache, (const gregset_t *) ®s);
- }
- static void
- store_regs (const struct regcache *regcache, int tid, int regnum)
- {
- gregset_t regs;
- ptrace_area parea;
- parea.len = sizeof (regs);
- parea.process_addr = (addr_t) ®s;
- parea.kernel_addr = offsetof (struct user_regs_struct, psw);
- if (ptrace (PTRACE_PEEKUSR_AREA, tid, (long) &parea) < 0)
- perror_with_name (_("Couldn't get registers"));
- fill_gregset (regcache, ®s, regnum);
- if (ptrace (PTRACE_POKEUSR_AREA, tid, (long) &parea) < 0)
- perror_with_name (_("Couldn't write registers"));
- }
- static void
- fetch_fpregs (struct regcache *regcache, int tid)
- {
- fpregset_t fpregs;
- ptrace_area parea;
- parea.len = sizeof (fpregs);
- parea.process_addr = (addr_t) &fpregs;
- parea.kernel_addr = offsetof (struct user_regs_struct, fp_regs);
- if (ptrace (PTRACE_PEEKUSR_AREA, tid, (long) &parea) < 0)
- perror_with_name (_("Couldn't get floating point status"));
- supply_fpregset (regcache, (const fpregset_t *) &fpregs);
- }
- static void
- store_fpregs (const struct regcache *regcache, int tid, int regnum)
- {
- fpregset_t fpregs;
- ptrace_area parea;
- parea.len = sizeof (fpregs);
- parea.process_addr = (addr_t) &fpregs;
- parea.kernel_addr = offsetof (struct user_regs_struct, fp_regs);
- if (ptrace (PTRACE_PEEKUSR_AREA, tid, (long) &parea) < 0)
- perror_with_name (_("Couldn't get floating point status"));
- fill_fpregset (regcache, &fpregs, regnum);
- if (ptrace (PTRACE_POKEUSR_AREA, tid, (long) &parea) < 0)
- perror_with_name (_("Couldn't write floating point status"));
- }
- static void
- fetch_regset (struct regcache *regcache, int tid,
- int regset_id, int regsize, const struct regset *regset)
- {
- gdb_byte *buf = alloca (regsize);
- struct iovec iov;
- iov.iov_base = buf;
- iov.iov_len = regsize;
- if (ptrace (PTRACE_GETREGSET, tid, (long) regset_id, (long) &iov) < 0)
- {
- if (errno == ENODATA)
- regcache_supply_regset (regset, regcache, -1, NULL, regsize);
- else
- perror_with_name (_("Couldn't get register set"));
- }
- else
- regcache_supply_regset (regset, regcache, -1, buf, regsize);
- }
- static void
- store_regset (struct regcache *regcache, int tid,
- int regset_id, int regsize, const struct regset *regset)
- {
- gdb_byte *buf = alloca (regsize);
- struct iovec iov;
- iov.iov_base = buf;
- iov.iov_len = regsize;
- if (ptrace (PTRACE_GETREGSET, tid, (long) regset_id, (long) &iov) < 0)
- perror_with_name (_("Couldn't get register set"));
- regcache_collect_regset (regset, regcache, -1, buf, regsize);
- if (ptrace (PTRACE_SETREGSET, tid, (long) regset_id, (long) &iov) < 0)
- perror_with_name (_("Couldn't set register set"));
- }
- static int
- check_regset (int tid, int regset, int regsize)
- {
- gdb_byte *buf = alloca (regsize);
- struct iovec iov;
- iov.iov_base = buf;
- iov.iov_len = regsize;
- if (ptrace (PTRACE_GETREGSET, tid, (long) regset, (long) &iov) >= 0
- || errno == ENODATA)
- return 1;
- return 0;
- }
- static void
- s390_linux_fetch_inferior_registers (struct target_ops *ops,
- struct regcache *regcache, int regnum)
- {
- int tid = s390_inferior_tid ();
- if (regnum == -1 || S390_IS_GREGSET_REGNUM (regnum))
- fetch_regs (regcache, tid);
- if (regnum == -1 || S390_IS_FPREGSET_REGNUM (regnum))
- fetch_fpregs (regcache, tid);
- if (have_regset_last_break)
- if (regnum == -1 || regnum == S390_LAST_BREAK_REGNUM)
- fetch_regset (regcache, tid, NT_S390_LAST_BREAK, 8,
- (gdbarch_ptr_bit (get_regcache_arch (regcache)) == 32
- ? &s390_last_break_regset : &s390x_last_break_regset));
- if (have_regset_system_call)
- if (regnum == -1 || regnum == S390_SYSTEM_CALL_REGNUM)
- fetch_regset (regcache, tid, NT_S390_SYSTEM_CALL, 4,
- &s390_system_call_regset);
- if (have_regset_tdb)
- if (regnum == -1 || S390_IS_TDBREGSET_REGNUM (regnum))
- fetch_regset (regcache, tid, NT_S390_TDB, s390_sizeof_tdbregset,
- &s390_tdb_regset);
- }
- static void
- s390_linux_store_inferior_registers (struct target_ops *ops,
- struct regcache *regcache, int regnum)
- {
- int tid = s390_inferior_tid ();
- if (regnum == -1 || S390_IS_GREGSET_REGNUM (regnum))
- store_regs (regcache, tid, regnum);
- if (regnum == -1 || S390_IS_FPREGSET_REGNUM (regnum))
- store_fpregs (regcache, tid, regnum);
-
- if (have_regset_system_call)
- if (regnum == -1 || regnum == S390_SYSTEM_CALL_REGNUM)
- store_regset (regcache, tid, NT_S390_SYSTEM_CALL, 4,
- &s390_system_call_regset);
- }
- struct watch_area
- {
- struct watch_area *next;
- CORE_ADDR lo_addr;
- CORE_ADDR hi_addr;
- };
- static struct watch_area *watch_base = NULL;
- static int
- s390_stopped_by_watchpoint (struct target_ops *ops)
- {
- per_lowcore_bits per_lowcore;
- ptrace_area parea;
- int result;
-
- if (!watch_base)
- return 0;
- parea.len = sizeof (per_lowcore);
- parea.process_addr = (addr_t) & per_lowcore;
- parea.kernel_addr = offsetof (struct user_regs_struct, per_info.lowcore);
- if (ptrace (PTRACE_PEEKUSR_AREA, s390_inferior_tid (), &parea) < 0)
- perror_with_name (_("Couldn't retrieve watchpoint status"));
- result = (per_lowcore.perc_storage_alteration == 1
- && per_lowcore.perc_store_real_address == 0);
- if (result)
- {
-
- memset (&per_lowcore, 0, sizeof (per_lowcore));
- if (ptrace (PTRACE_POKEUSR_AREA, s390_inferior_tid (), &parea) < 0)
- perror_with_name (_("Couldn't clear watchpoint status"));
- }
- return result;
- }
- static void
- s390_fix_watch_points (struct lwp_info *lp)
- {
- int tid;
- per_struct per_info;
- ptrace_area parea;
- CORE_ADDR watch_lo_addr = (CORE_ADDR)-1, watch_hi_addr = 0;
- struct watch_area *area;
- tid = ptid_get_lwp (lp->ptid);
- if (tid == 0)
- tid = ptid_get_pid (lp->ptid);
- for (area = watch_base; area; area = area->next)
- {
- watch_lo_addr = min (watch_lo_addr, area->lo_addr);
- watch_hi_addr = max (watch_hi_addr, area->hi_addr);
- }
- parea.len = sizeof (per_info);
- parea.process_addr = (addr_t) & per_info;
- parea.kernel_addr = offsetof (struct user_regs_struct, per_info);
- if (ptrace (PTRACE_PEEKUSR_AREA, tid, &parea) < 0)
- perror_with_name (_("Couldn't retrieve watchpoint status"));
- if (watch_base)
- {
- per_info.control_regs.bits.em_storage_alteration = 1;
- per_info.control_regs.bits.storage_alt_space_ctl = 1;
- }
- else
- {
- per_info.control_regs.bits.em_storage_alteration = 0;
- per_info.control_regs.bits.storage_alt_space_ctl = 0;
- }
- per_info.starting_addr = watch_lo_addr;
- per_info.ending_addr = watch_hi_addr;
- if (ptrace (PTRACE_POKEUSR_AREA, tid, &parea) < 0)
- perror_with_name (_("Couldn't modify watchpoint status"));
- }
- static int
- s390_insert_watchpoint (struct target_ops *self,
- CORE_ADDR addr, int len, int type,
- struct expression *cond)
- {
- struct lwp_info *lp;
- struct watch_area *area = xmalloc (sizeof (struct watch_area));
- if (!area)
- return -1;
- area->lo_addr = addr;
- area->hi_addr = addr + len - 1;
- area->next = watch_base;
- watch_base = area;
- ALL_LWPS (lp)
- s390_fix_watch_points (lp);
- return 0;
- }
- static int
- s390_remove_watchpoint (struct target_ops *self,
- CORE_ADDR addr, int len, int type,
- struct expression *cond)
- {
- struct lwp_info *lp;
- struct watch_area *area, **parea;
- for (parea = &watch_base; *parea; parea = &(*parea)->next)
- if ((*parea)->lo_addr == addr
- && (*parea)->hi_addr == addr + len - 1)
- break;
- if (!*parea)
- {
- fprintf_unfiltered (gdb_stderr,
- "Attempt to remove nonexistent watchpoint.\n");
- return -1;
- }
- area = *parea;
- *parea = area->next;
- xfree (area);
- ALL_LWPS (lp)
- s390_fix_watch_points (lp);
- return 0;
- }
- static int
- s390_can_use_hw_breakpoint (struct target_ops *self,
- int type, int cnt, int othertype)
- {
- return type == bp_hardware_watchpoint;
- }
- static int
- s390_region_ok_for_hw_watchpoint (struct target_ops *self,
- CORE_ADDR addr, int cnt)
- {
- return 1;
- }
- static int
- s390_target_wordsize (void)
- {
- int wordsize = 4;
-
- #ifdef __s390x__
- long pswm;
- errno = 0;
- pswm = (long) ptrace (PTRACE_PEEKUSER, s390_inferior_tid (), PT_PSWMASK, 0);
- if (errno == 0 && (pswm & 0x100000000ul) != 0)
- wordsize = 8;
- #endif
- return wordsize;
- }
- static int
- s390_auxv_parse (struct target_ops *ops, gdb_byte **readptr,
- gdb_byte *endptr, CORE_ADDR *typep, CORE_ADDR *valp)
- {
- int sizeof_auxv_field = s390_target_wordsize ();
- enum bfd_endian byte_order = gdbarch_byte_order (target_gdbarch ());
- gdb_byte *ptr = *readptr;
- if (endptr == ptr)
- return 0;
- if (endptr - ptr < sizeof_auxv_field * 2)
- return -1;
- *typep = extract_unsigned_integer (ptr, sizeof_auxv_field, byte_order);
- ptr += sizeof_auxv_field;
- *valp = extract_unsigned_integer (ptr, sizeof_auxv_field, byte_order);
- ptr += sizeof_auxv_field;
- *readptr = ptr;
- return 1;
- }
- #ifdef __s390x__
- static unsigned long
- s390_get_hwcap (void)
- {
- CORE_ADDR field;
- if (target_auxv_search (¤t_target, AT_HWCAP, &field))
- return (unsigned long) field;
- return 0;
- }
- #endif
- static const struct target_desc *
- s390_read_description (struct target_ops *ops)
- {
- int tid = s390_inferior_tid ();
- have_regset_last_break
- = check_regset (tid, NT_S390_LAST_BREAK, 8);
- have_regset_system_call
- = check_regset (tid, NT_S390_SYSTEM_CALL, 4);
- #ifdef __s390x__
-
- have_regset_tdb = (s390_get_hwcap () & HWCAP_S390_TE) ?
- check_regset (tid, NT_S390_TDB, s390_sizeof_tdbregset) : 0;
- if (s390_target_wordsize () == 8)
- return (have_regset_tdb ? tdesc_s390x_te_linux64 :
- have_regset_system_call? tdesc_s390x_linux64v2 :
- have_regset_last_break? tdesc_s390x_linux64v1 :
- tdesc_s390x_linux64);
- if (s390_get_hwcap () & HWCAP_S390_HIGH_GPRS)
- return (have_regset_tdb ? tdesc_s390_te_linux64 :
- have_regset_system_call? tdesc_s390_linux64v2 :
- have_regset_last_break? tdesc_s390_linux64v1 :
- tdesc_s390_linux64);
- #endif
-
- return (have_regset_system_call? tdesc_s390_linux32v2 :
- have_regset_last_break? tdesc_s390_linux32v1 :
- tdesc_s390_linux32);
- }
- void _initialize_s390_nat (void);
- void
- _initialize_s390_nat (void)
- {
- struct target_ops *t;
-
- t = linux_target ();
-
- t->to_fetch_registers = s390_linux_fetch_inferior_registers;
- t->to_store_registers = s390_linux_store_inferior_registers;
-
- t->to_can_use_hw_breakpoint = s390_can_use_hw_breakpoint;
- t->to_region_ok_for_hw_watchpoint = s390_region_ok_for_hw_watchpoint;
- t->to_have_continuable_watchpoint = 1;
- t->to_stopped_by_watchpoint = s390_stopped_by_watchpoint;
- t->to_insert_watchpoint = s390_insert_watchpoint;
- t->to_remove_watchpoint = s390_remove_watchpoint;
-
- t->to_read_description = s390_read_description;
- t->to_auxv_parse = s390_auxv_parse;
-
- linux_nat_add_target (t);
- linux_nat_set_new_thread (t, s390_fix_watch_points);
- }