diff options
author | John Baldwin <jhb@FreeBSD.org> | 2022-05-03 16:05:10 -0700 |
---|---|---|
committer | John Baldwin <jhb@FreeBSD.org> | 2022-05-03 16:05:10 -0700 |
commit | 684943d213b461a6a84ae67a9b8fcae5a28f007d (patch) | |
tree | d4fd2ea0e6e915f5d7495f5dfcad91958d31ed5c | |
parent | 2e686a74dc4782caeef75f76174909ab7ad358f8 (diff) | |
download | binutils-gdb-684943d213b461a6a84ae67a9b8fcae5a28f007d.tar.gz |
Fetch the NT_ARM_TLS register set for native FreeBSD/arm processes.
This permits resolving TLS variables.
-rw-r--r-- | gdb/arm-fbsd-nat.c | 52 |
1 files changed, 51 insertions, 1 deletions
diff --git a/gdb/arm-fbsd-nat.c b/gdb/arm-fbsd-nat.c index c32924de735..a306e1e2ee0 100644 --- a/gdb/arm-fbsd-nat.c +++ b/gdb/arm-fbsd-nat.c @@ -18,13 +18,17 @@ along with this program. If not, see <http://www.gnu.org/licenses/>. */ #include "defs.h" +#include "inferior.h" #include "target.h" +#include "elf/common.h" + #include <sys/types.h> #include <sys/ptrace.h> #include <machine/reg.h> #include "fbsd-nat.h" +#include "arm-tdep.h" #include "arm-fbsd-tdep.h" #include "inf-ptrace.h" @@ -49,6 +53,27 @@ arm_fbsd_nat_target::fetch_registers (struct regcache *regcache, int regnum) fetch_register_set<struct vfpreg> (regcache, regnum, PT_GETVFPREGS, &arm_fbsd_vfpregset); #endif +#ifdef PT_GETREGSET + gdbarch *gdbarch = regcache->arch (); + arm_gdbarch_tdep *tdep = (arm_gdbarch_tdep *) gdbarch_tdep (gdbarch); + + if (tdep->tls_regnum > 0) + { + const struct regcache_map_entry arm_fbsd_tlsregmap[] = + { + { 1, tdep->tls_regnum, 4 }, + { 0 } + }; + + const struct regset arm_fbsd_tlsregset = + { + arm_fbsd_tlsregmap, + regcache_supply_regset, regcache_collect_regset + }; + + fetch_regset<uint32_t> (regcache, regnum, NT_ARM_TLS, &arm_fbsd_tlsregset); + } +#endif } /* Store register REGNUM back into the inferior. If REGNUM is -1, do @@ -63,6 +88,27 @@ arm_fbsd_nat_target::store_registers (struct regcache *regcache, int regnum) store_register_set<struct vfpreg> (regcache, regnum, PT_GETVFPREGS, PT_SETVFPREGS, &arm_fbsd_vfpregset); #endif +#ifdef PT_GETREGSET + gdbarch *gdbarch = regcache->arch (); + arm_gdbarch_tdep *tdep = (arm_gdbarch_tdep *) gdbarch_tdep (gdbarch); + + if (tdep->tls_regnum > 0) + { + const struct regcache_map_entry arm_fbsd_tlsregmap[] = + { + { 1, tdep->tls_regnum, 4 }, + { 0 } + }; + + const struct regset arm_fbsd_tlsregset = + { + arm_fbsd_tlsregmap, + regcache_supply_regset, regcache_collect_regset + }; + + store_regset<uint32_t> (regcache, regnum, NT_ARM_TLS, &arm_fbsd_tlsregset); + } +#endif } /* Implement the to_read_description method. */ @@ -71,8 +117,12 @@ const struct target_desc * arm_fbsd_nat_target::read_description () { const struct target_desc *desc; + bool tls = false; - desc = arm_fbsd_read_description_auxv (this, false); +#ifdef PT_GETREGSET + tls = have_regset (inferior_ptid, NT_ARM_TLS) != 0; +#endif + desc = arm_fbsd_read_description_auxv (this, tls); if (desc == NULL) desc = this->beneath ()->read_description (); return desc; |