forked from OSchip/llvm-project
[lldb] [Process/FreeBSDRemote] Access GPR via reginfo offsets
Read and write registers from m_gpr using offsets from RegisterInfo rather than explicit switch-case. This eliminates a lot of redundant code, and avoids mistakes such as type mismatches seen recently (wrt segment registers). The same logic will be extended to other register sets in the future. Make m_gpr an uint8_t std::array to ease accesses. Ideally, we could avoid including <machine/reg.h> entirely in the future and instead get the correct GPR size from Utility/RegisterContextFreeBSD_* somehow. While at it, modify register set logic to use an explicit enum with llvm::Optional<>, making the code cleaner and at the same time enabling compiler warnings for unhandled sets. Since now we're fully relying on 'struct GPR' defined in Utility/RegisterContextFreeBSD_* being entirely in sync with the system structure, add unit tests to verify the field offsets and sizes. Differential Revision: https://reviews.llvm.org/D91216
This commit is contained in:
parent
7625742237
commit
58abbf821c
|
@ -248,14 +248,16 @@ NativeRegisterContextFreeBSD_x86_64::NativeRegisterContextFreeBSD_x86_64(
|
|||
const ArchSpec &target_arch, NativeThreadProtocol &native_thread)
|
||||
: NativeRegisterContextRegisterInfo(
|
||||
native_thread, CreateRegisterInfoInterface(target_arch)),
|
||||
m_gpr(), m_fpr(), m_dbr() {}
|
||||
m_fpr(), m_dbr() {
|
||||
assert(m_gpr.size() == GetRegisterInfoInterface().GetGPRSize());
|
||||
}
|
||||
|
||||
// CONSIDER after local and llgs debugging are merged, register set support can
|
||||
// be moved into a base x86-64 class with IsRegisterSetAvailable made virtual.
|
||||
uint32_t NativeRegisterContextFreeBSD_x86_64::GetRegisterSetCount() const {
|
||||
uint32_t sets = 0;
|
||||
for (uint32_t set_index = 0; set_index < k_num_register_sets; ++set_index) {
|
||||
if (GetSetForNativeRegNum(set_index) != -1)
|
||||
if (GetSetForNativeRegNum(set_index))
|
||||
++sets;
|
||||
}
|
||||
|
||||
|
@ -276,38 +278,6 @@ NativeRegisterContextFreeBSD_x86_64::GetRegisterSet(uint32_t set_index) const {
|
|||
|
||||
static constexpr int RegNumX86ToX86_64(int regnum) {
|
||||
switch (regnum) {
|
||||
case lldb_eax_i386:
|
||||
return lldb_rax_x86_64;
|
||||
case lldb_ebx_i386:
|
||||
return lldb_rbx_x86_64;
|
||||
case lldb_ecx_i386:
|
||||
return lldb_rcx_x86_64;
|
||||
case lldb_edx_i386:
|
||||
return lldb_rdx_x86_64;
|
||||
case lldb_edi_i386:
|
||||
return lldb_rdi_x86_64;
|
||||
case lldb_esi_i386:
|
||||
return lldb_rsi_x86_64;
|
||||
case lldb_ebp_i386:
|
||||
return lldb_rbp_x86_64;
|
||||
case lldb_esp_i386:
|
||||
return lldb_rsp_x86_64;
|
||||
case lldb_eip_i386:
|
||||
return lldb_rip_x86_64;
|
||||
case lldb_eflags_i386:
|
||||
return lldb_rflags_x86_64;
|
||||
case lldb_cs_i386:
|
||||
return lldb_cs_x86_64;
|
||||
case lldb_fs_i386:
|
||||
return lldb_fs_x86_64;
|
||||
case lldb_gs_i386:
|
||||
return lldb_gs_x86_64;
|
||||
case lldb_ss_i386:
|
||||
return lldb_ss_x86_64;
|
||||
case lldb_ds_i386:
|
||||
return lldb_ds_x86_64;
|
||||
case lldb_es_i386:
|
||||
return lldb_es_x86_64;
|
||||
case lldb_fctrl_i386:
|
||||
return lldb_fctrl_x86_64;
|
||||
case lldb_fstat_i386:
|
||||
|
@ -387,9 +357,8 @@ static constexpr int RegNumX86ToX86_64(int regnum) {
|
|||
}
|
||||
}
|
||||
|
||||
int NativeRegisterContextFreeBSD_x86_64::GetSetForNativeRegNum(
|
||||
int reg_num) const {
|
||||
|
||||
llvm::Optional<enum NativeRegisterContextFreeBSD_x86_64::RegSetKind>
|
||||
NativeRegisterContextFreeBSD_x86_64::GetSetForNativeRegNum(int reg_num) const {
|
||||
switch (GetRegisterInfoInterface().GetTargetArchitecture().GetMachine()) {
|
||||
case llvm::Triple::x86:
|
||||
if (reg_num >= k_first_gpr_i386 && reg_num <= k_last_gpr_i386)
|
||||
|
@ -399,9 +368,9 @@ int NativeRegisterContextFreeBSD_x86_64::GetSetForNativeRegNum(
|
|||
if (reg_num >= k_first_avx_i386 && reg_num <= k_last_avx_i386)
|
||||
return XSaveRegSet; // AVX
|
||||
if (reg_num >= k_first_mpxr_i386 && reg_num <= k_last_mpxr_i386)
|
||||
return -1; // MPXR
|
||||
return llvm::None; // MPXR
|
||||
if (reg_num >= k_first_mpxc_i386 && reg_num <= k_last_mpxc_i386)
|
||||
return -1; // MPXC
|
||||
return llvm::None; // MPXC
|
||||
if (reg_num >= k_first_dbr_i386 && reg_num <= k_last_dbr_i386)
|
||||
return DBRegSet; // DBR
|
||||
break;
|
||||
|
@ -413,9 +382,9 @@ int NativeRegisterContextFreeBSD_x86_64::GetSetForNativeRegNum(
|
|||
if (reg_num >= k_first_avx_x86_64 && reg_num <= k_last_avx_x86_64)
|
||||
return XSaveRegSet; // AVX
|
||||
if (reg_num >= k_first_mpxr_x86_64 && reg_num <= k_last_mpxr_x86_64)
|
||||
return -1; // MPXR
|
||||
return llvm::None; // MPXR
|
||||
if (reg_num >= k_first_mpxc_x86_64 && reg_num <= k_last_mpxc_x86_64)
|
||||
return -1; // MPXC
|
||||
return llvm::None; // MPXC
|
||||
if (reg_num >= k_first_dbr_x86_64 && reg_num <= k_last_dbr_x86_64)
|
||||
return DBRegSet; // DBR
|
||||
break;
|
||||
|
@ -430,7 +399,7 @@ Status NativeRegisterContextFreeBSD_x86_64::ReadRegisterSet(uint32_t set) {
|
|||
switch (set) {
|
||||
case GPRegSet:
|
||||
return NativeProcessFreeBSD::PtraceWrapper(PT_GETREGS, m_thread.GetID(),
|
||||
&m_gpr);
|
||||
m_gpr.data());
|
||||
case FPRegSet:
|
||||
#if defined(__x86_64__)
|
||||
return NativeProcessFreeBSD::PtraceWrapper(PT_GETFPREGS, m_thread.GetID(),
|
||||
|
@ -471,7 +440,7 @@ Status NativeRegisterContextFreeBSD_x86_64::WriteRegisterSet(uint32_t set) {
|
|||
switch (set) {
|
||||
case GPRegSet:
|
||||
return NativeProcessFreeBSD::PtraceWrapper(PT_SETREGS, m_thread.GetID(),
|
||||
&m_gpr);
|
||||
m_gpr.data());
|
||||
case FPRegSet:
|
||||
#if defined(__x86_64__)
|
||||
return NativeProcessFreeBSD::PtraceWrapper(PT_SETFPREGS, m_thread.GetID(),
|
||||
|
@ -512,8 +481,8 @@ NativeRegisterContextFreeBSD_x86_64::ReadRegister(const RegisterInfo *reg_info,
|
|||
return error;
|
||||
}
|
||||
|
||||
int set = GetSetForNativeRegNum(reg);
|
||||
if (set == -1) {
|
||||
llvm::Optional<enum RegSetKind> opt_set = GetSetForNativeRegNum(reg);
|
||||
if (!opt_set) {
|
||||
// This is likely an internal register for lldb use only and should not be
|
||||
// directly queried.
|
||||
error.SetErrorStringWithFormat("register \"%s\" is in unrecognized set",
|
||||
|
@ -521,6 +490,23 @@ NativeRegisterContextFreeBSD_x86_64::ReadRegister(const RegisterInfo *reg_info,
|
|||
return error;
|
||||
}
|
||||
|
||||
enum RegSetKind set = opt_set.getValue();
|
||||
error = ReadRegisterSet(set);
|
||||
if (error.Fail())
|
||||
return error;
|
||||
|
||||
switch (set) {
|
||||
case GPRegSet:
|
||||
reg_value.SetBytes(m_gpr.data() + reg_info->byte_offset,
|
||||
reg_info->byte_size, endian::InlHostByteOrder());
|
||||
return error;
|
||||
case FPRegSet:
|
||||
case XSaveRegSet:
|
||||
case DBRegSet:
|
||||
// legacy logic
|
||||
break;
|
||||
}
|
||||
|
||||
switch (GetRegisterInfoInterface().GetTargetArchitecture().GetMachine()) {
|
||||
case llvm::Triple::x86_64:
|
||||
break;
|
||||
|
@ -531,134 +517,7 @@ NativeRegisterContextFreeBSD_x86_64::ReadRegister(const RegisterInfo *reg_info,
|
|||
llvm_unreachable("Unhandled target architecture.");
|
||||
}
|
||||
|
||||
error = ReadRegisterSet(set);
|
||||
if (error.Fail())
|
||||
return error;
|
||||
|
||||
switch (reg) {
|
||||
#if defined(__x86_64__)
|
||||
case lldb_rax_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rax;
|
||||
break;
|
||||
case lldb_rbx_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rbx;
|
||||
break;
|
||||
case lldb_rcx_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rcx;
|
||||
break;
|
||||
case lldb_rdx_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rdx;
|
||||
break;
|
||||
case lldb_rdi_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rdi;
|
||||
break;
|
||||
case lldb_rsi_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rsi;
|
||||
break;
|
||||
case lldb_rbp_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rbp;
|
||||
break;
|
||||
case lldb_rsp_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rsp;
|
||||
break;
|
||||
case lldb_r8_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r8;
|
||||
break;
|
||||
case lldb_r9_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r9;
|
||||
break;
|
||||
case lldb_r10_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r10;
|
||||
break;
|
||||
case lldb_r11_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r11;
|
||||
break;
|
||||
case lldb_r12_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r12;
|
||||
break;
|
||||
case lldb_r13_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r13;
|
||||
break;
|
||||
case lldb_r14_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r14;
|
||||
break;
|
||||
case lldb_r15_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_r15;
|
||||
break;
|
||||
case lldb_rip_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rip;
|
||||
break;
|
||||
case lldb_rflags_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_rflags;
|
||||
break;
|
||||
case lldb_cs_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_cs;
|
||||
break;
|
||||
case lldb_fs_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_fs;
|
||||
break;
|
||||
case lldb_gs_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_gs;
|
||||
break;
|
||||
case lldb_ss_x86_64:
|
||||
reg_value = (uint64_t)m_gpr.r_ss;
|
||||
break;
|
||||
case lldb_ds_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_ds;
|
||||
break;
|
||||
case lldb_es_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_es;
|
||||
break;
|
||||
#else
|
||||
case lldb_rax_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_eax;
|
||||
break;
|
||||
case lldb_rbx_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_ebx;
|
||||
break;
|
||||
case lldb_rcx_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_ecx;
|
||||
break;
|
||||
case lldb_rdx_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_edx;
|
||||
break;
|
||||
case lldb_rdi_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_edi;
|
||||
break;
|
||||
case lldb_rsi_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_esi;
|
||||
break;
|
||||
case lldb_rsp_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_esp;
|
||||
break;
|
||||
case lldb_rbp_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_ebp;
|
||||
break;
|
||||
case lldb_rip_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_eip;
|
||||
break;
|
||||
case lldb_rflags_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_eflags;
|
||||
break;
|
||||
case lldb_cs_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_cs;
|
||||
break;
|
||||
case lldb_fs_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_fs;
|
||||
break;
|
||||
case lldb_gs_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_gs;
|
||||
break;
|
||||
case lldb_ss_x86_64:
|
||||
reg_value = (uint32_t)m_gpr.r_ss;
|
||||
break;
|
||||
case lldb_ds_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_ds;
|
||||
break;
|
||||
case lldb_es_x86_64:
|
||||
reg_value = (uint16_t)m_gpr.r_es;
|
||||
break;
|
||||
#endif
|
||||
#if defined(__x86_64__)
|
||||
// the 32-bit field carries more detail, so we don't have to reinvent
|
||||
// the wheel
|
||||
|
@ -822,8 +681,8 @@ Status NativeRegisterContextFreeBSD_x86_64::WriteRegister(
|
|||
return error;
|
||||
}
|
||||
|
||||
int set = GetSetForNativeRegNum(reg);
|
||||
if (set == -1) {
|
||||
llvm::Optional<enum RegSetKind> opt_set = GetSetForNativeRegNum(reg);
|
||||
if (!opt_set) {
|
||||
// This is likely an internal register for lldb use only and should not be
|
||||
// directly queried.
|
||||
error.SetErrorStringWithFormat("register \"%s\" is in unrecognized set",
|
||||
|
@ -831,6 +690,23 @@ Status NativeRegisterContextFreeBSD_x86_64::WriteRegister(
|
|||
return error;
|
||||
}
|
||||
|
||||
enum RegSetKind set = opt_set.getValue();
|
||||
error = ReadRegisterSet(set);
|
||||
if (error.Fail())
|
||||
return error;
|
||||
|
||||
switch (set) {
|
||||
case GPRegSet:
|
||||
::memcpy(m_gpr.data() + reg_info->byte_offset, reg_value.GetBytes(),
|
||||
reg_value.GetByteSize());
|
||||
return WriteRegisterSet(set);
|
||||
case FPRegSet:
|
||||
case XSaveRegSet:
|
||||
case DBRegSet:
|
||||
// legacy logic
|
||||
break;
|
||||
}
|
||||
|
||||
switch (GetRegisterInfoInterface().GetTargetArchitecture().GetMachine()) {
|
||||
case llvm::Triple::x86_64:
|
||||
break;
|
||||
|
@ -841,134 +717,7 @@ Status NativeRegisterContextFreeBSD_x86_64::WriteRegister(
|
|||
llvm_unreachable("Unhandled target architecture.");
|
||||
}
|
||||
|
||||
error = ReadRegisterSet(set);
|
||||
if (error.Fail())
|
||||
return error;
|
||||
|
||||
switch (reg) {
|
||||
#if defined(__x86_64__)
|
||||
case lldb_rax_x86_64:
|
||||
m_gpr.r_rax = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rbx_x86_64:
|
||||
m_gpr.r_rbx = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rcx_x86_64:
|
||||
m_gpr.r_rcx = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rdx_x86_64:
|
||||
m_gpr.r_rdx = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rdi_x86_64:
|
||||
m_gpr.r_rdi = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rsi_x86_64:
|
||||
m_gpr.r_rsi = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rbp_x86_64:
|
||||
m_gpr.r_rbp = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rsp_x86_64:
|
||||
m_gpr.r_rsp = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r8_x86_64:
|
||||
m_gpr.r_r8 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r9_x86_64:
|
||||
m_gpr.r_r9 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r10_x86_64:
|
||||
m_gpr.r_r10 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r11_x86_64:
|
||||
m_gpr.r_r11 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r12_x86_64:
|
||||
m_gpr.r_r12 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r13_x86_64:
|
||||
m_gpr.r_r13 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r14_x86_64:
|
||||
m_gpr.r_r14 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_r15_x86_64:
|
||||
m_gpr.r_r15 = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rip_x86_64:
|
||||
m_gpr.r_rip = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_rflags_x86_64:
|
||||
m_gpr.r_rflags = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_cs_x86_64:
|
||||
m_gpr.r_cs = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_fs_x86_64:
|
||||
m_gpr.r_fs = reg_value.GetAsUInt16();
|
||||
break;
|
||||
case lldb_gs_x86_64:
|
||||
m_gpr.r_gs = reg_value.GetAsUInt16();
|
||||
break;
|
||||
case lldb_ss_x86_64:
|
||||
m_gpr.r_ss = reg_value.GetAsUInt64();
|
||||
break;
|
||||
case lldb_ds_x86_64:
|
||||
m_gpr.r_ds = reg_value.GetAsUInt16();
|
||||
break;
|
||||
case lldb_es_x86_64:
|
||||
m_gpr.r_es = reg_value.GetAsUInt16();
|
||||
break;
|
||||
#else
|
||||
case lldb_rax_x86_64:
|
||||
m_gpr.r_eax = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rbx_x86_64:
|
||||
m_gpr.r_ebx = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rcx_x86_64:
|
||||
m_gpr.r_ecx = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rdx_x86_64:
|
||||
m_gpr.r_edx = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rdi_x86_64:
|
||||
m_gpr.r_edi = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rsi_x86_64:
|
||||
m_gpr.r_esi = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rsp_x86_64:
|
||||
m_gpr.r_esp = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rbp_x86_64:
|
||||
m_gpr.r_ebp = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rip_x86_64:
|
||||
m_gpr.r_eip = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_rflags_x86_64:
|
||||
m_gpr.r_eflags = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_cs_x86_64:
|
||||
m_gpr.r_cs = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_fs_x86_64:
|
||||
m_gpr.r_fs = reg_value.GetAsUInt16();
|
||||
break;
|
||||
case lldb_gs_x86_64:
|
||||
m_gpr.r_gs = reg_value.GetAsUInt16();
|
||||
break;
|
||||
case lldb_ss_x86_64:
|
||||
m_gpr.r_ss = reg_value.GetAsUInt32();
|
||||
break;
|
||||
case lldb_ds_x86_64:
|
||||
m_gpr.r_ds = reg_value.GetAsUInt16();
|
||||
break;
|
||||
case lldb_es_x86_64:
|
||||
m_gpr.r_es = reg_value.GetAsUInt16();
|
||||
break;
|
||||
#endif
|
||||
case lldb_fctrl_x86_64:
|
||||
FPR_ENV(en_cw) = reg_value.GetAsUInt16();
|
||||
break;
|
||||
|
@ -1114,7 +863,7 @@ Status NativeRegisterContextFreeBSD_x86_64::ReadAllRegisterValues(
|
|||
return error;
|
||||
|
||||
uint8_t *dst = data_sp->GetBytes();
|
||||
::memcpy(dst, &m_gpr, GetRegisterInfoInterface().GetGPRSize());
|
||||
::memcpy(dst, m_gpr.data(), GetRegisterInfoInterface().GetGPRSize());
|
||||
dst += GetRegisterInfoInterface().GetGPRSize();
|
||||
|
||||
return error;
|
||||
|
@ -1147,7 +896,7 @@ Status NativeRegisterContextFreeBSD_x86_64::WriteAllRegisterValues(
|
|||
__FUNCTION__);
|
||||
return error;
|
||||
}
|
||||
::memcpy(&m_gpr, src, GetRegisterInfoInterface().GetGPRSize());
|
||||
::memcpy(m_gpr.data(), src, GetRegisterInfoInterface().GetGPRSize());
|
||||
|
||||
error = WriteRegisterSet(GPRegSet);
|
||||
if (error.Fail())
|
||||
|
|
|
@ -18,6 +18,8 @@
|
|||
#include <machine/reg.h>
|
||||
// clang-format on
|
||||
|
||||
#include <array>
|
||||
|
||||
#include "Plugins/Process/FreeBSDRemote/NativeRegisterContextFreeBSD.h"
|
||||
#include "Plugins/Process/Utility/RegisterContext_x86.h"
|
||||
#include "Plugins/Process/Utility/NativeRegisterContextWatchpoint_x86.h"
|
||||
|
@ -55,14 +57,19 @@ public:
|
|||
|
||||
private:
|
||||
// Private member types.
|
||||
enum { GPRegSet, FPRegSet, XSaveRegSet, DBRegSet };
|
||||
enum RegSetKind {
|
||||
GPRegSet,
|
||||
FPRegSet,
|
||||
XSaveRegSet,
|
||||
DBRegSet,
|
||||
};
|
||||
enum {
|
||||
YMMXSaveSet,
|
||||
MaxXSaveSet = YMMXSaveSet,
|
||||
};
|
||||
|
||||
// Private member variables.
|
||||
struct reg m_gpr;
|
||||
std::array<uint8_t, sizeof(struct reg)> m_gpr;
|
||||
#if defined(__x86_64__)
|
||||
struct fpreg m_fpr;
|
||||
#else
|
||||
|
@ -72,7 +79,7 @@ private:
|
|||
std::vector<uint8_t> m_xsave;
|
||||
std::array<uint32_t, MaxXSaveSet + 1> m_xsave_offsets;
|
||||
|
||||
int GetSetForNativeRegNum(int reg_num) const;
|
||||
llvm::Optional<enum RegSetKind> GetSetForNativeRegNum(int reg_num) const;
|
||||
int GetDR(int num) const;
|
||||
|
||||
Status ReadRegisterSet(uint32_t set);
|
||||
|
|
|
@ -3,6 +3,7 @@ if (CMAKE_SYSTEM_NAME MATCHES "Linux|Android")
|
|||
add_subdirectory(Linux)
|
||||
add_subdirectory(POSIX)
|
||||
endif()
|
||||
add_subdirectory(Utility)
|
||||
add_subdirectory(minidump)
|
||||
|
||||
add_lldb_unittest(ProcessEventDataTests
|
||||
|
|
|
@ -0,0 +1,5 @@
|
|||
add_lldb_unittest(ProcessUtilityTests
|
||||
RegisterContextFreeBSDTest.cpp
|
||||
|
||||
LINK_LIBS
|
||||
lldbPluginProcessUtility)
|
|
@ -0,0 +1,108 @@
|
|||
//===-- RegisterContextFreeBSDTests.cpp -----------------------------------===//
|
||||
//
|
||||
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
||||
// See https://llvm.org/LICENSE.txt for license information.
|
||||
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
||||
//
|
||||
//===----------------------------------------------------------------------===//
|
||||
|
||||
#if defined(__FreeBSD__)
|
||||
|
||||
// clang-format off
|
||||
#include <sys/types.h>
|
||||
#include <machine/reg.h>
|
||||
// clang-format on
|
||||
|
||||
#include "gmock/gmock.h"
|
||||
#include "gtest/gtest.h"
|
||||
|
||||
#include "Plugins/Process/Utility/lldb-x86-register-enums.h"
|
||||
#include "Plugins/Process/Utility/RegisterContextFreeBSD_i386.h"
|
||||
#include "Plugins/Process/Utility/RegisterContextFreeBSD_x86_64.h"
|
||||
|
||||
using namespace lldb;
|
||||
using namespace lldb_private;
|
||||
|
||||
std::pair<size_t, size_t> GetRegParams(RegisterInfoInterface &ctx,
|
||||
uint32_t reg) {
|
||||
const RegisterInfo &info = ctx.GetRegisterInfo()[reg];
|
||||
return {info.byte_offset, info.byte_size};
|
||||
}
|
||||
|
||||
#if defined(__x86_64__)
|
||||
|
||||
#define EXPECT_GPR_X86_64(regname) \
|
||||
EXPECT_THAT( \
|
||||
GetRegParams(reg_ctx, lldb_##regname##_x86_64), \
|
||||
::testing::Pair(offsetof(reg, r_##regname), sizeof(reg::r_##regname)))
|
||||
|
||||
TEST(RegisterContextFreeBSDTest, x86_64) {
|
||||
ArchSpec arch{"x86_64-unknown-freebsd"};
|
||||
RegisterContextFreeBSD_x86_64 reg_ctx{arch};
|
||||
|
||||
EXPECT_GPR_X86_64(r15);
|
||||
EXPECT_GPR_X86_64(r14);
|
||||
EXPECT_GPR_X86_64(r13);
|
||||
EXPECT_GPR_X86_64(r12);
|
||||
EXPECT_GPR_X86_64(r11);
|
||||
EXPECT_GPR_X86_64(r10);
|
||||
EXPECT_GPR_X86_64(r9);
|
||||
EXPECT_GPR_X86_64(r8);
|
||||
EXPECT_GPR_X86_64(rdi);
|
||||
EXPECT_GPR_X86_64(rsi);
|
||||
EXPECT_GPR_X86_64(rbp);
|
||||
EXPECT_GPR_X86_64(rbx);
|
||||
EXPECT_GPR_X86_64(rdx);
|
||||
EXPECT_GPR_X86_64(rcx);
|
||||
EXPECT_GPR_X86_64(rax);
|
||||
EXPECT_GPR_X86_64(fs);
|
||||
EXPECT_GPR_X86_64(gs);
|
||||
EXPECT_GPR_X86_64(es);
|
||||
EXPECT_GPR_X86_64(ds);
|
||||
EXPECT_GPR_X86_64(rip);
|
||||
EXPECT_GPR_X86_64(cs);
|
||||
EXPECT_GPR_X86_64(rflags);
|
||||
EXPECT_GPR_X86_64(rsp);
|
||||
EXPECT_GPR_X86_64(ss);
|
||||
}
|
||||
|
||||
#endif // defined(__x86_64__)
|
||||
|
||||
#if defined(__i386__) || defined(__x86_64__)
|
||||
|
||||
#define EXPECT_GPR_I386(regname) \
|
||||
EXPECT_THAT(GetRegParams(reg_ctx, lldb_##regname##_i386), \
|
||||
::testing::Pair(offsetof(native_i386_regs, r_##regname), \
|
||||
sizeof(native_i386_regs::r_##regname)))
|
||||
|
||||
TEST(RegisterContextFreeBSDTest, i386) {
|
||||
ArchSpec arch{"i686-unknown-freebsd"};
|
||||
RegisterContextFreeBSD_i386 reg_ctx{arch};
|
||||
|
||||
#if defined(__i386__)
|
||||
using native_i386_regs = ::reg;
|
||||
#else
|
||||
using native_i386_regs = ::reg32;
|
||||
#endif
|
||||
|
||||
EXPECT_GPR_I386(fs);
|
||||
EXPECT_GPR_I386(es);
|
||||
EXPECT_GPR_I386(ds);
|
||||
EXPECT_GPR_I386(edi);
|
||||
EXPECT_GPR_I386(esi);
|
||||
EXPECT_GPR_I386(ebp);
|
||||
EXPECT_GPR_I386(ebx);
|
||||
EXPECT_GPR_I386(edx);
|
||||
EXPECT_GPR_I386(ecx);
|
||||
EXPECT_GPR_I386(eax);
|
||||
EXPECT_GPR_I386(eip);
|
||||
EXPECT_GPR_I386(cs);
|
||||
EXPECT_GPR_I386(eflags);
|
||||
EXPECT_GPR_I386(esp);
|
||||
EXPECT_GPR_I386(ss);
|
||||
EXPECT_GPR_I386(gs);
|
||||
}
|
||||
|
||||
#endif // defined(__i386__) || defined(__x86_64__)
|
||||
|
||||
#endif // defined(__FreeBSD__)
|
Loading…
Reference in New Issue