180814287SRaphael Isemann //===-- NativeRegisterContextWindows_x86_64.cpp ---------------------------===//
25146a9eaSAaron Smith //
35146a9eaSAaron Smith // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
45146a9eaSAaron Smith // See https://llvm.org/LICENSE.txt for license information.
55146a9eaSAaron Smith // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
65146a9eaSAaron Smith //
75146a9eaSAaron Smith //===----------------------------------------------------------------------===//
85146a9eaSAaron Smith
9900f9ba2STatyana Krasnukha #if defined(__x86_64__) || defined(_M_X64)
105146a9eaSAaron Smith
115146a9eaSAaron Smith #include "NativeRegisterContextWindows_x86_64.h"
125146a9eaSAaron Smith #include "NativeRegisterContextWindows_WoW64.h"
135146a9eaSAaron Smith #include "NativeThreadWindows.h"
145146a9eaSAaron Smith #include "Plugins/Process/Utility/RegisterContextWindows_i386.h"
155146a9eaSAaron Smith #include "Plugins/Process/Utility/RegisterContextWindows_x86_64.h"
165146a9eaSAaron Smith #include "ProcessWindowsLog.h"
175146a9eaSAaron Smith #include "lldb/Host/HostInfo.h"
185146a9eaSAaron Smith #include "lldb/Host/HostThread.h"
192e8b5755SSaleem Abdulrasool #include "lldb/Host/windows/HostThreadWindows.h"
205146a9eaSAaron Smith #include "lldb/Host/windows/windows.h"
215146a9eaSAaron Smith
225146a9eaSAaron Smith #include "lldb/Utility/Log.h"
235146a9eaSAaron Smith #include "lldb/Utility/RegisterValue.h"
245146a9eaSAaron Smith #include "llvm/ADT/STLExtras.h"
255146a9eaSAaron Smith
265146a9eaSAaron Smith using namespace lldb;
275146a9eaSAaron Smith using namespace lldb_private;
285146a9eaSAaron Smith
295146a9eaSAaron Smith #define REG_CONTEXT_SIZE sizeof(::CONTEXT)
305146a9eaSAaron Smith
315146a9eaSAaron Smith namespace {
325146a9eaSAaron Smith static const uint32_t g_gpr_regnums_x86_64[] = {
335146a9eaSAaron Smith lldb_rax_x86_64, lldb_rbx_x86_64, lldb_rcx_x86_64, lldb_rdx_x86_64,
345146a9eaSAaron Smith lldb_rdi_x86_64, lldb_rsi_x86_64, lldb_rbp_x86_64, lldb_rsp_x86_64,
355146a9eaSAaron Smith lldb_r8_x86_64, lldb_r9_x86_64, lldb_r10_x86_64, lldb_r11_x86_64,
365146a9eaSAaron Smith lldb_r12_x86_64, lldb_r13_x86_64, lldb_r14_x86_64, lldb_r15_x86_64,
375146a9eaSAaron Smith lldb_rip_x86_64, lldb_rflags_x86_64, lldb_cs_x86_64, lldb_fs_x86_64,
385146a9eaSAaron Smith lldb_gs_x86_64, lldb_ss_x86_64, lldb_ds_x86_64, lldb_es_x86_64,
395146a9eaSAaron Smith LLDB_INVALID_REGNUM // Register set must be terminated with this flag
405146a9eaSAaron Smith };
415146a9eaSAaron Smith
425146a9eaSAaron Smith static const uint32_t g_fpr_regnums_x86_64[] = {
435146a9eaSAaron Smith lldb_xmm0_x86_64, lldb_xmm1_x86_64, lldb_xmm2_x86_64, lldb_xmm3_x86_64,
445146a9eaSAaron Smith lldb_xmm4_x86_64, lldb_xmm5_x86_64, lldb_xmm6_x86_64, lldb_xmm7_x86_64,
455146a9eaSAaron Smith lldb_xmm8_x86_64, lldb_xmm9_x86_64, lldb_xmm10_x86_64, lldb_xmm11_x86_64,
465146a9eaSAaron Smith lldb_xmm12_x86_64, lldb_xmm13_x86_64, lldb_xmm14_x86_64, lldb_xmm15_x86_64,
475146a9eaSAaron Smith LLDB_INVALID_REGNUM // Register set must be terminated with this flag
485146a9eaSAaron Smith };
495146a9eaSAaron Smith
505146a9eaSAaron Smith static const RegisterSet g_reg_sets_x86_64[] = {
515146a9eaSAaron Smith {"General Purpose Registers", "gpr",
52f15014ffSBenjamin Kramer llvm::array_lengthof(g_gpr_regnums_x86_64) - 1, g_gpr_regnums_x86_64},
535146a9eaSAaron Smith {"Floating Point Registers", "fpr",
54f15014ffSBenjamin Kramer llvm::array_lengthof(g_fpr_regnums_x86_64) - 1, g_fpr_regnums_x86_64}};
555146a9eaSAaron Smith
565146a9eaSAaron Smith enum { k_num_register_sets = 2 };
575146a9eaSAaron Smith
585146a9eaSAaron Smith } // namespace
595146a9eaSAaron Smith
605146a9eaSAaron Smith static RegisterInfoInterface *
CreateRegisterInfoInterface(const ArchSpec & target_arch)615146a9eaSAaron Smith CreateRegisterInfoInterface(const ArchSpec &target_arch) {
625146a9eaSAaron Smith assert((HostInfo::GetArchitecture().GetAddressByteSize() == 8) &&
635146a9eaSAaron Smith "Register setting path assumes this is a 64-bit host");
645146a9eaSAaron Smith return new RegisterContextWindows_x86_64(target_arch);
655146a9eaSAaron Smith }
665146a9eaSAaron Smith
GetThreadContextHelper(lldb::thread_t thread_handle,PCONTEXT context_ptr,const DWORD control_flag)675146a9eaSAaron Smith static Status GetThreadContextHelper(lldb::thread_t thread_handle,
685146a9eaSAaron Smith PCONTEXT context_ptr,
695146a9eaSAaron Smith const DWORD control_flag) {
706730df47SPavel Labath Log *log = GetLog(WindowsLog::Registers);
715146a9eaSAaron Smith Status error;
725146a9eaSAaron Smith
735146a9eaSAaron Smith memset(context_ptr, 0, sizeof(::CONTEXT));
745146a9eaSAaron Smith context_ptr->ContextFlags = control_flag;
755146a9eaSAaron Smith if (!::GetThreadContext(thread_handle, context_ptr)) {
765146a9eaSAaron Smith error.SetError(GetLastError(), eErrorTypeWin32);
775146a9eaSAaron Smith LLDB_LOG(log, "{0} GetThreadContext failed with error {1}", __FUNCTION__,
785146a9eaSAaron Smith error);
795146a9eaSAaron Smith return error;
805146a9eaSAaron Smith }
815146a9eaSAaron Smith return Status();
825146a9eaSAaron Smith }
835146a9eaSAaron Smith
SetThreadContextHelper(lldb::thread_t thread_handle,PCONTEXT context_ptr)845146a9eaSAaron Smith static Status SetThreadContextHelper(lldb::thread_t thread_handle,
855146a9eaSAaron Smith PCONTEXT context_ptr) {
866730df47SPavel Labath Log *log = GetLog(WindowsLog::Registers);
875146a9eaSAaron Smith Status error;
885146a9eaSAaron Smith // It's assumed that the thread has stopped.
895146a9eaSAaron Smith if (!::SetThreadContext(thread_handle, context_ptr)) {
905146a9eaSAaron Smith error.SetError(GetLastError(), eErrorTypeWin32);
915146a9eaSAaron Smith LLDB_LOG(log, "{0} SetThreadContext failed with error {1}", __FUNCTION__,
925146a9eaSAaron Smith error);
935146a9eaSAaron Smith return error;
945146a9eaSAaron Smith }
955146a9eaSAaron Smith return Status();
965146a9eaSAaron Smith }
975146a9eaSAaron Smith
985146a9eaSAaron Smith std::unique_ptr<NativeRegisterContextWindows>
CreateHostNativeRegisterContextWindows(const ArchSpec & target_arch,NativeThreadProtocol & native_thread)995146a9eaSAaron Smith NativeRegisterContextWindows::CreateHostNativeRegisterContextWindows(
1005146a9eaSAaron Smith const ArchSpec &target_arch, NativeThreadProtocol &native_thread) {
1015146a9eaSAaron Smith // Register context for a WoW64 application.
1025146a9eaSAaron Smith if (target_arch.GetAddressByteSize() == 4)
103a8f3ae7cSJonas Devlieghere return std::make_unique<NativeRegisterContextWindows_WoW64>(target_arch,
1045146a9eaSAaron Smith native_thread);
1055146a9eaSAaron Smith
1065146a9eaSAaron Smith // Register context for a native 64-bit application.
107a8f3ae7cSJonas Devlieghere return std::make_unique<NativeRegisterContextWindows_x86_64>(target_arch,
1085146a9eaSAaron Smith native_thread);
1095146a9eaSAaron Smith }
1105146a9eaSAaron Smith
NativeRegisterContextWindows_x86_64(const ArchSpec & target_arch,NativeThreadProtocol & native_thread)1115146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::NativeRegisterContextWindows_x86_64(
1125146a9eaSAaron Smith const ArchSpec &target_arch, NativeThreadProtocol &native_thread)
1135146a9eaSAaron Smith : NativeRegisterContextWindows(native_thread,
1145146a9eaSAaron Smith CreateRegisterInfoInterface(target_arch)) {}
1155146a9eaSAaron Smith
IsGPR(uint32_t reg_index) const1165146a9eaSAaron Smith bool NativeRegisterContextWindows_x86_64::IsGPR(uint32_t reg_index) const {
1175146a9eaSAaron Smith return (reg_index >= k_first_gpr_x86_64 && reg_index < k_first_alias_x86_64);
1185146a9eaSAaron Smith }
1195146a9eaSAaron Smith
IsFPR(uint32_t reg_index) const1205146a9eaSAaron Smith bool NativeRegisterContextWindows_x86_64::IsFPR(uint32_t reg_index) const {
1215146a9eaSAaron Smith return (reg_index >= lldb_xmm0_x86_64 && reg_index <= k_last_fpr_x86_64);
1225146a9eaSAaron Smith }
1235146a9eaSAaron Smith
IsDR(uint32_t reg_index) const1245b5274eaSAleksandr Urakov bool NativeRegisterContextWindows_x86_64::IsDR(uint32_t reg_index) const {
1255b5274eaSAleksandr Urakov return (reg_index >= lldb_dr0_x86_64 && reg_index <= lldb_dr7_x86_64);
1265b5274eaSAleksandr Urakov }
1275b5274eaSAleksandr Urakov
GetRegisterSetCount() const1285146a9eaSAaron Smith uint32_t NativeRegisterContextWindows_x86_64::GetRegisterSetCount() const {
1295146a9eaSAaron Smith return k_num_register_sets;
1305146a9eaSAaron Smith }
1315146a9eaSAaron Smith
1325146a9eaSAaron Smith const RegisterSet *
GetRegisterSet(uint32_t set_index) const1335146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::GetRegisterSet(uint32_t set_index) const {
1345146a9eaSAaron Smith if (set_index >= k_num_register_sets)
1355146a9eaSAaron Smith return nullptr;
1365146a9eaSAaron Smith return &g_reg_sets_x86_64[set_index];
1375146a9eaSAaron Smith }
1385146a9eaSAaron Smith
GPRRead(const uint32_t reg,RegisterValue & reg_value)1395146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::GPRRead(const uint32_t reg,
1405146a9eaSAaron Smith RegisterValue ®_value) {
1415146a9eaSAaron Smith ::CONTEXT tls_context;
1425146a9eaSAaron Smith DWORD context_flag = CONTEXT_CONTROL | CONTEXT_INTEGER | CONTEXT_SEGMENTS;
1435146a9eaSAaron Smith Status error =
1445146a9eaSAaron Smith GetThreadContextHelper(GetThreadHandle(), &tls_context, context_flag);
1455146a9eaSAaron Smith if (error.Fail())
1465146a9eaSAaron Smith return error;
1475146a9eaSAaron Smith
1485146a9eaSAaron Smith switch (reg) {
1495146a9eaSAaron Smith case lldb_rax_x86_64:
1505146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rax);
1515146a9eaSAaron Smith break;
1525146a9eaSAaron Smith case lldb_rbx_x86_64:
1535146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rbx);
1545146a9eaSAaron Smith break;
1555146a9eaSAaron Smith case lldb_rcx_x86_64:
1565146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rcx);
1575146a9eaSAaron Smith break;
1585146a9eaSAaron Smith case lldb_rdx_x86_64:
1595146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rdx);
1605146a9eaSAaron Smith break;
1615146a9eaSAaron Smith case lldb_rdi_x86_64:
1625146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rdi);
1635146a9eaSAaron Smith break;
1645146a9eaSAaron Smith case lldb_rsi_x86_64:
1655146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rsi);
1665146a9eaSAaron Smith break;
1675146a9eaSAaron Smith case lldb_rbp_x86_64:
1685146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rbp);
1695146a9eaSAaron Smith break;
1705146a9eaSAaron Smith case lldb_rsp_x86_64:
1715146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rsp);
1725146a9eaSAaron Smith break;
1735146a9eaSAaron Smith case lldb_r8_x86_64:
1745146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R8);
1755146a9eaSAaron Smith break;
1765146a9eaSAaron Smith case lldb_r9_x86_64:
1775146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R9);
1785146a9eaSAaron Smith break;
1795146a9eaSAaron Smith case lldb_r10_x86_64:
1805146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R10);
1815146a9eaSAaron Smith break;
1825146a9eaSAaron Smith case lldb_r11_x86_64:
1835146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R11);
1845146a9eaSAaron Smith break;
1855146a9eaSAaron Smith case lldb_r12_x86_64:
1865146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R12);
1875146a9eaSAaron Smith break;
1885146a9eaSAaron Smith case lldb_r13_x86_64:
1895146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R13);
1905146a9eaSAaron Smith break;
1915146a9eaSAaron Smith case lldb_r14_x86_64:
1925146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R14);
1935146a9eaSAaron Smith break;
1945146a9eaSAaron Smith case lldb_r15_x86_64:
1955146a9eaSAaron Smith reg_value.SetUInt64(tls_context.R15);
1965146a9eaSAaron Smith break;
1975146a9eaSAaron Smith case lldb_rip_x86_64:
1985146a9eaSAaron Smith reg_value.SetUInt64(tls_context.Rip);
1995146a9eaSAaron Smith break;
2005146a9eaSAaron Smith case lldb_rflags_x86_64:
2015146a9eaSAaron Smith reg_value.SetUInt64(tls_context.EFlags | 0x2); // Bit #1 always 1
2025146a9eaSAaron Smith break;
2035146a9eaSAaron Smith case lldb_cs_x86_64:
2045146a9eaSAaron Smith reg_value.SetUInt16(tls_context.SegCs);
2055146a9eaSAaron Smith break;
2065146a9eaSAaron Smith case lldb_fs_x86_64:
2075146a9eaSAaron Smith reg_value.SetUInt16(tls_context.SegFs);
2085146a9eaSAaron Smith break;
2095146a9eaSAaron Smith case lldb_gs_x86_64:
2105146a9eaSAaron Smith reg_value.SetUInt16(tls_context.SegGs);
2115146a9eaSAaron Smith break;
2125146a9eaSAaron Smith case lldb_ss_x86_64:
2135146a9eaSAaron Smith reg_value.SetUInt16(tls_context.SegSs);
2145146a9eaSAaron Smith break;
2155146a9eaSAaron Smith case lldb_ds_x86_64:
2165146a9eaSAaron Smith reg_value.SetUInt16(tls_context.SegDs);
2175146a9eaSAaron Smith break;
2185146a9eaSAaron Smith case lldb_es_x86_64:
2195146a9eaSAaron Smith reg_value.SetUInt16(tls_context.SegEs);
2205146a9eaSAaron Smith break;
2215146a9eaSAaron Smith }
2225146a9eaSAaron Smith
2235146a9eaSAaron Smith return error;
2245146a9eaSAaron Smith }
2255146a9eaSAaron Smith
2265146a9eaSAaron Smith Status
GPRWrite(const uint32_t reg,const RegisterValue & reg_value)2275146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::GPRWrite(const uint32_t reg,
2285146a9eaSAaron Smith const RegisterValue ®_value) {
2295146a9eaSAaron Smith ::CONTEXT tls_context;
2305146a9eaSAaron Smith DWORD context_flag = CONTEXT_CONTROL | CONTEXT_INTEGER | CONTEXT_SEGMENTS;
2315146a9eaSAaron Smith auto thread_handle = GetThreadHandle();
2325146a9eaSAaron Smith Status error =
2335146a9eaSAaron Smith GetThreadContextHelper(thread_handle, &tls_context, context_flag);
2345146a9eaSAaron Smith if (error.Fail())
2355146a9eaSAaron Smith return error;
2365146a9eaSAaron Smith
2375146a9eaSAaron Smith switch (reg) {
2385146a9eaSAaron Smith case lldb_rax_x86_64:
2395146a9eaSAaron Smith tls_context.Rax = reg_value.GetAsUInt64();
2405146a9eaSAaron Smith break;
2415146a9eaSAaron Smith case lldb_rbx_x86_64:
2425146a9eaSAaron Smith tls_context.Rbx = reg_value.GetAsUInt64();
2435146a9eaSAaron Smith break;
2445146a9eaSAaron Smith case lldb_rcx_x86_64:
2455146a9eaSAaron Smith tls_context.Rcx = reg_value.GetAsUInt64();
2465146a9eaSAaron Smith break;
2475146a9eaSAaron Smith case lldb_rdx_x86_64:
2485146a9eaSAaron Smith tls_context.Rdx = reg_value.GetAsUInt64();
2495146a9eaSAaron Smith break;
2505146a9eaSAaron Smith case lldb_rdi_x86_64:
2515146a9eaSAaron Smith tls_context.Rdi = reg_value.GetAsUInt64();
2525146a9eaSAaron Smith break;
2535146a9eaSAaron Smith case lldb_rsi_x86_64:
2545146a9eaSAaron Smith tls_context.Rsi = reg_value.GetAsUInt64();
2555146a9eaSAaron Smith break;
2565146a9eaSAaron Smith case lldb_rbp_x86_64:
2575146a9eaSAaron Smith tls_context.Rbp = reg_value.GetAsUInt64();
2585146a9eaSAaron Smith break;
2595146a9eaSAaron Smith case lldb_rsp_x86_64:
2605146a9eaSAaron Smith tls_context.Rsp = reg_value.GetAsUInt64();
2615146a9eaSAaron Smith break;
2625146a9eaSAaron Smith case lldb_r8_x86_64:
2635146a9eaSAaron Smith tls_context.R8 = reg_value.GetAsUInt64();
2645146a9eaSAaron Smith break;
2655146a9eaSAaron Smith case lldb_r9_x86_64:
2665146a9eaSAaron Smith tls_context.R9 = reg_value.GetAsUInt64();
2675146a9eaSAaron Smith break;
2685146a9eaSAaron Smith case lldb_r10_x86_64:
2695146a9eaSAaron Smith tls_context.R10 = reg_value.GetAsUInt64();
2705146a9eaSAaron Smith break;
2715146a9eaSAaron Smith case lldb_r11_x86_64:
2725146a9eaSAaron Smith tls_context.R11 = reg_value.GetAsUInt64();
2735146a9eaSAaron Smith break;
2745146a9eaSAaron Smith case lldb_r12_x86_64:
2755146a9eaSAaron Smith tls_context.R12 = reg_value.GetAsUInt64();
2765146a9eaSAaron Smith break;
2775146a9eaSAaron Smith case lldb_r13_x86_64:
2785146a9eaSAaron Smith tls_context.R13 = reg_value.GetAsUInt64();
2795146a9eaSAaron Smith break;
2805146a9eaSAaron Smith case lldb_r14_x86_64:
2815146a9eaSAaron Smith tls_context.R14 = reg_value.GetAsUInt64();
2825146a9eaSAaron Smith break;
2835146a9eaSAaron Smith case lldb_r15_x86_64:
2845146a9eaSAaron Smith tls_context.R15 = reg_value.GetAsUInt64();
2855146a9eaSAaron Smith break;
2865146a9eaSAaron Smith case lldb_rip_x86_64:
2875146a9eaSAaron Smith tls_context.Rip = reg_value.GetAsUInt64();
2885146a9eaSAaron Smith break;
2895146a9eaSAaron Smith case lldb_rflags_x86_64:
2905146a9eaSAaron Smith tls_context.EFlags = reg_value.GetAsUInt64();
2915146a9eaSAaron Smith break;
2925146a9eaSAaron Smith case lldb_cs_x86_64:
2935146a9eaSAaron Smith tls_context.SegCs = reg_value.GetAsUInt16();
2945146a9eaSAaron Smith break;
2955146a9eaSAaron Smith case lldb_fs_x86_64:
2965146a9eaSAaron Smith tls_context.SegFs = reg_value.GetAsUInt16();
2975146a9eaSAaron Smith break;
2985146a9eaSAaron Smith case lldb_gs_x86_64:
2995146a9eaSAaron Smith tls_context.SegGs = reg_value.GetAsUInt16();
3005146a9eaSAaron Smith break;
3015146a9eaSAaron Smith case lldb_ss_x86_64:
3025146a9eaSAaron Smith tls_context.SegSs = reg_value.GetAsUInt16();
3035146a9eaSAaron Smith break;
3045146a9eaSAaron Smith case lldb_ds_x86_64:
3055146a9eaSAaron Smith tls_context.SegDs = reg_value.GetAsUInt16();
3065146a9eaSAaron Smith break;
3075146a9eaSAaron Smith case lldb_es_x86_64:
3085146a9eaSAaron Smith tls_context.SegEs = reg_value.GetAsUInt16();
3095146a9eaSAaron Smith break;
3105146a9eaSAaron Smith }
3115146a9eaSAaron Smith
3125146a9eaSAaron Smith return SetThreadContextHelper(thread_handle, &tls_context);
3135146a9eaSAaron Smith }
3145146a9eaSAaron Smith
FPRRead(const uint32_t reg,RegisterValue & reg_value)3155146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::FPRRead(const uint32_t reg,
3165146a9eaSAaron Smith RegisterValue ®_value) {
3175146a9eaSAaron Smith ::CONTEXT tls_context;
3185146a9eaSAaron Smith DWORD context_flag = CONTEXT_CONTROL | CONTEXT_FLOATING_POINT;
3195146a9eaSAaron Smith Status error =
3205146a9eaSAaron Smith GetThreadContextHelper(GetThreadHandle(), &tls_context, context_flag);
3215146a9eaSAaron Smith if (error.Fail())
3225146a9eaSAaron Smith return error;
3235146a9eaSAaron Smith
3245146a9eaSAaron Smith switch (reg) {
3255146a9eaSAaron Smith case lldb_xmm0_x86_64:
3265146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm0, 16, endian::InlHostByteOrder());
3275146a9eaSAaron Smith break;
3285146a9eaSAaron Smith case lldb_xmm1_x86_64:
3295146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm1, 16, endian::InlHostByteOrder());
3305146a9eaSAaron Smith break;
3315146a9eaSAaron Smith case lldb_xmm2_x86_64:
3325146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm2, 16, endian::InlHostByteOrder());
3335146a9eaSAaron Smith break;
3345146a9eaSAaron Smith case lldb_xmm3_x86_64:
3355146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm3, 16, endian::InlHostByteOrder());
3365146a9eaSAaron Smith break;
3375146a9eaSAaron Smith case lldb_xmm4_x86_64:
3385146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm4, 16, endian::InlHostByteOrder());
3395146a9eaSAaron Smith break;
3405146a9eaSAaron Smith case lldb_xmm5_x86_64:
3415146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm5, 16, endian::InlHostByteOrder());
3425146a9eaSAaron Smith break;
3435146a9eaSAaron Smith case lldb_xmm6_x86_64:
3445146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm6, 16, endian::InlHostByteOrder());
3455146a9eaSAaron Smith break;
3465146a9eaSAaron Smith case lldb_xmm7_x86_64:
3475146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm7, 16, endian::InlHostByteOrder());
3485146a9eaSAaron Smith break;
3495146a9eaSAaron Smith case lldb_xmm8_x86_64:
3505146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm8, 16, endian::InlHostByteOrder());
3515146a9eaSAaron Smith break;
3525146a9eaSAaron Smith case lldb_xmm9_x86_64:
3535146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm9, 16, endian::InlHostByteOrder());
3545146a9eaSAaron Smith break;
3555146a9eaSAaron Smith case lldb_xmm10_x86_64:
3565146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm10, 16, endian::InlHostByteOrder());
3575146a9eaSAaron Smith break;
3585146a9eaSAaron Smith case lldb_xmm11_x86_64:
3595146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm11, 16, endian::InlHostByteOrder());
3605146a9eaSAaron Smith break;
3615146a9eaSAaron Smith case lldb_xmm12_x86_64:
3625146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm12, 16, endian::InlHostByteOrder());
3635146a9eaSAaron Smith break;
3645146a9eaSAaron Smith case lldb_xmm13_x86_64:
3655146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm13, 16, endian::InlHostByteOrder());
3665146a9eaSAaron Smith break;
3675146a9eaSAaron Smith case lldb_xmm14_x86_64:
3685146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm14, 16, endian::InlHostByteOrder());
3695146a9eaSAaron Smith break;
3705146a9eaSAaron Smith case lldb_xmm15_x86_64:
3715146a9eaSAaron Smith reg_value.SetBytes(&tls_context.Xmm15, 16, endian::InlHostByteOrder());
3725146a9eaSAaron Smith break;
3735146a9eaSAaron Smith }
3745146a9eaSAaron Smith
3755146a9eaSAaron Smith return error;
3765146a9eaSAaron Smith }
3775146a9eaSAaron Smith
3785146a9eaSAaron Smith Status
FPRWrite(const uint32_t reg,const RegisterValue & reg_value)3795146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::FPRWrite(const uint32_t reg,
3805146a9eaSAaron Smith const RegisterValue ®_value) {
3815146a9eaSAaron Smith ::CONTEXT tls_context;
3825146a9eaSAaron Smith DWORD context_flag = CONTEXT_CONTROL | CONTEXT_FLOATING_POINT;
3835146a9eaSAaron Smith auto thread_handle = GetThreadHandle();
3845146a9eaSAaron Smith Status error =
3855146a9eaSAaron Smith GetThreadContextHelper(thread_handle, &tls_context, context_flag);
3865146a9eaSAaron Smith if (error.Fail())
3875146a9eaSAaron Smith return error;
3885146a9eaSAaron Smith
3895146a9eaSAaron Smith switch (reg) {
3905146a9eaSAaron Smith case lldb_xmm0_x86_64:
3915146a9eaSAaron Smith memcpy(&tls_context.Xmm0, reg_value.GetBytes(), 16);
3925146a9eaSAaron Smith break;
3935146a9eaSAaron Smith case lldb_xmm1_x86_64:
3945146a9eaSAaron Smith memcpy(&tls_context.Xmm1, reg_value.GetBytes(), 16);
3955146a9eaSAaron Smith break;
3965146a9eaSAaron Smith case lldb_xmm2_x86_64:
3975146a9eaSAaron Smith memcpy(&tls_context.Xmm2, reg_value.GetBytes(), 16);
3985146a9eaSAaron Smith break;
3995146a9eaSAaron Smith case lldb_xmm3_x86_64:
4005146a9eaSAaron Smith memcpy(&tls_context.Xmm3, reg_value.GetBytes(), 16);
4015146a9eaSAaron Smith break;
4025146a9eaSAaron Smith case lldb_xmm4_x86_64:
4035146a9eaSAaron Smith memcpy(&tls_context.Xmm4, reg_value.GetBytes(), 16);
4045146a9eaSAaron Smith break;
4055146a9eaSAaron Smith case lldb_xmm5_x86_64:
4065146a9eaSAaron Smith memcpy(&tls_context.Xmm5, reg_value.GetBytes(), 16);
4075146a9eaSAaron Smith break;
4085146a9eaSAaron Smith case lldb_xmm6_x86_64:
4095146a9eaSAaron Smith memcpy(&tls_context.Xmm6, reg_value.GetBytes(), 16);
4105146a9eaSAaron Smith break;
4115146a9eaSAaron Smith case lldb_xmm7_x86_64:
4125146a9eaSAaron Smith memcpy(&tls_context.Xmm7, reg_value.GetBytes(), 16);
4135146a9eaSAaron Smith break;
4145146a9eaSAaron Smith case lldb_xmm8_x86_64:
4155146a9eaSAaron Smith memcpy(&tls_context.Xmm8, reg_value.GetBytes(), 16);
4165146a9eaSAaron Smith break;
4175146a9eaSAaron Smith case lldb_xmm9_x86_64:
4185146a9eaSAaron Smith memcpy(&tls_context.Xmm9, reg_value.GetBytes(), 16);
4195146a9eaSAaron Smith break;
4205146a9eaSAaron Smith case lldb_xmm10_x86_64:
4215146a9eaSAaron Smith memcpy(&tls_context.Xmm10, reg_value.GetBytes(), 16);
4225146a9eaSAaron Smith break;
4235146a9eaSAaron Smith case lldb_xmm11_x86_64:
4245146a9eaSAaron Smith memcpy(&tls_context.Xmm11, reg_value.GetBytes(), 16);
4255146a9eaSAaron Smith break;
4265146a9eaSAaron Smith case lldb_xmm12_x86_64:
4275146a9eaSAaron Smith memcpy(&tls_context.Xmm12, reg_value.GetBytes(), 16);
4285146a9eaSAaron Smith break;
4295146a9eaSAaron Smith case lldb_xmm13_x86_64:
4305146a9eaSAaron Smith memcpy(&tls_context.Xmm13, reg_value.GetBytes(), 16);
4315146a9eaSAaron Smith break;
4325146a9eaSAaron Smith case lldb_xmm14_x86_64:
4335146a9eaSAaron Smith memcpy(&tls_context.Xmm14, reg_value.GetBytes(), 16);
4345146a9eaSAaron Smith break;
4355146a9eaSAaron Smith case lldb_xmm15_x86_64:
4365146a9eaSAaron Smith memcpy(&tls_context.Xmm15, reg_value.GetBytes(), 16);
4375146a9eaSAaron Smith break;
4385146a9eaSAaron Smith }
4395146a9eaSAaron Smith
4405146a9eaSAaron Smith return SetThreadContextHelper(thread_handle, &tls_context);
4415146a9eaSAaron Smith }
4425146a9eaSAaron Smith
DRRead(const uint32_t reg,RegisterValue & reg_value)4435b5274eaSAleksandr Urakov Status NativeRegisterContextWindows_x86_64::DRRead(const uint32_t reg,
4445b5274eaSAleksandr Urakov RegisterValue ®_value) {
4455b5274eaSAleksandr Urakov ::CONTEXT tls_context;
4465b5274eaSAleksandr Urakov DWORD context_flag = CONTEXT_DEBUG_REGISTERS;
4475b5274eaSAleksandr Urakov Status error =
4485b5274eaSAleksandr Urakov GetThreadContextHelper(GetThreadHandle(), &tls_context, context_flag);
4495b5274eaSAleksandr Urakov if (error.Fail())
4505b5274eaSAleksandr Urakov return error;
4515b5274eaSAleksandr Urakov
4525b5274eaSAleksandr Urakov switch (reg) {
4535b5274eaSAleksandr Urakov case lldb_dr0_x86_64:
4545b5274eaSAleksandr Urakov reg_value.SetUInt64(tls_context.Dr0);
4555b5274eaSAleksandr Urakov break;
4565b5274eaSAleksandr Urakov case lldb_dr1_x86_64:
4575b5274eaSAleksandr Urakov reg_value.SetUInt64(tls_context.Dr1);
4585b5274eaSAleksandr Urakov break;
4595b5274eaSAleksandr Urakov case lldb_dr2_x86_64:
4605b5274eaSAleksandr Urakov reg_value.SetUInt64(tls_context.Dr2);
4615b5274eaSAleksandr Urakov break;
4625b5274eaSAleksandr Urakov case lldb_dr3_x86_64:
4635b5274eaSAleksandr Urakov reg_value.SetUInt64(tls_context.Dr3);
4645b5274eaSAleksandr Urakov break;
4655b5274eaSAleksandr Urakov case lldb_dr4_x86_64:
4665b5274eaSAleksandr Urakov return Status("register DR4 is obsolete");
4675b5274eaSAleksandr Urakov case lldb_dr5_x86_64:
4685b5274eaSAleksandr Urakov return Status("register DR5 is obsolete");
4695b5274eaSAleksandr Urakov case lldb_dr6_x86_64:
4705b5274eaSAleksandr Urakov reg_value.SetUInt64(tls_context.Dr6);
4715b5274eaSAleksandr Urakov break;
4725b5274eaSAleksandr Urakov case lldb_dr7_x86_64:
4735b5274eaSAleksandr Urakov reg_value.SetUInt64(tls_context.Dr7);
4745b5274eaSAleksandr Urakov break;
4755b5274eaSAleksandr Urakov }
4765b5274eaSAleksandr Urakov
4775b5274eaSAleksandr Urakov return {};
4785b5274eaSAleksandr Urakov }
4795b5274eaSAleksandr Urakov
4805b5274eaSAleksandr Urakov Status
DRWrite(const uint32_t reg,const RegisterValue & reg_value)4815b5274eaSAleksandr Urakov NativeRegisterContextWindows_x86_64::DRWrite(const uint32_t reg,
4825b5274eaSAleksandr Urakov const RegisterValue ®_value) {
4835b5274eaSAleksandr Urakov ::CONTEXT tls_context;
4845b5274eaSAleksandr Urakov DWORD context_flag = CONTEXT_DEBUG_REGISTERS;
4855b5274eaSAleksandr Urakov auto thread_handle = GetThreadHandle();
4865b5274eaSAleksandr Urakov Status error =
4875b5274eaSAleksandr Urakov GetThreadContextHelper(thread_handle, &tls_context, context_flag);
4885b5274eaSAleksandr Urakov if (error.Fail())
4895b5274eaSAleksandr Urakov return error;
4905b5274eaSAleksandr Urakov
4915b5274eaSAleksandr Urakov switch (reg) {
4925b5274eaSAleksandr Urakov case lldb_dr0_x86_64:
4935b5274eaSAleksandr Urakov tls_context.Dr0 = reg_value.GetAsUInt64();
4945b5274eaSAleksandr Urakov break;
4955b5274eaSAleksandr Urakov case lldb_dr1_x86_64:
4965b5274eaSAleksandr Urakov tls_context.Dr1 = reg_value.GetAsUInt64();
4975b5274eaSAleksandr Urakov break;
4985b5274eaSAleksandr Urakov case lldb_dr2_x86_64:
4995b5274eaSAleksandr Urakov tls_context.Dr2 = reg_value.GetAsUInt64();
5005b5274eaSAleksandr Urakov break;
5015b5274eaSAleksandr Urakov case lldb_dr3_x86_64:
5025b5274eaSAleksandr Urakov tls_context.Dr3 = reg_value.GetAsUInt64();
5035b5274eaSAleksandr Urakov break;
5045b5274eaSAleksandr Urakov case lldb_dr4_x86_64:
5055b5274eaSAleksandr Urakov return Status("register DR4 is obsolete");
5065b5274eaSAleksandr Urakov case lldb_dr5_x86_64:
5075b5274eaSAleksandr Urakov return Status("register DR5 is obsolete");
5085b5274eaSAleksandr Urakov case lldb_dr6_x86_64:
5095b5274eaSAleksandr Urakov tls_context.Dr6 = reg_value.GetAsUInt64();
5105b5274eaSAleksandr Urakov break;
5115b5274eaSAleksandr Urakov case lldb_dr7_x86_64:
5125b5274eaSAleksandr Urakov tls_context.Dr7 = reg_value.GetAsUInt64();
5135b5274eaSAleksandr Urakov break;
5145b5274eaSAleksandr Urakov }
5155b5274eaSAleksandr Urakov
5165b5274eaSAleksandr Urakov return SetThreadContextHelper(thread_handle, &tls_context);
5175b5274eaSAleksandr Urakov }
5185b5274eaSAleksandr Urakov
5195146a9eaSAaron Smith Status
ReadRegister(const RegisterInfo * reg_info,RegisterValue & reg_value)5205146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::ReadRegister(const RegisterInfo *reg_info,
5215146a9eaSAaron Smith RegisterValue ®_value) {
5225146a9eaSAaron Smith Status error;
5235146a9eaSAaron Smith if (!reg_info) {
5245146a9eaSAaron Smith error.SetErrorString("reg_info NULL");
5255146a9eaSAaron Smith return error;
5265146a9eaSAaron Smith }
5275146a9eaSAaron Smith
5285146a9eaSAaron Smith const uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
5295146a9eaSAaron Smith if (reg == LLDB_INVALID_REGNUM) {
5305146a9eaSAaron Smith // This is likely an internal register for lldb use only and should not be
5315146a9eaSAaron Smith // directly queried.
5325146a9eaSAaron Smith error.SetErrorStringWithFormat("register \"%s\" is an internal-only lldb "
5335146a9eaSAaron Smith "register, cannot read directly",
5345146a9eaSAaron Smith reg_info->name);
5355146a9eaSAaron Smith return error;
5365146a9eaSAaron Smith }
5375146a9eaSAaron Smith
5385146a9eaSAaron Smith if (IsGPR(reg))
5395146a9eaSAaron Smith return GPRRead(reg, reg_value);
5405146a9eaSAaron Smith
5415146a9eaSAaron Smith if (IsFPR(reg))
5425146a9eaSAaron Smith return FPRRead(reg, reg_value);
5435146a9eaSAaron Smith
5445b5274eaSAleksandr Urakov if (IsDR(reg))
5455b5274eaSAleksandr Urakov return DRRead(reg, reg_value);
5465b5274eaSAleksandr Urakov
5475146a9eaSAaron Smith return Status("unimplemented");
5485146a9eaSAaron Smith }
5495146a9eaSAaron Smith
WriteRegister(const RegisterInfo * reg_info,const RegisterValue & reg_value)5505146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::WriteRegister(
5515146a9eaSAaron Smith const RegisterInfo *reg_info, const RegisterValue ®_value) {
5525146a9eaSAaron Smith Status error;
5535146a9eaSAaron Smith
5545146a9eaSAaron Smith if (!reg_info) {
5555146a9eaSAaron Smith error.SetErrorString("reg_info NULL");
5565146a9eaSAaron Smith return error;
5575146a9eaSAaron Smith }
5585146a9eaSAaron Smith
5595146a9eaSAaron Smith const uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
5605146a9eaSAaron Smith if (reg == LLDB_INVALID_REGNUM) {
5615146a9eaSAaron Smith // This is likely an internal register for lldb use only and should not be
562900f9ba2STatyana Krasnukha // directly written.
5635146a9eaSAaron Smith error.SetErrorStringWithFormat("register \"%s\" is an internal-only lldb "
564900f9ba2STatyana Krasnukha "register, cannot write directly",
5655146a9eaSAaron Smith reg_info->name);
5665146a9eaSAaron Smith return error;
5675146a9eaSAaron Smith }
5685146a9eaSAaron Smith
5695146a9eaSAaron Smith if (IsGPR(reg))
5705146a9eaSAaron Smith return GPRWrite(reg, reg_value);
5715146a9eaSAaron Smith
5725146a9eaSAaron Smith if (IsFPR(reg))
5735146a9eaSAaron Smith return FPRWrite(reg, reg_value);
5745146a9eaSAaron Smith
5755b5274eaSAleksandr Urakov if (IsDR(reg))
5765b5274eaSAleksandr Urakov return DRWrite(reg, reg_value);
5775b5274eaSAleksandr Urakov
5785146a9eaSAaron Smith return Status("unimplemented");
5795146a9eaSAaron Smith }
5805146a9eaSAaron Smith
ReadAllRegisterValues(lldb::WritableDataBufferSP & data_sp)5815146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::ReadAllRegisterValues(
582*c2f64601SJonas Devlieghere lldb::WritableDataBufferSP &data_sp) {
5835146a9eaSAaron Smith const size_t data_size = REG_CONTEXT_SIZE;
5845146a9eaSAaron Smith data_sp = std::make_shared<DataBufferHeap>(data_size, 0);
5855146a9eaSAaron Smith ::CONTEXT tls_context;
5865146a9eaSAaron Smith Status error =
5875146a9eaSAaron Smith GetThreadContextHelper(GetThreadHandle(), &tls_context, CONTEXT_ALL);
5885146a9eaSAaron Smith if (error.Fail())
5895146a9eaSAaron Smith return error;
5905146a9eaSAaron Smith
5915146a9eaSAaron Smith uint8_t *dst = data_sp->GetBytes();
5925146a9eaSAaron Smith ::memcpy(dst, &tls_context, data_size);
5935146a9eaSAaron Smith return error;
5945146a9eaSAaron Smith }
5955146a9eaSAaron Smith
WriteAllRegisterValues(const lldb::DataBufferSP & data_sp)5965146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::WriteAllRegisterValues(
5975146a9eaSAaron Smith const lldb::DataBufferSP &data_sp) {
5985146a9eaSAaron Smith Status error;
5995146a9eaSAaron Smith const size_t data_size = REG_CONTEXT_SIZE;
6005146a9eaSAaron Smith if (!data_sp) {
6015146a9eaSAaron Smith error.SetErrorStringWithFormat(
6025146a9eaSAaron Smith "NativeRegisterContextWindows_x86_64::%s invalid data_sp provided",
6035146a9eaSAaron Smith __FUNCTION__);
6045146a9eaSAaron Smith return error;
6055146a9eaSAaron Smith }
6065146a9eaSAaron Smith
6075146a9eaSAaron Smith if (data_sp->GetByteSize() != data_size) {
6085146a9eaSAaron Smith error.SetErrorStringWithFormatv(
6095146a9eaSAaron Smith "data_sp contained mismatched data size, expected {0}, actual {1}",
6105146a9eaSAaron Smith data_size, data_sp->GetByteSize());
6115146a9eaSAaron Smith return error;
6125146a9eaSAaron Smith }
6135146a9eaSAaron Smith
6145146a9eaSAaron Smith ::CONTEXT tls_context;
6155146a9eaSAaron Smith memcpy(&tls_context, data_sp->GetBytes(), data_size);
6165146a9eaSAaron Smith return SetThreadContextHelper(GetThreadHandle(), &tls_context);
6175146a9eaSAaron Smith }
6185146a9eaSAaron Smith
IsWatchpointHit(uint32_t wp_index,bool & is_hit)6195146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::IsWatchpointHit(uint32_t wp_index,
6205146a9eaSAaron Smith bool &is_hit) {
6215b5274eaSAleksandr Urakov is_hit = false;
6225b5274eaSAleksandr Urakov
6235b5274eaSAleksandr Urakov if (wp_index >= NumSupportedHardwareWatchpoints())
6245b5274eaSAleksandr Urakov return Status("watchpoint index out of range");
6255b5274eaSAleksandr Urakov
6265b5274eaSAleksandr Urakov RegisterValue reg_value;
6275b5274eaSAleksandr Urakov Status error = DRRead(lldb_dr6_x86_64, reg_value);
6285b5274eaSAleksandr Urakov if (error.Fail())
6295b5274eaSAleksandr Urakov return error;
6305b5274eaSAleksandr Urakov
6315b5274eaSAleksandr Urakov is_hit = reg_value.GetAsUInt64() & (1ULL << wp_index);
6325b5274eaSAleksandr Urakov
6335b5274eaSAleksandr Urakov return {};
6345146a9eaSAaron Smith }
6355146a9eaSAaron Smith
GetWatchpointHitIndex(uint32_t & wp_index,lldb::addr_t trap_addr)6365146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::GetWatchpointHitIndex(
6375146a9eaSAaron Smith uint32_t &wp_index, lldb::addr_t trap_addr) {
6385b5274eaSAleksandr Urakov wp_index = LLDB_INVALID_INDEX32;
6395b5274eaSAleksandr Urakov
6405b5274eaSAleksandr Urakov for (uint32_t i = 0; i < NumSupportedHardwareWatchpoints(); i++) {
6415b5274eaSAleksandr Urakov bool is_hit;
6425b5274eaSAleksandr Urakov Status error = IsWatchpointHit(i, is_hit);
6435b5274eaSAleksandr Urakov if (error.Fail())
6445b5274eaSAleksandr Urakov return error;
6455b5274eaSAleksandr Urakov
6465b5274eaSAleksandr Urakov if (is_hit) {
6475b5274eaSAleksandr Urakov wp_index = i;
6485b5274eaSAleksandr Urakov return {};
6495b5274eaSAleksandr Urakov }
6505b5274eaSAleksandr Urakov }
6515b5274eaSAleksandr Urakov
6525b5274eaSAleksandr Urakov return {};
6535146a9eaSAaron Smith }
6545146a9eaSAaron Smith
6555146a9eaSAaron Smith Status
IsWatchpointVacant(uint32_t wp_index,bool & is_vacant)6565146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::IsWatchpointVacant(uint32_t wp_index,
6575146a9eaSAaron Smith bool &is_vacant) {
6585b5274eaSAleksandr Urakov is_vacant = false;
6595146a9eaSAaron Smith
6605b5274eaSAleksandr Urakov if (wp_index >= NumSupportedHardwareWatchpoints())
6615b5274eaSAleksandr Urakov return Status("Watchpoint index out of range");
6625b5274eaSAleksandr Urakov
6635b5274eaSAleksandr Urakov RegisterValue reg_value;
6645b5274eaSAleksandr Urakov Status error = DRRead(lldb_dr7_x86_64, reg_value);
6655b5274eaSAleksandr Urakov if (error.Fail())
6665b5274eaSAleksandr Urakov return error;
6675b5274eaSAleksandr Urakov
6685b5274eaSAleksandr Urakov is_vacant = !(reg_value.GetAsUInt64() & (1ULL << (2 * wp_index)));
6695b5274eaSAleksandr Urakov
6705b5274eaSAleksandr Urakov return error;
6715146a9eaSAaron Smith }
6725146a9eaSAaron Smith
ClearHardwareWatchpoint(uint32_t wp_index)6735146a9eaSAaron Smith bool NativeRegisterContextWindows_x86_64::ClearHardwareWatchpoint(
6745146a9eaSAaron Smith uint32_t wp_index) {
6755b5274eaSAleksandr Urakov if (wp_index >= NumSupportedHardwareWatchpoints())
6765146a9eaSAaron Smith return false;
6775b5274eaSAleksandr Urakov
6785b5274eaSAleksandr Urakov // for watchpoints 0, 1, 2, or 3, respectively, clear bits 0, 1, 2, or 3 of
6795b5274eaSAleksandr Urakov // the debug status register (DR6)
6805b5274eaSAleksandr Urakov
6815b5274eaSAleksandr Urakov RegisterValue reg_value;
6825b5274eaSAleksandr Urakov Status error = DRRead(lldb_dr6_x86_64, reg_value);
6835b5274eaSAleksandr Urakov if (error.Fail())
6845b5274eaSAleksandr Urakov return false;
6855b5274eaSAleksandr Urakov
6865b5274eaSAleksandr Urakov uint64_t bit_mask = 1ULL << wp_index;
6875b5274eaSAleksandr Urakov uint64_t status_bits = reg_value.GetAsUInt64() & ~bit_mask;
6885b5274eaSAleksandr Urakov error = DRWrite(lldb_dr6_x86_64, RegisterValue(status_bits));
6895b5274eaSAleksandr Urakov if (error.Fail())
6905b5274eaSAleksandr Urakov return false;
6915b5274eaSAleksandr Urakov
6925b5274eaSAleksandr Urakov // for watchpoints 0, 1, 2, or 3, respectively, clear bits {0-1,16-19},
6935b5274eaSAleksandr Urakov // {2-3,20-23}, {4-5,24-27}, or {6-7,28-31} of the debug control register
6945b5274eaSAleksandr Urakov // (DR7)
6955b5274eaSAleksandr Urakov
6965b5274eaSAleksandr Urakov error = DRRead(lldb_dr7_x86_64, reg_value);
6975b5274eaSAleksandr Urakov if (error.Fail())
6985b5274eaSAleksandr Urakov return false;
6995b5274eaSAleksandr Urakov
7005b5274eaSAleksandr Urakov bit_mask = (0x3 << (2 * wp_index)) | (0xF << (16 + 4 * wp_index));
7015b5274eaSAleksandr Urakov uint64_t control_bits = reg_value.GetAsUInt64() & ~bit_mask;
7025b5274eaSAleksandr Urakov return DRWrite(lldb_dr7_x86_64, RegisterValue(control_bits)).Success();
7035146a9eaSAaron Smith }
7045146a9eaSAaron Smith
ClearAllHardwareWatchpoints()7055146a9eaSAaron Smith Status NativeRegisterContextWindows_x86_64::ClearAllHardwareWatchpoints() {
7065b5274eaSAleksandr Urakov RegisterValue reg_value;
7075b5274eaSAleksandr Urakov
7085b5274eaSAleksandr Urakov // clear bits {0-4} of the debug status register (DR6)
7095b5274eaSAleksandr Urakov
7105b5274eaSAleksandr Urakov Status error = DRRead(lldb_dr6_x86_64, reg_value);
7115b5274eaSAleksandr Urakov if (error.Fail())
7125b5274eaSAleksandr Urakov return error;
7135b5274eaSAleksandr Urakov
7145b5274eaSAleksandr Urakov uint64_t status_bits = reg_value.GetAsUInt64() & ~0xFULL;
7155b5274eaSAleksandr Urakov error = DRWrite(lldb_dr6_x86_64, RegisterValue(status_bits));
7165b5274eaSAleksandr Urakov if (error.Fail())
7175b5274eaSAleksandr Urakov return error;
7185b5274eaSAleksandr Urakov
7195b5274eaSAleksandr Urakov // clear bits {0-7,16-31} of the debug control register (DR7)
7205b5274eaSAleksandr Urakov
7215b5274eaSAleksandr Urakov error = DRRead(lldb_dr7_x86_64, reg_value);
7225b5274eaSAleksandr Urakov if (error.Fail())
7235b5274eaSAleksandr Urakov return error;
7245b5274eaSAleksandr Urakov
7255b5274eaSAleksandr Urakov uint64_t control_bits = reg_value.GetAsUInt64() & ~0xFFFF00FFULL;
7265b5274eaSAleksandr Urakov return DRWrite(lldb_dr7_x86_64, RegisterValue(control_bits));
7275146a9eaSAaron Smith }
7285146a9eaSAaron Smith
SetHardwareWatchpoint(lldb::addr_t addr,size_t size,uint32_t watch_flags)7295146a9eaSAaron Smith uint32_t NativeRegisterContextWindows_x86_64::SetHardwareWatchpoint(
7305146a9eaSAaron Smith lldb::addr_t addr, size_t size, uint32_t watch_flags) {
7315b5274eaSAleksandr Urakov switch (size) {
7325b5274eaSAleksandr Urakov case 1:
7335b5274eaSAleksandr Urakov case 2:
7345b5274eaSAleksandr Urakov case 4:
7355b5274eaSAleksandr Urakov case 8:
7365b5274eaSAleksandr Urakov break;
7375b5274eaSAleksandr Urakov default:
7385146a9eaSAaron Smith return LLDB_INVALID_INDEX32;
7395146a9eaSAaron Smith }
7405146a9eaSAaron Smith
7415b5274eaSAleksandr Urakov if (watch_flags == 0x2)
7425b5274eaSAleksandr Urakov watch_flags = 0x3;
7435b5274eaSAleksandr Urakov
7445b5274eaSAleksandr Urakov if (watch_flags != 0x1 && watch_flags != 0x3)
7455b5274eaSAleksandr Urakov return LLDB_INVALID_INDEX32;
7465b5274eaSAleksandr Urakov
7475b5274eaSAleksandr Urakov for (uint32_t wp_index = 0; wp_index < NumSupportedHardwareWatchpoints();
7485b5274eaSAleksandr Urakov ++wp_index) {
7495b5274eaSAleksandr Urakov bool is_vacant;
7505b5274eaSAleksandr Urakov if (IsWatchpointVacant(wp_index, is_vacant).Fail())
7515b5274eaSAleksandr Urakov return LLDB_INVALID_INDEX32;
7525b5274eaSAleksandr Urakov
7535b5274eaSAleksandr Urakov if (is_vacant) {
7545b5274eaSAleksandr Urakov if (!ClearHardwareWatchpoint(wp_index))
7555b5274eaSAleksandr Urakov return LLDB_INVALID_INDEX32;
7565b5274eaSAleksandr Urakov
7575b5274eaSAleksandr Urakov if (ApplyHardwareBreakpoint(wp_index, addr, size, watch_flags).Fail())
7585b5274eaSAleksandr Urakov return LLDB_INVALID_INDEX32;
7595b5274eaSAleksandr Urakov
7605b5274eaSAleksandr Urakov return wp_index;
7615b5274eaSAleksandr Urakov }
7625b5274eaSAleksandr Urakov }
7635b5274eaSAleksandr Urakov return LLDB_INVALID_INDEX32;
7645b5274eaSAleksandr Urakov }
7655b5274eaSAleksandr Urakov
ApplyHardwareBreakpoint(uint32_t wp_index,lldb::addr_t addr,size_t size,uint32_t flags)7665b5274eaSAleksandr Urakov Status NativeRegisterContextWindows_x86_64::ApplyHardwareBreakpoint(
7675b5274eaSAleksandr Urakov uint32_t wp_index, lldb::addr_t addr, size_t size, uint32_t flags) {
7685b5274eaSAleksandr Urakov RegisterValue reg_value;
7695b5274eaSAleksandr Urakov auto error = DRRead(lldb_dr7_x86_64, reg_value);
7705b5274eaSAleksandr Urakov if (error.Fail())
7715b5274eaSAleksandr Urakov return error;
7725b5274eaSAleksandr Urakov
7735b5274eaSAleksandr Urakov // for watchpoints 0, 1, 2, or 3, respectively, set bits 1, 3, 5, or 7
7745b5274eaSAleksandr Urakov uint64_t enable_bit = 1ULL << (2 * wp_index);
7755b5274eaSAleksandr Urakov
7765b5274eaSAleksandr Urakov // set bits 16-17, 20-21, 24-25, or 28-29
7775b5274eaSAleksandr Urakov // with 0b01 for write, and 0b11 for read/write
7785b5274eaSAleksandr Urakov uint64_t rw_bits = flags << (16 + 4 * wp_index);
7795b5274eaSAleksandr Urakov
7805b5274eaSAleksandr Urakov // set bits 18-19, 22-23, 26-27, or 30-31
7815b5274eaSAleksandr Urakov // with 0b00, 0b01, 0b10, or 0b11
7825b5274eaSAleksandr Urakov // for 1, 2, 8 (if supported), or 4 bytes, respectively
7835b5274eaSAleksandr Urakov uint64_t size_bits = (size == 8 ? 0x2 : size - 1) << (18 + 4 * wp_index);
7845b5274eaSAleksandr Urakov
7855b5274eaSAleksandr Urakov uint64_t bit_mask = (0x3 << (2 * wp_index)) | (0xF << (16 + 4 * wp_index));
7865b5274eaSAleksandr Urakov
7875b5274eaSAleksandr Urakov uint64_t control_bits = reg_value.GetAsUInt64() & ~bit_mask;
7885b5274eaSAleksandr Urakov control_bits |= enable_bit | rw_bits | size_bits;
7895b5274eaSAleksandr Urakov
7905b5274eaSAleksandr Urakov error = DRWrite(lldb_dr7_x86_64, RegisterValue(control_bits));
7915b5274eaSAleksandr Urakov if (error.Fail())
7925b5274eaSAleksandr Urakov return error;
7935b5274eaSAleksandr Urakov
7945b5274eaSAleksandr Urakov error = DRWrite(lldb_dr0_x86_64 + wp_index, RegisterValue(addr));
7955b5274eaSAleksandr Urakov if (error.Fail())
7965b5274eaSAleksandr Urakov return error;
7975b5274eaSAleksandr Urakov
7985b5274eaSAleksandr Urakov return {};
7995b5274eaSAleksandr Urakov }
8005b5274eaSAleksandr Urakov
8015146a9eaSAaron Smith lldb::addr_t
GetWatchpointAddress(uint32_t wp_index)8025146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::GetWatchpointAddress(uint32_t wp_index) {
8035b5274eaSAleksandr Urakov if (wp_index >= NumSupportedHardwareWatchpoints())
8045146a9eaSAaron Smith return LLDB_INVALID_ADDRESS;
8055b5274eaSAleksandr Urakov
8065b5274eaSAleksandr Urakov RegisterValue reg_value;
8075b5274eaSAleksandr Urakov if (DRRead(lldb_dr0_x86_64 + wp_index, reg_value).Fail())
8085b5274eaSAleksandr Urakov return LLDB_INVALID_ADDRESS;
8095b5274eaSAleksandr Urakov
8105b5274eaSAleksandr Urakov return reg_value.GetAsUInt64();
8115146a9eaSAaron Smith }
8125146a9eaSAaron Smith
8135146a9eaSAaron Smith uint32_t
NumSupportedHardwareWatchpoints()8145146a9eaSAaron Smith NativeRegisterContextWindows_x86_64::NumSupportedHardwareWatchpoints() {
8155b5274eaSAleksandr Urakov return 4;
8165146a9eaSAaron Smith }
8175146a9eaSAaron Smith
818900f9ba2STatyana Krasnukha #endif // defined(__x86_64__) || defined(_M_X64)
819