9#if defined(__arm64__) || defined(__aarch64__)
38#define NT_ARM_SVE 0x405
47#define NT_ARM_ZA 0x40c
55#ifndef NT_ARM_PAC_MASK
56#define NT_ARM_PAC_MASK 0x406
59#ifndef NT_ARM_TAGGED_ADDR_CTRL
60#define NT_ARM_TAGGED_ADDR_CTRL 0x409
64#define NT_ARM_FPMR 0x40e
67#define HWCAP_PACA (1 << 30)
69#define HWCAP2_MTE (1 << 18)
71#define HWCAP2_FPMR (1UL << 48)
81static std::mutex g_register_flags_detector_mutex;
84std::unique_ptr<NativeRegisterContextLinux>
85NativeRegisterContextLinux::CreateHostNativeRegisterContextLinux(
88 case llvm::Triple::arm:
89 return std::make_unique<NativeRegisterContextLinux_arm>(target_arch,
91 case llvm::Triple::aarch64: {
96 ioVec.iov_base = &sve_header;
97 ioVec.iov_len =
sizeof(sve_header);
98 unsigned int regset = NT_ARM_SVE;
102 native_thread.
GetID(), ®set,
103 &ioVec,
sizeof(sve_header))
109 ioVec.iov_len =
sizeof(sve_header);
110 regset = NT_ARM_SSVE;
112 native_thread.
GetID(), ®set,
113 &ioVec,
sizeof(sve_header))
119 ioVec.iov_base = &za_header;
120 ioVec.iov_len =
sizeof(za_header);
123 native_thread.
GetID(), ®set,
124 &ioVec,
sizeof(za_header))
129 std::array<uint8_t, 64> zt_reg;
130 ioVec.iov_base = zt_reg.data();
131 ioVec.iov_len = zt_reg.size();
134 native_thread.
GetID(), ®set,
135 &ioVec, zt_reg.size())
141 std::optional<uint64_t> auxv_at_hwcap =
143 if (auxv_at_hwcap && (*auxv_at_hwcap & HWCAP_PACA))
146 std::optional<uint64_t> auxv_at_hwcap2 =
148 if (auxv_at_hwcap2) {
157 std::lock_guard<std::mutex> lock(g_register_flags_detector_mutex);
159 g_register_flags_detector.
DetectFields(auxv_at_hwcap.value_or(0),
160 auxv_at_hwcap2.value_or(0));
162 auto register_info_up =
163 std::make_unique<RegisterInfoPOSIX_arm64>(target_arch, opt_regsets);
164 return std::make_unique<NativeRegisterContextLinux_arm64>(
165 target_arch, native_thread, std::move(register_info_up));
168 llvm_unreachable(
"have no register context for architecture");
172llvm::Expected<ArchSpec>
173NativeRegisterContextLinux::DetermineArchitecture(
lldb::tid_t tid) {
174 return DetermineArchitectureViaGPR(
178NativeRegisterContextLinux_arm64::NativeRegisterContextLinux_arm64(
180 std::unique_ptr<RegisterInfoPOSIX_arm64> register_info_up)
182 register_info_up.release()),
185 GetRegisterInfoInterface().GetRegisterInfo(),
186 GetRegisterInfoInterface().GetRegisterCount());
188 ::memset(&m_fpr, 0,
sizeof(m_fpr));
189 ::memset(&m_gpr_arm64, 0,
sizeof(m_gpr_arm64));
190 ::memset(&m_hwp_regs, 0,
sizeof(m_hwp_regs));
191 ::memset(&m_hbp_regs, 0,
sizeof(m_hbp_regs));
192 ::memset(&m_sve_header, 0,
sizeof(m_sve_header));
193 ::memset(&m_pac_mask, 0,
sizeof(m_pac_mask));
194 ::memset(&m_tls_regs, 0,
sizeof(m_tls_regs));
195 ::memset(&m_sme_pseudo_regs, 0,
sizeof(m_sme_pseudo_regs));
196 std::fill(m_zt_reg.begin(), m_zt_reg.end(), 0);
202 m_max_hwp_supported = 16;
203 m_max_hbp_supported = 16;
205 m_refresh_hwdebug_info =
true;
207 m_gpr_is_valid =
false;
208 m_fpu_is_valid =
false;
209 m_sve_buffer_is_valid =
false;
210 m_sve_header_is_valid =
false;
211 m_pac_mask_is_valid =
false;
212 m_mte_ctrl_is_valid =
false;
213 m_tls_is_valid =
false;
214 m_zt_buffer_is_valid =
false;
215 m_fpmr_is_valid =
false;
218 m_tls_size = GetRegisterInfo().IsSSVEPresent() ?
sizeof(m_tls_regs)
219 :
sizeof(m_tls_regs.tpidr_reg);
221 if (GetRegisterInfo().IsSVEPresent() || GetRegisterInfo().IsSSVEPresent())
228NativeRegisterContextLinux_arm64::GetRegisterInfo()
const {
232uint32_t NativeRegisterContextLinux_arm64::GetRegisterSetCount()
const {
237NativeRegisterContextLinux_arm64::GetRegisterSet(uint32_t set_index)
const {
238 return GetRegisterInfo().GetRegisterSet(set_index);
241uint32_t NativeRegisterContextLinux_arm64::GetUserRegisterCount()
const {
243 for (uint32_t set_index = 0; set_index < GetRegisterSetCount(); ++set_index)
249NativeRegisterContextLinux_arm64::ReadRegister(
const RegisterInfo *reg_info,
254 error = Status::FromErrorString(
"reg_info NULL");
261 return Status::FromErrorStringWithFormat(
262 "no lldb regnum for %s",
263 reg_info && reg_info->
name ? reg_info->
name :
"<unknown register>");
268 std::vector<uint8_t> sve_reg_non_live;
276 assert(offset < GetGPRSize());
277 src = (uint8_t *)GetGPRBuffer() + offset;
279 }
else if (IsFPR(reg)) {
286 offset = CalculateFprOffset(reg_info);
287 assert(offset < GetFPRSize());
288 src = (uint8_t *)GetFPRBuffer() + offset;
293 error = ReadAllSVE();
302 if (reg == GetRegisterInfo().GetRegNumFPSR()) {
305 offset = sve::PTraceFPSROffset(sve::vq_from_vl(m_sve_header.vl));
307 offset = sve::ptrace_fpsimd_offset + (32 * 16);
308 }
else if (reg == GetRegisterInfo().GetRegNumFPCR()) {
311 offset = sve::PTraceFPCROffset(sve::vq_from_vl(m_sve_header.vl));
313 offset = sve::ptrace_fpsimd_offset + (32 * 16) + 4;
319 offset = CalculateSVEOffset(GetRegisterInfoAtIndex(sve_reg_num));
322 assert(offset < GetSVEBufferSize());
323 src = (uint8_t *)GetSVEBuffer() + offset;
325 }
else if (IsTLS(reg)) {
330 offset = reg_info->
byte_offset - GetRegisterInfo().GetTLSOffset();
331 assert(offset < GetTLSBufferSize());
332 src = (uint8_t *)GetTLSBuffer() + offset;
333 }
else if (IsSVE(reg)) {
335 return Status::FromErrorString(
"SVE disabled or not supported");
337 if (GetRegisterInfo().IsSVERegVG(reg)) {
338 sve_vg = GetSVERegVG();
339 src = (uint8_t *)&sve_vg;
342 error = ReadAllSVE();
350 sve_reg_non_live.resize(reg_info->
byte_size, 0);
351 src = sve_reg_non_live.data();
353 if (GetRegisterInfo().IsSVEZReg(reg)) {
354 offset = CalculateSVEOffset(reg_info);
355 assert(offset < GetSVEBufferSize());
356 ::memcpy(sve_reg_non_live.data(), (uint8_t *)GetSVEBuffer() + offset,
360 offset = CalculateSVEOffset(reg_info);
361 assert(offset < GetSVEBufferSize());
362 src = (uint8_t *)GetSVEBuffer() + offset;
365 }
else if (IsPAuth(reg)) {
366 error = ReadPAuthMask();
370 offset = reg_info->
byte_offset - GetRegisterInfo().GetPAuthOffset();
371 assert(offset < GetPACMaskSize());
372 src = (uint8_t *)GetPACMask() + offset;
373 }
else if (IsMTE(reg)) {
374 error = ReadMTEControl();
378 offset = reg_info->
byte_offset - GetRegisterInfo().GetMTEOffset();
379 assert(offset < GetMTEControlSize());
380 src = (uint8_t *)GetMTEControl() + offset;
381 }
else if (IsSME(reg)) {
382 if (GetRegisterInfo().IsSMERegZA(reg)) {
383 error = ReadZAHeader();
389 if (m_za_header.size ==
sizeof(m_za_header)) {
393 m_za_ptrace_payload.resize(((m_za_header.vl) * (m_za_header.vl)) +
395 std::fill(m_za_ptrace_payload.begin(), m_za_ptrace_payload.end(), 0);
406 src = (uint8_t *)GetZABuffer() + GetZAHeaderSize();
407 }
else if (GetRegisterInfo().IsSMERegZT(reg)) {
415 src = (uint8_t *)GetZTBuffer();
417 error = ReadSMESVG();
424 offset = reg_info->
byte_offset - GetRegisterInfo().GetSMEOffset();
425 assert(offset < GetSMEPseudoBufferSize());
426 src = (uint8_t *)GetSMEPseudoBuffer() + offset;
428 }
else if (IsFPMR(reg)) {
433 offset = reg_info->
byte_offset - GetRegisterInfo().GetFPMROffset();
434 assert(offset < GetFPMRBufferSize());
435 src = (uint8_t *)GetFPMRBuffer() + offset;
437 return Status::FromErrorString(
438 "failed - register wasn't recognized to be a GPR or an FPR, "
439 "write strategy unknown");
442 eByteOrderLittle,
error);
447Status NativeRegisterContextLinux_arm64::WriteRegister(
452 return Status::FromErrorString(
"reg_info NULL");
457 return Status::FromErrorStringWithFormat(
458 "no lldb regnum for %s",
459 reg_info && reg_info->
name ? reg_info->
name :
"<unknown register>");
463 std::vector<uint8_t> sve_reg_non_live;
471 dst = (uint8_t *)GetGPRBuffer() + reg_info->
byte_offset;
475 }
else if (IsFPR(reg)) {
482 offset = CalculateFprOffset(reg_info);
483 assert(offset < GetFPRSize());
484 dst = (uint8_t *)GetFPRBuffer() + offset;
490 error = ReadAllSVE();
499 if (reg == GetRegisterInfo().GetRegNumFPSR()) {
502 offset = sve::PTraceFPSROffset(sve::vq_from_vl(m_sve_header.vl));
504 offset = sve::ptrace_fpsimd_offset + (32 * 16);
505 }
else if (reg == GetRegisterInfo().GetRegNumFPCR()) {
508 offset = sve::PTraceFPCROffset(sve::vq_from_vl(m_sve_header.vl));
510 offset = sve::ptrace_fpsimd_offset + (32 * 16) + 4;
516 offset = CalculateSVEOffset(GetRegisterInfoAtIndex(sve_reg_num));
519 assert(offset < GetSVEBufferSize());
520 dst = (uint8_t *)GetSVEBuffer() + offset;
522 return WriteAllSVE();
524 }
else if (IsSVE(reg)) {
526 return Status::FromErrorString(
"SVE disabled or not supported");
529 error = ReadAllSVE();
533 if (GetRegisterInfo().IsSVERegVG(reg)) {
536 if (sve::vl_valid(vg_value * 8)) {
537 if (m_sve_header_is_valid && vg_value == GetSVERegVG())
540 SetSVERegVG(vg_value);
542 error = WriteSVEHeader();
543 if (
error.Success()) {
546 m_za_header_is_valid =
false;
547 ConfigureRegisterContext();
550 if (m_sve_header_is_valid && vg_value == GetSVERegVG())
554 return Status::FromErrorString(
"SVE vector length update failed.");
561 bool set_sve_state_full =
false;
562 const uint8_t *reg_bytes = (
const uint8_t *)reg_value.
GetBytes();
563 if (GetRegisterInfo().IsSVEZReg(reg)) {
564 for (uint32_t i = 16; i < reg_info->
byte_size; i++) {
566 set_sve_state_full =
true;
570 }
else if (GetRegisterInfo().IsSVEPReg(reg) ||
571 reg == GetRegisterInfo().GetRegNumSVEFFR()) {
572 for (uint32_t i = 0; i < reg_info->
byte_size; i++) {
574 set_sve_state_full =
true;
580 if (!set_sve_state_full && GetRegisterInfo().IsSVEZReg(reg)) {
583 offset = CalculateSVEOffset(reg_info);
584 assert(offset < GetSVEBufferSize());
585 dst = (uint8_t *)GetSVEBuffer() + offset;
586 ::memcpy(dst, reg_value.
GetBytes(), 16);
588 return WriteAllSVE();
590 return Status::FromErrorString(
591 "SVE state change operation not supported");
593 offset = CalculateSVEOffset(reg_info);
594 assert(offset < GetSVEBufferSize());
595 dst = (uint8_t *)GetSVEBuffer() + offset;
597 return WriteAllSVE();
600 }
else if (IsMTE(reg)) {
601 error = ReadMTEControl();
605 offset = reg_info->
byte_offset - GetRegisterInfo().GetMTEOffset();
606 assert(offset < GetMTEControlSize());
607 dst = (uint8_t *)GetMTEControl() + offset;
610 return WriteMTEControl();
611 }
else if (IsTLS(reg)) {
616 offset = reg_info->
byte_offset - GetRegisterInfo().GetTLSOffset();
617 assert(offset < GetTLSBufferSize());
618 dst = (uint8_t *)GetTLSBuffer() + offset;
622 }
else if (IsSME(reg)) {
623 if (GetRegisterInfo().IsSMERegZA(reg)) {
630 dst = (uint8_t *)GetZABuffer() + GetZAHeaderSize();
637 }
else if (GetRegisterInfo().IsSMERegZT(reg)) {
642 dst = (uint8_t *)GetZTBuffer();
647 return Status::FromErrorString(
648 "Writing to SVG or SVCR is not supported.");
649 }
else if (IsFPMR(reg)) {
654 offset = reg_info->
byte_offset - GetRegisterInfo().GetFPMROffset();
655 assert(offset < GetFPMRBufferSize());
656 dst = (uint8_t *)GetFPMRBuffer() + offset;
662 return Status::FromErrorString(
"Failed to write register value");
665enum RegisterSetType : uint32_t {
677static uint8_t *AddRegisterSetType(uint8_t *dst,
678 RegisterSetType register_set_type) {
679 *(
reinterpret_cast<uint32_t *
>(dst)) = register_set_type;
680 return dst +
sizeof(uint32_t);
683static uint8_t *AddSavedRegistersData(uint8_t *dst,
void *src,
size_t size) {
684 ::memcpy(dst, src, size);
688static uint8_t *AddSavedRegisters(uint8_t *dst,
689 enum RegisterSetType register_set_type,
690 void *src,
size_t size) {
691 dst = AddRegisterSetType(dst, register_set_type);
692 return AddSavedRegistersData(dst, src, size);
696NativeRegisterContextLinux_arm64::CacheAllRegisters(uint32_t &cached_size) {
698 cached_size =
sizeof(RegisterSetType) + GetGPRBufferSize();
703 if (GetRegisterInfo().IsZAPresent()) {
704 error = ReadZAHeader();
711 cached_size +=
sizeof(RegisterSetType) + m_za_header.size;
714 m_za_buffer_is_valid =
false;
723 GetRegisterInfo().IsZTPresent() &&
725 m_za_header.size >
sizeof(m_za_header)) {
726 cached_size +=
sizeof(RegisterSetType) + GetZTBufferSize();
736 if (GetRegisterInfo().IsSVEPresent() || GetRegisterInfo().IsSSVEPresent()) {
739 sizeof(RegisterSetType) +
sizeof(m_sve_state) + GetSVEBufferSize();
740 error = ReadAllSVE();
742 cached_size +=
sizeof(RegisterSetType) + GetFPRSize();
748 if (GetRegisterInfo().IsMTEPresent()) {
749 cached_size +=
sizeof(RegisterSetType) + GetMTEControlSize();
750 error = ReadMTEControl();
755 if (GetRegisterInfo().IsFPMRPresent()) {
756 cached_size +=
sizeof(RegisterSetType) + GetFPMRBufferSize();
763 cached_size +=
sizeof(RegisterSetType) + GetTLSBufferSize();
769Status NativeRegisterContextLinux_arm64::ReadAllRegisterValues(
781 uint32_t reg_data_byte_size = 0;
782 Status error = CacheAllRegisters(reg_data_byte_size);
787 uint8_t *dst = data_sp->GetBytes();
789 dst = AddSavedRegisters(dst, RegisterSetType::GPR, GetGPRBuffer(),
826 assert(m_za_header.size <= GetZABufferSize());
827 dst = AddSavedRegisters(dst, RegisterSetType::SME, GetZABuffer(),
831 if (GetRegisterInfo().IsSVEPresent() || GetRegisterInfo().IsSSVEPresent()) {
832 dst = AddRegisterSetType(dst, RegisterSetType::SVE);
833 *(
reinterpret_cast<SVEState *
>(dst)) = m_sve_state;
834 dst +=
sizeof(m_sve_state);
835 dst = AddSavedRegistersData(dst, GetSVEBuffer(), GetSVEBufferSize());
837 dst = AddSavedRegisters(dst, RegisterSetType::FPR, GetFPRBuffer(),
842 assert(m_za_header.size <= GetZABufferSize());
843 dst = AddSavedRegisters(dst, RegisterSetType::SME, GetZABuffer(),
854 GetRegisterInfo().IsZTPresent() &&
856 m_za_header.size >
sizeof(m_za_header))
857 dst = AddSavedRegisters(dst, RegisterSetType::SME2, GetZTBuffer(),
860 if (GetRegisterInfo().IsMTEPresent()) {
861 dst = AddSavedRegisters(dst, RegisterSetType::MTE, GetMTEControl(),
862 GetMTEControlSize());
865 if (GetRegisterInfo().IsFPMRPresent()) {
866 dst = AddSavedRegisters(dst, RegisterSetType::FPMR, GetFPMRBuffer(),
867 GetFPMRBufferSize());
870 dst = AddSavedRegisters(dst, RegisterSetType::TLS, GetTLSBuffer(),
876static Status RestoreRegisters(
void *buffer,
const uint8_t **src,
size_t len,
877 bool &is_valid, std::function<
Status()> writer) {
878 ::memcpy(buffer, *src, len);
884Status NativeRegisterContextLinux_arm64::WriteAllRegisterValues(
899 error = Status::FromErrorStringWithFormat(
900 "NativeRegisterContextLinux_arm64::%s invalid data_sp provided",
905 const uint8_t *src = data_sp->GetBytes();
906 if (src ==
nullptr) {
907 error = Status::FromErrorStringWithFormat(
908 "NativeRegisterContextLinux_arm64::%s "
909 "DataBuffer::GetBytes() returned a null "
915 uint64_t reg_data_min_size =
916 GetGPRBufferSize() + GetFPRSize() + 2 * (
sizeof(RegisterSetType));
917 if (data_sp->GetByteSize() < reg_data_min_size) {
918 error = Status::FromErrorStringWithFormat(
919 "NativeRegisterContextLinux_arm64::%s data_sp contained insufficient "
920 "register data bytes, expected at least %" PRIu64
", actual %" PRIu64,
921 __FUNCTION__, reg_data_min_size, data_sp->GetByteSize());
925 const uint8_t *end = src + data_sp->GetByteSize();
927 const RegisterSetType kind =
928 *
reinterpret_cast<const RegisterSetType *
>(src);
929 src +=
sizeof(RegisterSetType);
932 case RegisterSetType::GPR:
933 error = RestoreRegisters(
934 GetGPRBuffer(), &src, GetGPRBufferSize(), m_gpr_is_valid,
935 std::bind(&NativeRegisterContextLinux_arm64::WriteGPR,
this));
937 case RegisterSetType::SVE:
939 m_sve_state =
static_cast<SVEState>(*src);
940 src +=
sizeof(m_sve_state);
944 ::memcpy(GetSVEHeader(), src, GetSVEHeaderSize());
945 if (!sve::vl_valid(m_sve_header.vl)) {
946 m_sve_header_is_valid =
false;
947 error = Status::FromErrorStringWithFormat(
948 "NativeRegisterContextLinux_arm64::%s "
949 "Invalid SVE header in data_sp",
953 m_sve_header_is_valid =
true;
954 error = WriteSVEHeader();
961 ConfigureRegisterContext();
964 error = RestoreRegisters(
965 GetSVEBuffer(), &src, GetSVEBufferSize(), m_sve_buffer_is_valid,
966 std::bind(&NativeRegisterContextLinux_arm64::WriteAllSVE,
this));
968 case RegisterSetType::FPR:
969 error = RestoreRegisters(
970 GetFPRBuffer(), &src, GetFPRSize(), m_fpu_is_valid,
971 std::bind(&NativeRegisterContextLinux_arm64::WriteFPR,
this));
973 case RegisterSetType::MTE:
974 error = RestoreRegisters(
975 GetMTEControl(), &src, GetMTEControlSize(), m_mte_ctrl_is_valid,
976 std::bind(&NativeRegisterContextLinux_arm64::WriteMTEControl,
this));
978 case RegisterSetType::TLS:
979 error = RestoreRegisters(
980 GetTLSBuffer(), &src, GetTLSBufferSize(), m_tls_is_valid,
981 std::bind(&NativeRegisterContextLinux_arm64::WriteTLS,
this));
983 case RegisterSetType::SME:
989 ::memcpy(GetZAHeader(), src, GetZAHeaderSize());
993 m_za_ptrace_payload.resize(m_za_header.size);
994 ::memcpy(GetZABuffer(), src, GetZABufferSize());
995 m_za_buffer_is_valid =
true;
1003 ConfigureRegisterContext();
1008 src += GetZABufferSize();
1010 case RegisterSetType::SME2:
1014 error = RestoreRegisters(
1015 GetZTBuffer(), &src, GetZTBufferSize(), m_zt_buffer_is_valid,
1016 std::bind(&NativeRegisterContextLinux_arm64::WriteZT,
this));
1018 case RegisterSetType::FPMR:
1019 error = RestoreRegisters(
1020 GetFPMRBuffer(), &src, GetFPMRBufferSize(), m_fpmr_is_valid,
1021 std::bind(&NativeRegisterContextLinux_arm64::WriteFPMR,
this));
1032bool NativeRegisterContextLinux_arm64::IsGPR(
unsigned reg)
const {
1033 if (GetRegisterInfo().GetRegisterSetFromRegisterIndex(reg) ==
1039bool NativeRegisterContextLinux_arm64::IsFPR(
unsigned reg)
const {
1040 if (GetRegisterInfo().GetRegisterSetFromRegisterIndex(reg) ==
1046bool NativeRegisterContextLinux_arm64::IsSVE(
unsigned reg)
const {
1047 return GetRegisterInfo().IsSVEReg(reg);
1050bool NativeRegisterContextLinux_arm64::IsSME(
unsigned reg)
const {
1051 return GetRegisterInfo().IsSMEReg(reg);
1054bool NativeRegisterContextLinux_arm64::IsPAuth(
unsigned reg)
const {
1055 return GetRegisterInfo().IsPAuthReg(reg);
1058bool NativeRegisterContextLinux_arm64::IsMTE(
unsigned reg)
const {
1059 return GetRegisterInfo().IsMTEReg(reg);
1062bool NativeRegisterContextLinux_arm64::IsTLS(
unsigned reg)
const {
1063 return GetRegisterInfo().IsTLSReg(reg);
1066bool NativeRegisterContextLinux_arm64::IsFPMR(
unsigned reg)
const {
1067 return GetRegisterInfo().IsFPMRReg(reg);
1070llvm::Error NativeRegisterContextLinux_arm64::ReadHardwareDebugInfo() {
1071 if (!m_refresh_hwdebug_info) {
1072 return llvm::Error::success();
1075 ::pid_t tid = m_thread.GetID();
1077 int regset = NT_ARM_HW_WATCH;
1079 struct user_hwdebug_state dreg_state;
1082 ioVec.iov_base = &dreg_state;
1083 ioVec.iov_len =
sizeof(dreg_state);
1085 &ioVec, ioVec.iov_len);
1088 return error.ToError();
1090 m_max_hwp_supported = dreg_state.dbg_info & 0xff;
1092 regset = NT_ARM_HW_BREAK;
1094 &ioVec, ioVec.iov_len);
1097 return error.ToError();
1099 m_max_hbp_supported = dreg_state.dbg_info & 0xff;
1100 m_refresh_hwdebug_info =
false;
1102 return llvm::Error::success();
1106NativeRegisterContextLinux_arm64::WriteHardwareDebugRegs(DREGType hwbType) {
1108 struct user_hwdebug_state dreg_state;
1111 memset(&dreg_state, 0,
sizeof(dreg_state));
1112 ioVec.iov_base = &dreg_state;
1115 case eDREGTypeWATCH:
1116 regset = NT_ARM_HW_WATCH;
1117 ioVec.iov_len =
sizeof(dreg_state.dbg_info) +
sizeof(dreg_state.pad) +
1118 (
sizeof(dreg_state.dbg_regs[0]) * m_max_hwp_supported);
1120 for (uint32_t i = 0; i < m_max_hwp_supported; i++) {
1121 dreg_state.dbg_regs[i].addr = m_hwp_regs[i].address;
1122 dreg_state.dbg_regs[i].ctrl = m_hwp_regs[i].control;
1125 case eDREGTypeBREAK:
1126 regset = NT_ARM_HW_BREAK;
1127 ioVec.iov_len =
sizeof(dreg_state.dbg_info) +
sizeof(dreg_state.pad) +
1128 (
sizeof(dreg_state.dbg_regs[0]) * m_max_hbp_supported);
1130 for (uint32_t i = 0; i < m_max_hbp_supported; i++) {
1131 dreg_state.dbg_regs[i].addr = m_hbp_regs[i].address;
1132 dreg_state.dbg_regs[i].ctrl = m_hbp_regs[i].control;
1137 return NativeProcessLinux::PtraceWrapper(
PTRACE_SETREGSET, m_thread.GetID(),
1138 ®set, &ioVec, ioVec.iov_len)
1142Status NativeRegisterContextLinux_arm64::ReadGPR() {
1149 ioVec.iov_base = GetGPRBuffer();
1150 ioVec.iov_len = GetGPRBufferSize();
1152 error = ReadRegisterSet(&ioVec, GetGPRBufferSize(), NT_PRSTATUS);
1154 if (
error.Success())
1155 m_gpr_is_valid =
true;
1160Status NativeRegisterContextLinux_arm64::WriteGPR() {
1166 ioVec.iov_base = GetGPRBuffer();
1167 ioVec.iov_len = GetGPRBufferSize();
1169 m_gpr_is_valid =
false;
1171 return WriteRegisterSet(&ioVec, GetGPRBufferSize(), NT_PRSTATUS);
1174Status NativeRegisterContextLinux_arm64::ReadFPR() {
1181 ioVec.iov_base = GetFPRBuffer();
1182 ioVec.iov_len = GetFPRSize();
1184 error = ReadRegisterSet(&ioVec, GetFPRSize(), NT_FPREGSET);
1186 if (
error.Success())
1187 m_fpu_is_valid =
true;
1192Status NativeRegisterContextLinux_arm64::WriteFPR() {
1198 ioVec.iov_base = GetFPRBuffer();
1199 ioVec.iov_len = GetFPRSize();
1201 m_fpu_is_valid =
false;
1203 return WriteRegisterSet(&ioVec, GetFPRSize(), NT_FPREGSET);
1206void NativeRegisterContextLinux_arm64::InvalidateAllRegisters() {
1207 m_gpr_is_valid =
false;
1208 m_fpu_is_valid =
false;
1209 m_sve_buffer_is_valid =
false;
1210 m_sve_header_is_valid =
false;
1211 m_za_buffer_is_valid =
false;
1212 m_za_header_is_valid =
false;
1213 m_pac_mask_is_valid =
false;
1214 m_mte_ctrl_is_valid =
false;
1215 m_tls_is_valid =
false;
1216 m_zt_buffer_is_valid =
false;
1217 m_fpmr_is_valid =
false;
1220 ConfigureRegisterContext();
1223unsigned NativeRegisterContextLinux_arm64::GetSVERegSet() {
1227Status NativeRegisterContextLinux_arm64::ReadSVEHeader() {
1230 if (m_sve_header_is_valid)
1234 ioVec.iov_base = GetSVEHeader();
1235 ioVec.iov_len = GetSVEHeaderSize();
1237 error = ReadRegisterSet(&ioVec, GetSVEHeaderSize(), GetSVERegSet());
1239 if (
error.Success())
1240 m_sve_header_is_valid =
true;
1245Status NativeRegisterContextLinux_arm64::ReadPAuthMask() {
1248 if (m_pac_mask_is_valid)
1252 ioVec.iov_base = GetPACMask();
1253 ioVec.iov_len = GetPACMaskSize();
1255 error = ReadRegisterSet(&ioVec, GetPACMaskSize(), NT_ARM_PAC_MASK);
1257 if (
error.Success())
1258 m_pac_mask_is_valid =
true;
1263Status NativeRegisterContextLinux_arm64::WriteSVEHeader() {
1266 error = ReadSVEHeader();
1271 ioVec.iov_base = GetSVEHeader();
1272 ioVec.iov_len = GetSVEHeaderSize();
1274 m_sve_buffer_is_valid =
false;
1275 m_sve_header_is_valid =
false;
1276 m_fpu_is_valid =
false;
1278 return WriteRegisterSet(&ioVec, GetSVEHeaderSize(), GetSVERegSet());
1281Status NativeRegisterContextLinux_arm64::ReadAllSVE() {
1283 if (m_sve_buffer_is_valid)
1287 ioVec.iov_base = GetSVEBuffer();
1288 ioVec.iov_len = GetSVEBufferSize();
1290 error = ReadRegisterSet(&ioVec, GetSVEBufferSize(), GetSVERegSet());
1292 if (
error.Success())
1293 m_sve_buffer_is_valid =
true;
1298Status NativeRegisterContextLinux_arm64::WriteAllSVE() {
1301 error = ReadAllSVE();
1307 ioVec.iov_base = GetSVEBuffer();
1308 ioVec.iov_len = GetSVEBufferSize();
1310 m_sve_buffer_is_valid =
false;
1311 m_sve_header_is_valid =
false;
1312 m_fpu_is_valid =
false;
1314 return WriteRegisterSet(&ioVec, GetSVEBufferSize(), GetSVERegSet());
1317Status NativeRegisterContextLinux_arm64::ReadSMEControl() {
1329 if (
error.Success() && (m_za_header.size >
sizeof(m_za_header)))
1330 m_sme_pseudo_regs.ctrl_reg |= 2;
1335Status NativeRegisterContextLinux_arm64::ReadMTEControl() {
1338 if (m_mte_ctrl_is_valid)
1342 ioVec.iov_base = GetMTEControl();
1343 ioVec.iov_len = GetMTEControlSize();
1345 error = ReadRegisterSet(&ioVec, GetMTEControlSize(), NT_ARM_TAGGED_ADDR_CTRL);
1347 if (
error.Success())
1348 m_mte_ctrl_is_valid =
true;
1353Status NativeRegisterContextLinux_arm64::WriteMTEControl() {
1356 error = ReadMTEControl();
1361 ioVec.iov_base = GetMTEControl();
1362 ioVec.iov_len = GetMTEControlSize();
1364 m_mte_ctrl_is_valid =
false;
1366 return WriteRegisterSet(&ioVec, GetMTEControlSize(), NT_ARM_TAGGED_ADDR_CTRL);
1369Status NativeRegisterContextLinux_arm64::ReadTLS() {
1376 ioVec.iov_base = GetTLSBuffer();
1377 ioVec.iov_len = GetTLSBufferSize();
1379 error = ReadRegisterSet(&ioVec, GetTLSBufferSize(), NT_ARM_TLS);
1381 if (
error.Success())
1382 m_tls_is_valid =
true;
1387Status NativeRegisterContextLinux_arm64::WriteTLS() {
1395 ioVec.iov_base = GetTLSBuffer();
1396 ioVec.iov_len = GetTLSBufferSize();
1398 m_tls_is_valid =
false;
1400 return WriteRegisterSet(&ioVec, GetTLSBufferSize(), NT_ARM_TLS);
1403Status NativeRegisterContextLinux_arm64::ReadZAHeader() {
1406 if (m_za_header_is_valid)
1410 ioVec.iov_base = GetZAHeader();
1411 ioVec.iov_len = GetZAHeaderSize();
1413 error = ReadRegisterSet(&ioVec, GetZAHeaderSize(), NT_ARM_ZA);
1415 if (
error.Success())
1416 m_za_header_is_valid =
true;
1421Status NativeRegisterContextLinux_arm64::ReadZA() {
1424 if (m_za_buffer_is_valid)
1428 ioVec.iov_base = GetZABuffer();
1429 ioVec.iov_len = GetZABufferSize();
1431 error = ReadRegisterSet(&ioVec, GetZABufferSize(), NT_ARM_ZA);
1433 if (
error.Success())
1434 m_za_buffer_is_valid =
true;
1439Status NativeRegisterContextLinux_arm64::WriteZA() {
1450 ioVec.iov_base = GetZABuffer();
1451 ioVec.iov_len = GetZABufferSize();
1453 m_za_buffer_is_valid =
false;
1454 m_za_header_is_valid =
false;
1456 m_zt_buffer_is_valid =
false;
1458 return WriteRegisterSet(&ioVec, GetZABufferSize(), NT_ARM_ZA);
1461Status NativeRegisterContextLinux_arm64::ReadZT() {
1464 if (m_zt_buffer_is_valid)
1468 ioVec.iov_base = GetZTBuffer();
1469 ioVec.iov_len = GetZTBufferSize();
1471 error = ReadRegisterSet(&ioVec, GetZTBufferSize(), NT_ARM_ZT);
1472 m_zt_buffer_is_valid =
error.Success();
1477Status NativeRegisterContextLinux_arm64::WriteZT() {
1485 ioVec.iov_base = GetZTBuffer();
1486 ioVec.iov_len = GetZTBufferSize();
1488 m_zt_buffer_is_valid =
false;
1491 m_za_buffer_is_valid =
false;
1492 m_za_header_is_valid =
false;
1494 return WriteRegisterSet(&ioVec, GetZTBufferSize(), NT_ARM_ZT);
1497Status NativeRegisterContextLinux_arm64::ReadFPMR() {
1500 if (m_fpmr_is_valid)
1504 ioVec.iov_base = GetFPMRBuffer();
1505 ioVec.iov_len = GetFPMRBufferSize();
1507 error = ReadRegisterSet(&ioVec, GetFPMRBufferSize(), NT_ARM_FPMR);
1509 if (
error.Success())
1510 m_fpmr_is_valid =
true;
1515Status NativeRegisterContextLinux_arm64::WriteFPMR() {
1523 ioVec.iov_base = GetFPMRBuffer();
1524 ioVec.iov_len = GetFPMRBufferSize();
1526 m_fpmr_is_valid =
false;
1528 return WriteRegisterSet(&ioVec, GetFPMRBufferSize(), NT_ARM_FPMR);
1531void NativeRegisterContextLinux_arm64::ConfigureRegisterContext() {
1543 m_sve_header_is_valid =
false;
1544 m_sve_buffer_is_valid =
false;
1549 if (!(
error.Success() && ((m_sve_header.flags & sve::ptrace_regs_mask) ==
1550 sve::ptrace_regs_sve))) {
1552 m_sve_header_is_valid =
false;
1553 m_sve_buffer_is_valid =
false;
1555 error = ReadSVEHeader();
1556 if (
error.Success()) {
1559 if ((m_sve_header.flags & sve::ptrace_regs_mask) ==
1560 sve::ptrace_regs_fpsimd)
1573 if (sve::vl_valid(m_sve_header.vl))
1574 vq = sve::vq_from_vl(m_sve_header.vl);
1576 GetRegisterInfo().ConfigureVectorLengthSVE(vq);
1577 m_sve_ptrace_payload.resize(sve::PTraceSize(vq, sve::ptrace_regs_sve));
1581 if (!m_za_header_is_valid) {
1583 if (
error.Success()) {
1585 if (sve::vl_valid(m_za_header.vl))
1586 vq = sve::vq_from_vl(m_za_header.vl);
1588 GetRegisterInfo().ConfigureVectorLengthZA(vq);
1589 m_za_ptrace_payload.resize(m_za_header.size);
1590 m_za_buffer_is_valid =
false;
1595uint32_t NativeRegisterContextLinux_arm64::CalculateFprOffset(
1600uint32_t NativeRegisterContextLinux_arm64::CalculateSVEOffset(
1606 sve_reg_offset = sve::ptrace_fpsimd_offset +
1607 (reg - GetRegisterInfo().GetRegNumSVEZ0()) * 16;
1611 uint32_t sve_z0_offset = GetGPRSize() + 16;
1613 sve::SigRegsOffset() + reg_info->
byte_offset - sve_z0_offset;
1615 return sve_reg_offset;
1618Status NativeRegisterContextLinux_arm64::ReadSMESVG() {
1622 if (
error.Success())
1623 m_sme_pseudo_regs.svg_reg = m_za_header.vl / 8;
1628std::vector<uint32_t> NativeRegisterContextLinux_arm64::GetExpeditedRegisters(
1630 std::vector<uint32_t> expedited_reg_nums =
1631 NativeRegisterContext::GetExpeditedRegisters(expType);
1634 expedited_reg_nums.push_back(GetRegisterInfo().GetRegNumSVEVG());
1637 if (GetRegisterInfo().IsSSVEPresent())
1638 expedited_reg_nums.push_back(GetRegisterInfo().GetRegNumSMESVG());
1640 return expedited_reg_nums;
1643llvm::Expected<NativeRegisterContextLinux::MemoryTaggingDetails>
1644NativeRegisterContextLinux_arm64::GetMemoryTaggingDetails(int32_t type) {
1645 if (type == MemoryTagManagerAArch64MTE::eMTE_allocation) {
1646 return MemoryTaggingDetails{std::make_unique<MemoryTagManagerAArch64MTE>(),
1650 return llvm::createStringError(llvm::inconvertibleErrorCode(),
1651 "Unknown AArch64 memory tag type %d", type);
1654lldb::addr_t NativeRegisterContextLinux_arm64::FixWatchpointHitAddress(
1662 if (ReadPAuthMask().Success())
1663 mask |= m_pac_mask.data_mask;
1665 return hit_addr & ~mask;
static llvm::raw_ostream & error(Stream &strm)
#define PTRACE_PEEKMTETAGS
#define PTRACE_POKEMTETAGS
@ AUXV_AT_HWCAP2
Extension of AT_HWCAP.
@ AUXV_AT_HWCAP
Machine dependent hints about processor capabilities.
static size_t GetGPRSizeStatic()
size_t GetRegisterSetCount() const override
@ eVectorQuadwordAArch64SVE
An architecture specification class.
llvm::Triple::ArchType GetMachine() const
Returns a machine family for the current architecture.
This class manages the storage and detection of register field information.
void UpdateRegisterInfo(const RegisterInfo *reg_info, uint32_t num_regs)
Add the field information of any registers named in this class, to the relevant RegisterInfo instance...
void DetectFields(uint64_t hwcap, uint64_t hwcap2)
For the registers listed in this class, detect which fields are present.
bool HasDetected() const
Returns true if field detection has been run at least once.
A subclass of DataBuffer that stores a data buffer on the heap.
ValueType Set(ValueType mask)
Set one or more flags by logical OR'ing mask with the current flags.
std::optional< uint64_t > GetAuxValue(enum AuxVector::EntryType type)
lldb::tid_t GetID() const
uint32_t SetFromMemoryData(const RegisterInfo ®_info, const void *src, uint32_t src_len, lldb::ByteOrder src_byte_order, Status &error)
uint64_t GetAsUInt64(uint64_t fail_value=UINT64_MAX, bool *success_ptr=nullptr) const
const void * GetBytes() const
Manages communication with the inferior (debugee) process.
NativeProcessLinux & GetProcess()
#define LLDB_INVALID_INDEX32
#define LLDB_INVALID_REGNUM
A class that represents a running process on the host machine.
std::shared_ptr< lldb_private::DataBuffer > DataBufferSP
std::shared_ptr< lldb_private::WritableDataBuffer > WritableDataBufferSP
@ eRegisterKindLLDB
lldb's internal register numbers
Every register is described in detail including its name, alternate name (optional),...
uint32_t * value_regs
List of registers (terminated with LLDB_INVALID_REGNUM).
uint32_t byte_offset
The byte offset in the register context data where this register's value is found.
uint32_t byte_size
Size in bytes of the register.
uint32_t kinds[lldb::kNumRegisterKinds]
Holds all of the various register numbers for all register kinds.
const char * name
Name of this register, can't be NULL.
Registers are grouped into register sets.
size_t num_registers
The number of registers in REGISTERS array below.