180814287SRaphael Isemann //===-- NativeRegisterContextWindows_WoW64.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_WoW64.h"
125146a9eaSAaron Smith 
135146a9eaSAaron Smith #include "NativeThreadWindows.h"
145146a9eaSAaron Smith #include "Plugins/Process/Utility/RegisterContextWindows_i386.h"
155146a9eaSAaron Smith #include "ProcessWindowsLog.h"
165146a9eaSAaron Smith #include "lldb/Host/HostInfo.h"
175146a9eaSAaron Smith #include "lldb/Host/HostThread.h"
182e8b5755SSaleem Abdulrasool #include "lldb/Host/windows/HostThreadWindows.h"
195146a9eaSAaron Smith #include "lldb/Host/windows/windows.h"
205146a9eaSAaron Smith 
215146a9eaSAaron Smith #include "lldb/Utility/Log.h"
225146a9eaSAaron Smith #include "lldb/Utility/RegisterValue.h"
235146a9eaSAaron Smith #include "llvm/ADT/STLExtras.h"
245146a9eaSAaron Smith 
255146a9eaSAaron Smith using namespace lldb;
265146a9eaSAaron Smith using namespace lldb_private;
275146a9eaSAaron Smith 
285146a9eaSAaron Smith #define REG_CONTEXT_SIZE sizeof(::WOW64_CONTEXT)
295146a9eaSAaron Smith 
305146a9eaSAaron Smith namespace {
315146a9eaSAaron Smith static const uint32_t g_gpr_regnums_WoW64[] = {
325146a9eaSAaron Smith     lldb_eax_i386,      lldb_ebx_i386,    lldb_ecx_i386, lldb_edx_i386,
335146a9eaSAaron Smith     lldb_edi_i386,      lldb_esi_i386,    lldb_ebp_i386, lldb_esp_i386,
345146a9eaSAaron Smith     lldb_eip_i386,      lldb_eflags_i386, lldb_cs_i386,  lldb_fs_i386,
355146a9eaSAaron Smith     lldb_gs_i386,       lldb_ss_i386,     lldb_ds_i386,  lldb_es_i386,
365146a9eaSAaron Smith     LLDB_INVALID_REGNUM // Register set must be terminated with this flag.
375146a9eaSAaron Smith };
385146a9eaSAaron Smith 
395146a9eaSAaron Smith static const RegisterSet g_reg_sets_WoW64[] = {
405146a9eaSAaron Smith     {"General Purpose Registers", "gpr",
41f15014ffSBenjamin Kramer      llvm::array_lengthof(g_gpr_regnums_WoW64) - 1, g_gpr_regnums_WoW64},
425146a9eaSAaron Smith };
435146a9eaSAaron Smith enum { k_num_register_sets = 1 };
445146a9eaSAaron Smith 
455146a9eaSAaron Smith static const DWORD kWoW64ContextFlags =
465146a9eaSAaron Smith     WOW64_CONTEXT_CONTROL | WOW64_CONTEXT_INTEGER | WOW64_CONTEXT_SEGMENTS;
475146a9eaSAaron Smith 
485146a9eaSAaron Smith } // namespace
495146a9eaSAaron Smith 
505146a9eaSAaron Smith static RegisterInfoInterface *
CreateRegisterInfoInterface(const ArchSpec & target_arch)515146a9eaSAaron Smith CreateRegisterInfoInterface(const ArchSpec &target_arch) {
525146a9eaSAaron Smith   // i686 32bit instruction set.
535146a9eaSAaron Smith   assert((target_arch.GetAddressByteSize() == 4 &&
545146a9eaSAaron Smith           HostInfo::GetArchitecture().GetAddressByteSize() == 8) &&
555146a9eaSAaron Smith          "Register setting path assumes this is a 64-bit host");
565146a9eaSAaron Smith   return new RegisterContextWindows_i386(target_arch);
575146a9eaSAaron Smith }
585146a9eaSAaron Smith 
595b5274eaSAleksandr Urakov static Status
GetWoW64ThreadContextHelper(lldb::thread_t thread_handle,PWOW64_CONTEXT context_ptr,const DWORD control_flag=kWoW64ContextFlags)605b5274eaSAleksandr Urakov GetWoW64ThreadContextHelper(lldb::thread_t thread_handle,
615b5274eaSAleksandr Urakov                             PWOW64_CONTEXT context_ptr,
625b5274eaSAleksandr Urakov                             const DWORD control_flag = kWoW64ContextFlags) {
636730df47SPavel Labath   Log *log = GetLog(WindowsLog::Registers);
645146a9eaSAaron Smith   Status error;
655146a9eaSAaron Smith   memset(context_ptr, 0, sizeof(::WOW64_CONTEXT));
665b5274eaSAleksandr Urakov   context_ptr->ContextFlags = control_flag;
675146a9eaSAaron Smith   if (!::Wow64GetThreadContext(thread_handle, context_ptr)) {
685146a9eaSAaron Smith     error.SetError(GetLastError(), eErrorTypeWin32);
695146a9eaSAaron Smith     LLDB_LOG(log, "{0} Wow64GetThreadContext failed with error {1}",
705146a9eaSAaron Smith              __FUNCTION__, error);
715146a9eaSAaron Smith     return error;
725146a9eaSAaron Smith   }
735146a9eaSAaron Smith   return Status();
745146a9eaSAaron Smith }
755146a9eaSAaron Smith 
SetWoW64ThreadContextHelper(lldb::thread_t thread_handle,PWOW64_CONTEXT context_ptr)765146a9eaSAaron Smith static Status SetWoW64ThreadContextHelper(lldb::thread_t thread_handle,
775146a9eaSAaron Smith                                           PWOW64_CONTEXT context_ptr) {
786730df47SPavel Labath   Log *log = GetLog(WindowsLog::Registers);
795146a9eaSAaron Smith   Status error;
805146a9eaSAaron Smith   if (!::Wow64SetThreadContext(thread_handle, context_ptr)) {
815146a9eaSAaron Smith     error.SetError(GetLastError(), eErrorTypeWin32);
825146a9eaSAaron Smith     LLDB_LOG(log, "{0} Wow64SetThreadContext failed with error {1}",
835146a9eaSAaron Smith              __FUNCTION__, error);
845146a9eaSAaron Smith     return error;
855146a9eaSAaron Smith   }
865146a9eaSAaron Smith   return Status();
875146a9eaSAaron Smith }
885146a9eaSAaron Smith 
NativeRegisterContextWindows_WoW64(const ArchSpec & target_arch,NativeThreadProtocol & native_thread)895146a9eaSAaron Smith NativeRegisterContextWindows_WoW64::NativeRegisterContextWindows_WoW64(
905146a9eaSAaron Smith     const ArchSpec &target_arch, NativeThreadProtocol &native_thread)
915146a9eaSAaron Smith     : NativeRegisterContextWindows(native_thread,
925146a9eaSAaron Smith                                    CreateRegisterInfoInterface(target_arch)) {}
935146a9eaSAaron Smith 
IsGPR(uint32_t reg_index) const945146a9eaSAaron Smith bool NativeRegisterContextWindows_WoW64::IsGPR(uint32_t reg_index) const {
955146a9eaSAaron Smith   return (reg_index >= k_first_gpr_i386 && reg_index < k_first_alias_i386);
965146a9eaSAaron Smith }
975146a9eaSAaron Smith 
IsDR(uint32_t reg_index) const985b5274eaSAleksandr Urakov bool NativeRegisterContextWindows_WoW64::IsDR(uint32_t reg_index) const {
995b5274eaSAleksandr Urakov   return (reg_index >= lldb_dr0_i386 && reg_index <= lldb_dr7_i386);
1005b5274eaSAleksandr Urakov }
1015b5274eaSAleksandr Urakov 
GetRegisterSetCount() const1025146a9eaSAaron Smith uint32_t NativeRegisterContextWindows_WoW64::GetRegisterSetCount() const {
1035146a9eaSAaron Smith   return k_num_register_sets;
1045146a9eaSAaron Smith }
1055146a9eaSAaron Smith 
1065146a9eaSAaron Smith const RegisterSet *
GetRegisterSet(uint32_t set_index) const1075146a9eaSAaron Smith NativeRegisterContextWindows_WoW64::GetRegisterSet(uint32_t set_index) const {
1085146a9eaSAaron Smith   if (set_index >= k_num_register_sets)
1095146a9eaSAaron Smith     return nullptr;
1105146a9eaSAaron Smith   return &g_reg_sets_WoW64[set_index];
1115146a9eaSAaron Smith }
1125146a9eaSAaron Smith 
GPRRead(const uint32_t reg,RegisterValue & reg_value)1135146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::GPRRead(const uint32_t reg,
1145146a9eaSAaron Smith                                                    RegisterValue &reg_value) {
1155146a9eaSAaron Smith   ::WOW64_CONTEXT tls_context;
1165146a9eaSAaron Smith   Status error = GetWoW64ThreadContextHelper(GetThreadHandle(), &tls_context);
1175146a9eaSAaron Smith   if (error.Fail())
1185146a9eaSAaron Smith     return error;
1195146a9eaSAaron Smith 
1205146a9eaSAaron Smith   switch (reg) {
1215146a9eaSAaron Smith   case lldb_eax_i386:
1225146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Eax);
1235146a9eaSAaron Smith     break;
1245146a9eaSAaron Smith   case lldb_ebx_i386:
1255146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Ebx);
1265146a9eaSAaron Smith     break;
1275146a9eaSAaron Smith   case lldb_ecx_i386:
1285146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Ecx);
1295146a9eaSAaron Smith     break;
1305146a9eaSAaron Smith   case lldb_edx_i386:
1315146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Edx);
1325146a9eaSAaron Smith     break;
1335146a9eaSAaron Smith   case lldb_edi_i386:
1345146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Edi);
1355146a9eaSAaron Smith     break;
1365146a9eaSAaron Smith   case lldb_esi_i386:
1375146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Esi);
1385146a9eaSAaron Smith     break;
1395146a9eaSAaron Smith   case lldb_ebp_i386:
1405146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Ebp);
1415146a9eaSAaron Smith     break;
1425146a9eaSAaron Smith   case lldb_esp_i386:
1435146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Esp);
1445146a9eaSAaron Smith     break;
1455146a9eaSAaron Smith   case lldb_eip_i386:
1465146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.Eip);
1475146a9eaSAaron Smith     break;
1485146a9eaSAaron Smith   case lldb_eflags_i386:
1495146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.EFlags);
1505146a9eaSAaron Smith     break;
1515146a9eaSAaron Smith   case lldb_cs_i386:
1525146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.SegCs);
1535146a9eaSAaron Smith     break;
1545146a9eaSAaron Smith   case lldb_fs_i386:
1555146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.SegFs);
1565146a9eaSAaron Smith     break;
1575146a9eaSAaron Smith   case lldb_gs_i386:
1585146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.SegGs);
1595146a9eaSAaron Smith     break;
1605146a9eaSAaron Smith   case lldb_ss_i386:
1615146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.SegSs);
1625146a9eaSAaron Smith     break;
1635146a9eaSAaron Smith   case lldb_ds_i386:
1645146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.SegDs);
1655146a9eaSAaron Smith     break;
1665146a9eaSAaron Smith   case lldb_es_i386:
1675146a9eaSAaron Smith     reg_value.SetUInt32(tls_context.SegEs);
1685146a9eaSAaron Smith     break;
1695146a9eaSAaron Smith   }
1705146a9eaSAaron Smith 
1715146a9eaSAaron Smith   return error;
1725146a9eaSAaron Smith }
1735146a9eaSAaron Smith 
1745146a9eaSAaron Smith Status
GPRWrite(const uint32_t reg,const RegisterValue & reg_value)1755146a9eaSAaron Smith NativeRegisterContextWindows_WoW64::GPRWrite(const uint32_t reg,
1765146a9eaSAaron Smith                                              const RegisterValue &reg_value) {
1775146a9eaSAaron Smith   ::WOW64_CONTEXT tls_context;
1785146a9eaSAaron Smith   auto thread_handle = GetThreadHandle();
1795146a9eaSAaron Smith   Status error = GetWoW64ThreadContextHelper(thread_handle, &tls_context);
1805146a9eaSAaron Smith   if (error.Fail())
1815146a9eaSAaron Smith     return error;
1825146a9eaSAaron Smith 
1835146a9eaSAaron Smith   switch (reg) {
1845146a9eaSAaron Smith   case lldb_eax_i386:
1855146a9eaSAaron Smith     tls_context.Eax = reg_value.GetAsUInt32();
1865146a9eaSAaron Smith     break;
1875146a9eaSAaron Smith   case lldb_ebx_i386:
1885146a9eaSAaron Smith     tls_context.Ebx = reg_value.GetAsUInt32();
1895146a9eaSAaron Smith     break;
1905146a9eaSAaron Smith   case lldb_ecx_i386:
1915146a9eaSAaron Smith     tls_context.Ecx = reg_value.GetAsUInt32();
1925146a9eaSAaron Smith     break;
1935146a9eaSAaron Smith   case lldb_edx_i386:
1945146a9eaSAaron Smith     tls_context.Edx = reg_value.GetAsUInt32();
1955146a9eaSAaron Smith     break;
1965146a9eaSAaron Smith   case lldb_edi_i386:
1975146a9eaSAaron Smith     tls_context.Edi = reg_value.GetAsUInt32();
1985146a9eaSAaron Smith     break;
1995146a9eaSAaron Smith   case lldb_esi_i386:
2005146a9eaSAaron Smith     tls_context.Esi = reg_value.GetAsUInt32();
2015146a9eaSAaron Smith     break;
2025146a9eaSAaron Smith   case lldb_ebp_i386:
2035146a9eaSAaron Smith     tls_context.Ebp = reg_value.GetAsUInt32();
2045146a9eaSAaron Smith     break;
2055146a9eaSAaron Smith   case lldb_esp_i386:
2065146a9eaSAaron Smith     tls_context.Esp = reg_value.GetAsUInt32();
2075146a9eaSAaron Smith     break;
2085146a9eaSAaron Smith   case lldb_eip_i386:
2095146a9eaSAaron Smith     tls_context.Eip = reg_value.GetAsUInt32();
2105146a9eaSAaron Smith     break;
2115146a9eaSAaron Smith   case lldb_eflags_i386:
2125146a9eaSAaron Smith     tls_context.EFlags = reg_value.GetAsUInt32();
2135146a9eaSAaron Smith     break;
2145146a9eaSAaron Smith   case lldb_cs_i386:
2155146a9eaSAaron Smith     tls_context.SegCs = reg_value.GetAsUInt32();
2165146a9eaSAaron Smith     break;
2175146a9eaSAaron Smith   case lldb_fs_i386:
2185146a9eaSAaron Smith     tls_context.SegFs = reg_value.GetAsUInt32();
2195146a9eaSAaron Smith     break;
2205146a9eaSAaron Smith   case lldb_gs_i386:
2215146a9eaSAaron Smith     tls_context.SegGs = reg_value.GetAsUInt32();
2225146a9eaSAaron Smith     break;
2235146a9eaSAaron Smith   case lldb_ss_i386:
2245146a9eaSAaron Smith     tls_context.SegSs = reg_value.GetAsUInt32();
2255146a9eaSAaron Smith     break;
2265146a9eaSAaron Smith   case lldb_ds_i386:
2275146a9eaSAaron Smith     tls_context.SegDs = reg_value.GetAsUInt32();
2285146a9eaSAaron Smith     break;
2295146a9eaSAaron Smith   case lldb_es_i386:
2305146a9eaSAaron Smith     tls_context.SegEs = reg_value.GetAsUInt32();
2315146a9eaSAaron Smith     break;
2325146a9eaSAaron Smith   }
2335146a9eaSAaron Smith 
2345146a9eaSAaron Smith   return SetWoW64ThreadContextHelper(thread_handle, &tls_context);
2355146a9eaSAaron Smith }
2365146a9eaSAaron Smith 
DRRead(const uint32_t reg,RegisterValue & reg_value)2375b5274eaSAleksandr Urakov Status NativeRegisterContextWindows_WoW64::DRRead(const uint32_t reg,
2385b5274eaSAleksandr Urakov                                                   RegisterValue &reg_value) {
2395b5274eaSAleksandr Urakov   ::WOW64_CONTEXT tls_context;
2405b5274eaSAleksandr Urakov   DWORD context_flag = CONTEXT_DEBUG_REGISTERS;
2415b5274eaSAleksandr Urakov   Status error = GetWoW64ThreadContextHelper(GetThreadHandle(), &tls_context,
2425b5274eaSAleksandr Urakov                                              context_flag);
2435b5274eaSAleksandr Urakov   if (error.Fail())
2445b5274eaSAleksandr Urakov     return error;
2455b5274eaSAleksandr Urakov 
2465b5274eaSAleksandr Urakov   switch (reg) {
2475b5274eaSAleksandr Urakov   case lldb_dr0_i386:
2485b5274eaSAleksandr Urakov     reg_value.SetUInt32(tls_context.Dr0);
2495b5274eaSAleksandr Urakov     break;
2505b5274eaSAleksandr Urakov   case lldb_dr1_i386:
2515b5274eaSAleksandr Urakov     reg_value.SetUInt32(tls_context.Dr1);
2525b5274eaSAleksandr Urakov     break;
2535b5274eaSAleksandr Urakov   case lldb_dr2_i386:
2545b5274eaSAleksandr Urakov     reg_value.SetUInt32(tls_context.Dr2);
2555b5274eaSAleksandr Urakov     break;
2565b5274eaSAleksandr Urakov   case lldb_dr3_i386:
2575b5274eaSAleksandr Urakov     reg_value.SetUInt32(tls_context.Dr3);
2585b5274eaSAleksandr Urakov     break;
2595b5274eaSAleksandr Urakov   case lldb_dr4_i386:
2605b5274eaSAleksandr Urakov     return Status("register DR4 is obsolete");
2615b5274eaSAleksandr Urakov   case lldb_dr5_i386:
2625b5274eaSAleksandr Urakov     return Status("register DR5 is obsolete");
2635b5274eaSAleksandr Urakov   case lldb_dr6_i386:
2645b5274eaSAleksandr Urakov     reg_value.SetUInt32(tls_context.Dr6);
2655b5274eaSAleksandr Urakov     break;
2665b5274eaSAleksandr Urakov   case lldb_dr7_i386:
2675b5274eaSAleksandr Urakov     reg_value.SetUInt32(tls_context.Dr7);
2685b5274eaSAleksandr Urakov     break;
2695b5274eaSAleksandr Urakov   }
2705b5274eaSAleksandr Urakov 
2715b5274eaSAleksandr Urakov   return {};
2725b5274eaSAleksandr Urakov }
2735b5274eaSAleksandr Urakov 
2745b5274eaSAleksandr Urakov Status
DRWrite(const uint32_t reg,const RegisterValue & reg_value)2755b5274eaSAleksandr Urakov NativeRegisterContextWindows_WoW64::DRWrite(const uint32_t reg,
2765b5274eaSAleksandr Urakov                                             const RegisterValue &reg_value) {
2775b5274eaSAleksandr Urakov   ::WOW64_CONTEXT tls_context;
2785b5274eaSAleksandr Urakov   DWORD context_flag = CONTEXT_DEBUG_REGISTERS;
2795b5274eaSAleksandr Urakov   auto thread_handle = GetThreadHandle();
2805b5274eaSAleksandr Urakov   Status error =
2815b5274eaSAleksandr Urakov       GetWoW64ThreadContextHelper(thread_handle, &tls_context, context_flag);
2825b5274eaSAleksandr Urakov   if (error.Fail())
2835b5274eaSAleksandr Urakov     return error;
2845b5274eaSAleksandr Urakov 
2855b5274eaSAleksandr Urakov   switch (reg) {
2865b5274eaSAleksandr Urakov   case lldb_dr0_i386:
2875b5274eaSAleksandr Urakov     tls_context.Dr0 = reg_value.GetAsUInt32();
2885b5274eaSAleksandr Urakov     break;
2895b5274eaSAleksandr Urakov   case lldb_dr1_i386:
2905b5274eaSAleksandr Urakov     tls_context.Dr1 = reg_value.GetAsUInt32();
2915b5274eaSAleksandr Urakov     break;
2925b5274eaSAleksandr Urakov   case lldb_dr2_i386:
2935b5274eaSAleksandr Urakov     tls_context.Dr2 = reg_value.GetAsUInt32();
2945b5274eaSAleksandr Urakov     break;
2955b5274eaSAleksandr Urakov   case lldb_dr3_i386:
2965b5274eaSAleksandr Urakov     tls_context.Dr3 = reg_value.GetAsUInt32();
2975b5274eaSAleksandr Urakov     break;
2985b5274eaSAleksandr Urakov   case lldb_dr4_i386:
2995b5274eaSAleksandr Urakov     return Status("register DR4 is obsolete");
3005b5274eaSAleksandr Urakov   case lldb_dr5_i386:
3015b5274eaSAleksandr Urakov     return Status("register DR5 is obsolete");
3025b5274eaSAleksandr Urakov   case lldb_dr6_i386:
3035b5274eaSAleksandr Urakov     tls_context.Dr6 = reg_value.GetAsUInt32();
3045b5274eaSAleksandr Urakov     break;
3055b5274eaSAleksandr Urakov   case lldb_dr7_i386:
3065b5274eaSAleksandr Urakov     tls_context.Dr7 = reg_value.GetAsUInt32();
3075b5274eaSAleksandr Urakov     break;
3085b5274eaSAleksandr Urakov   }
3095b5274eaSAleksandr Urakov 
3105b5274eaSAleksandr Urakov   return SetWoW64ThreadContextHelper(thread_handle, &tls_context);
3115b5274eaSAleksandr Urakov }
3125b5274eaSAleksandr Urakov 
3135146a9eaSAaron Smith Status
ReadRegister(const RegisterInfo * reg_info,RegisterValue & reg_value)3145146a9eaSAaron Smith NativeRegisterContextWindows_WoW64::ReadRegister(const RegisterInfo *reg_info,
3155146a9eaSAaron Smith                                                  RegisterValue &reg_value) {
3165146a9eaSAaron Smith   Status error;
3175146a9eaSAaron Smith   if (!reg_info) {
3185146a9eaSAaron Smith     error.SetErrorString("reg_info NULL");
3195146a9eaSAaron Smith     return error;
3205146a9eaSAaron Smith   }
3215146a9eaSAaron Smith 
3225146a9eaSAaron Smith   const uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
3235146a9eaSAaron Smith   if (reg == LLDB_INVALID_REGNUM) {
3245146a9eaSAaron Smith     // This is likely an internal register for lldb use only and should not be
3255146a9eaSAaron Smith     // directly queried.
3265146a9eaSAaron Smith     error.SetErrorStringWithFormat("register \"%s\" is an internal-only lldb "
3275146a9eaSAaron Smith                                    "register, cannot read directly",
3285146a9eaSAaron Smith                                    reg_info->name);
3295146a9eaSAaron Smith     return error;
3305146a9eaSAaron Smith   }
3315146a9eaSAaron Smith 
3325146a9eaSAaron Smith   if (IsGPR(reg))
3335146a9eaSAaron Smith     return GPRRead(reg, reg_value);
3345146a9eaSAaron Smith 
3355b5274eaSAleksandr Urakov   if (IsDR(reg))
3365b5274eaSAleksandr Urakov     return DRRead(reg, reg_value);
3375b5274eaSAleksandr Urakov 
3385146a9eaSAaron Smith   return Status("unimplemented");
3395146a9eaSAaron Smith }
3405146a9eaSAaron Smith 
WriteRegister(const RegisterInfo * reg_info,const RegisterValue & reg_value)3415146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::WriteRegister(
3425146a9eaSAaron Smith     const RegisterInfo *reg_info, const RegisterValue &reg_value) {
3435146a9eaSAaron Smith   Status error;
3445146a9eaSAaron Smith 
3455146a9eaSAaron Smith   if (!reg_info) {
3465146a9eaSAaron Smith     error.SetErrorString("reg_info NULL");
3475146a9eaSAaron Smith     return error;
3485146a9eaSAaron Smith   }
3495146a9eaSAaron Smith 
3505146a9eaSAaron Smith   const uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
3515146a9eaSAaron Smith   if (reg == LLDB_INVALID_REGNUM) {
3525146a9eaSAaron Smith     // This is likely an internal register for lldb use only and should not be
353900f9ba2STatyana Krasnukha     // directly written.
3545146a9eaSAaron Smith     error.SetErrorStringWithFormat("register \"%s\" is an internal-only lldb "
355900f9ba2STatyana Krasnukha                                    "register, cannot write directly",
3565146a9eaSAaron Smith                                    reg_info->name);
3575146a9eaSAaron Smith     return error;
3585146a9eaSAaron Smith   }
3595146a9eaSAaron Smith 
3605146a9eaSAaron Smith   if (IsGPR(reg))
3615146a9eaSAaron Smith     return GPRWrite(reg, reg_value);
3625146a9eaSAaron Smith 
3635b5274eaSAleksandr Urakov   if (IsDR(reg))
3645b5274eaSAleksandr Urakov     return DRWrite(reg, reg_value);
3655b5274eaSAleksandr Urakov 
3665146a9eaSAaron Smith   return Status("unimplemented");
3675146a9eaSAaron Smith }
3685146a9eaSAaron Smith 
ReadAllRegisterValues(lldb::WritableDataBufferSP & data_sp)3695146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::ReadAllRegisterValues(
370*c2f64601SJonas Devlieghere     lldb::WritableDataBufferSP &data_sp) {
3715146a9eaSAaron Smith   const size_t data_size = REG_CONTEXT_SIZE;
3725146a9eaSAaron Smith   data_sp = std::make_shared<DataBufferHeap>(data_size, 0);
3735146a9eaSAaron Smith   ::WOW64_CONTEXT tls_context;
3745146a9eaSAaron Smith   Status error = GetWoW64ThreadContextHelper(GetThreadHandle(), &tls_context);
3755146a9eaSAaron Smith   if (error.Fail())
3765146a9eaSAaron Smith     return error;
3775146a9eaSAaron Smith 
3785146a9eaSAaron Smith   uint8_t *dst = data_sp->GetBytes();
3795146a9eaSAaron Smith   ::memcpy(dst, &tls_context, data_size);
3805146a9eaSAaron Smith   return error;
3815146a9eaSAaron Smith }
3825146a9eaSAaron Smith 
WriteAllRegisterValues(const lldb::DataBufferSP & data_sp)3835146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::WriteAllRegisterValues(
3845146a9eaSAaron Smith     const lldb::DataBufferSP &data_sp) {
3855146a9eaSAaron Smith   Status error;
3865146a9eaSAaron Smith   const size_t data_size = REG_CONTEXT_SIZE;
3875146a9eaSAaron Smith   if (!data_sp) {
3885146a9eaSAaron Smith     error.SetErrorStringWithFormat(
3895146a9eaSAaron Smith         "NativeRegisterContextWindows_WoW64::%s invalid data_sp provided",
3905146a9eaSAaron Smith         __FUNCTION__);
3915146a9eaSAaron Smith     return error;
3925146a9eaSAaron Smith   }
3935146a9eaSAaron Smith 
3945146a9eaSAaron Smith   if (data_sp->GetByteSize() != data_size) {
3955146a9eaSAaron Smith     error.SetErrorStringWithFormatv(
3965146a9eaSAaron Smith         "data_sp contained mismatched data size, expected {0}, actual {1}",
3975146a9eaSAaron Smith         data_size, data_sp->GetByteSize());
3985146a9eaSAaron Smith     return error;
3995146a9eaSAaron Smith   }
4005146a9eaSAaron Smith 
4015146a9eaSAaron Smith   ::WOW64_CONTEXT tls_context;
4025146a9eaSAaron Smith   memcpy(&tls_context, data_sp->GetBytes(), data_size);
4035146a9eaSAaron Smith   return SetWoW64ThreadContextHelper(GetThreadHandle(), &tls_context);
4045146a9eaSAaron Smith }
4055146a9eaSAaron Smith 
IsWatchpointHit(uint32_t wp_index,bool & is_hit)4065146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::IsWatchpointHit(uint32_t wp_index,
4075146a9eaSAaron Smith                                                            bool &is_hit) {
4085b5274eaSAleksandr Urakov   is_hit = false;
4095b5274eaSAleksandr Urakov 
4105b5274eaSAleksandr Urakov   if (wp_index >= NumSupportedHardwareWatchpoints())
4115b5274eaSAleksandr Urakov     return Status("watchpoint index out of range");
4125b5274eaSAleksandr Urakov 
4135b5274eaSAleksandr Urakov   RegisterValue reg_value;
4145b5274eaSAleksandr Urakov   Status error = DRRead(lldb_dr6_i386, reg_value);
4155b5274eaSAleksandr Urakov   if (error.Fail())
4165b5274eaSAleksandr Urakov     return error;
4175b5274eaSAleksandr Urakov 
4185b5274eaSAleksandr Urakov   is_hit = reg_value.GetAsUInt32() & (1 << wp_index);
4195b5274eaSAleksandr Urakov 
4205b5274eaSAleksandr Urakov   return {};
4215146a9eaSAaron Smith }
4225146a9eaSAaron Smith 
GetWatchpointHitIndex(uint32_t & wp_index,lldb::addr_t trap_addr)4235146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::GetWatchpointHitIndex(
4245146a9eaSAaron Smith     uint32_t &wp_index, lldb::addr_t trap_addr) {
4255b5274eaSAleksandr Urakov   wp_index = LLDB_INVALID_INDEX32;
4265b5274eaSAleksandr Urakov 
4275b5274eaSAleksandr Urakov   for (uint32_t i = 0; i < NumSupportedHardwareWatchpoints(); i++) {
4285b5274eaSAleksandr Urakov     bool is_hit;
4295b5274eaSAleksandr Urakov     Status error = IsWatchpointHit(i, is_hit);
4305b5274eaSAleksandr Urakov     if (error.Fail())
4315b5274eaSAleksandr Urakov       return error;
4325b5274eaSAleksandr Urakov 
4335b5274eaSAleksandr Urakov     if (is_hit) {
4345b5274eaSAleksandr Urakov       wp_index = i;
4355b5274eaSAleksandr Urakov       return {};
4365b5274eaSAleksandr Urakov     }
4375b5274eaSAleksandr Urakov   }
4385b5274eaSAleksandr Urakov 
4395b5274eaSAleksandr Urakov   return {};
4405146a9eaSAaron Smith }
4415146a9eaSAaron Smith 
IsWatchpointVacant(uint32_t wp_index,bool & is_vacant)4425146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::IsWatchpointVacant(uint32_t wp_index,
4435146a9eaSAaron Smith                                                               bool &is_vacant) {
4445b5274eaSAleksandr Urakov   is_vacant = false;
4455146a9eaSAaron Smith 
4465b5274eaSAleksandr Urakov   if (wp_index >= NumSupportedHardwareWatchpoints())
4475b5274eaSAleksandr Urakov     return Status("Watchpoint index out of range");
4485b5274eaSAleksandr Urakov 
4495b5274eaSAleksandr Urakov   RegisterValue reg_value;
4505b5274eaSAleksandr Urakov   Status error = DRRead(lldb_dr7_i386, reg_value);
4515b5274eaSAleksandr Urakov   if (error.Fail())
4525b5274eaSAleksandr Urakov     return error;
4535b5274eaSAleksandr Urakov 
4545b5274eaSAleksandr Urakov   is_vacant = !(reg_value.GetAsUInt32() & (1 << (2 * wp_index)));
4555b5274eaSAleksandr Urakov 
4565b5274eaSAleksandr Urakov   return error;
4575146a9eaSAaron Smith }
4585146a9eaSAaron Smith 
ClearHardwareWatchpoint(uint32_t wp_index)4595146a9eaSAaron Smith bool NativeRegisterContextWindows_WoW64::ClearHardwareWatchpoint(
4605146a9eaSAaron Smith     uint32_t wp_index) {
4615b5274eaSAleksandr Urakov   if (wp_index >= NumSupportedHardwareWatchpoints())
4625146a9eaSAaron Smith     return false;
4635b5274eaSAleksandr Urakov 
4645b5274eaSAleksandr Urakov   // for watchpoints 0, 1, 2, or 3, respectively, clear bits 0, 1, 2, or 3 of
4655b5274eaSAleksandr Urakov   // the debug status register (DR6)
4665b5274eaSAleksandr Urakov 
4675b5274eaSAleksandr Urakov   RegisterValue reg_value;
4685b5274eaSAleksandr Urakov   Status error = DRRead(lldb_dr6_i386, reg_value);
4695b5274eaSAleksandr Urakov   if (error.Fail())
4705b5274eaSAleksandr Urakov     return false;
4715b5274eaSAleksandr Urakov 
4725b5274eaSAleksandr Urakov   uint32_t bit_mask = 1 << wp_index;
4735b5274eaSAleksandr Urakov   uint32_t status_bits = reg_value.GetAsUInt32() & ~bit_mask;
4745b5274eaSAleksandr Urakov   error = DRWrite(lldb_dr6_i386, RegisterValue(status_bits));
4755b5274eaSAleksandr Urakov   if (error.Fail())
4765b5274eaSAleksandr Urakov     return false;
4775b5274eaSAleksandr Urakov 
4785b5274eaSAleksandr Urakov   // for watchpoints 0, 1, 2, or 3, respectively, clear bits {0-1,16-19},
4795b5274eaSAleksandr Urakov   // {2-3,20-23}, {4-5,24-27}, or {6-7,28-31} of the debug control register
4805b5274eaSAleksandr Urakov   // (DR7)
4815b5274eaSAleksandr Urakov 
4825b5274eaSAleksandr Urakov   error = DRRead(lldb_dr7_i386, reg_value);
4835b5274eaSAleksandr Urakov   if (error.Fail())
4845b5274eaSAleksandr Urakov     return false;
4855b5274eaSAleksandr Urakov 
4865b5274eaSAleksandr Urakov   bit_mask = (0x3 << (2 * wp_index)) | (0xF << (16 + 4 * wp_index));
4875b5274eaSAleksandr Urakov   uint32_t control_bits = reg_value.GetAsUInt32() & ~bit_mask;
4885b5274eaSAleksandr Urakov   return DRWrite(lldb_dr7_i386, RegisterValue(control_bits)).Success();
4895146a9eaSAaron Smith }
4905146a9eaSAaron Smith 
ClearAllHardwareWatchpoints()4915146a9eaSAaron Smith Status NativeRegisterContextWindows_WoW64::ClearAllHardwareWatchpoints() {
4925b5274eaSAleksandr Urakov   RegisterValue reg_value;
4935b5274eaSAleksandr Urakov 
4945b5274eaSAleksandr Urakov   // clear bits {0-4} of the debug status register (DR6)
4955b5274eaSAleksandr Urakov 
4965b5274eaSAleksandr Urakov   Status error = DRRead(lldb_dr6_i386, reg_value);
4975b5274eaSAleksandr Urakov   if (error.Fail())
4985b5274eaSAleksandr Urakov     return error;
4995b5274eaSAleksandr Urakov 
5005b5274eaSAleksandr Urakov   uint32_t status_bits = reg_value.GetAsUInt32() & ~0xF;
5015b5274eaSAleksandr Urakov   error = DRWrite(lldb_dr6_i386, RegisterValue(status_bits));
5025b5274eaSAleksandr Urakov   if (error.Fail())
5035b5274eaSAleksandr Urakov     return error;
5045b5274eaSAleksandr Urakov 
5055b5274eaSAleksandr Urakov   // clear bits {0-7,16-31} of the debug control register (DR7)
5065b5274eaSAleksandr Urakov 
5075b5274eaSAleksandr Urakov   error = DRRead(lldb_dr7_i386, reg_value);
5085b5274eaSAleksandr Urakov   if (error.Fail())
5095b5274eaSAleksandr Urakov     return error;
5105b5274eaSAleksandr Urakov 
5115b5274eaSAleksandr Urakov   uint32_t control_bits = reg_value.GetAsUInt32() & ~0xFFFF00FF;
5125b5274eaSAleksandr Urakov   return DRWrite(lldb_dr7_i386, RegisterValue(control_bits));
5135146a9eaSAaron Smith }
5145146a9eaSAaron Smith 
SetHardwareWatchpoint(lldb::addr_t addr,size_t size,uint32_t watch_flags)5155146a9eaSAaron Smith uint32_t NativeRegisterContextWindows_WoW64::SetHardwareWatchpoint(
5165146a9eaSAaron Smith     lldb::addr_t addr, size_t size, uint32_t watch_flags) {
5175b5274eaSAleksandr Urakov   switch (size) {
5185b5274eaSAleksandr Urakov   case 1:
5195b5274eaSAleksandr Urakov   case 2:
5205b5274eaSAleksandr Urakov   case 4:
5215b5274eaSAleksandr Urakov     break;
5225b5274eaSAleksandr Urakov   default:
5235146a9eaSAaron Smith     return LLDB_INVALID_INDEX32;
5245146a9eaSAaron Smith   }
5255146a9eaSAaron Smith 
5265b5274eaSAleksandr Urakov   if (watch_flags == 0x2)
5275b5274eaSAleksandr Urakov     watch_flags = 0x3;
5285b5274eaSAleksandr Urakov 
5295b5274eaSAleksandr Urakov   if (watch_flags != 0x1 && watch_flags != 0x3)
5305b5274eaSAleksandr Urakov     return LLDB_INVALID_INDEX32;
5315b5274eaSAleksandr Urakov 
5325b5274eaSAleksandr Urakov   for (uint32_t wp_index = 0; wp_index < NumSupportedHardwareWatchpoints();
5335b5274eaSAleksandr Urakov        ++wp_index) {
5345b5274eaSAleksandr Urakov     bool is_vacant;
5355b5274eaSAleksandr Urakov     if (IsWatchpointVacant(wp_index, is_vacant).Fail())
5365b5274eaSAleksandr Urakov       return LLDB_INVALID_INDEX32;
5375b5274eaSAleksandr Urakov 
5385b5274eaSAleksandr Urakov     if (is_vacant) {
5395b5274eaSAleksandr Urakov       if (!ClearHardwareWatchpoint(wp_index))
5405b5274eaSAleksandr Urakov         return LLDB_INVALID_INDEX32;
5415b5274eaSAleksandr Urakov 
5425b5274eaSAleksandr Urakov       if (ApplyHardwareBreakpoint(wp_index, addr, size, watch_flags).Fail())
5435b5274eaSAleksandr Urakov         return LLDB_INVALID_INDEX32;
5445b5274eaSAleksandr Urakov 
5455b5274eaSAleksandr Urakov       return wp_index;
5465b5274eaSAleksandr Urakov     }
5475b5274eaSAleksandr Urakov   }
5485b5274eaSAleksandr Urakov   return LLDB_INVALID_INDEX32;
5495b5274eaSAleksandr Urakov }
5505b5274eaSAleksandr Urakov 
ApplyHardwareBreakpoint(uint32_t wp_index,lldb::addr_t addr,size_t size,uint32_t flags)5515b5274eaSAleksandr Urakov Status NativeRegisterContextWindows_WoW64::ApplyHardwareBreakpoint(
5525b5274eaSAleksandr Urakov     uint32_t wp_index, lldb::addr_t addr, size_t size, uint32_t flags) {
5535b5274eaSAleksandr Urakov   RegisterValue reg_value;
5545b5274eaSAleksandr Urakov   auto error = DRRead(lldb_dr7_i386, reg_value);
5555b5274eaSAleksandr Urakov   if (error.Fail())
5565b5274eaSAleksandr Urakov     return error;
5575b5274eaSAleksandr Urakov 
5585b5274eaSAleksandr Urakov   // for watchpoints 0, 1, 2, or 3, respectively, set bits 1, 3, 5, or 7
5595b5274eaSAleksandr Urakov   uint32_t enable_bit = 1 << (2 * wp_index);
5605b5274eaSAleksandr Urakov 
5615b5274eaSAleksandr Urakov   // set bits 16-17, 20-21, 24-25, or 28-29
5625b5274eaSAleksandr Urakov   // with 0b01 for write, and 0b11 for read/write
5635b5274eaSAleksandr Urakov   uint32_t rw_bits = flags << (16 + 4 * wp_index);
5645b5274eaSAleksandr Urakov 
5655b5274eaSAleksandr Urakov   // set bits 18-19, 22-23, 26-27, or 30-31
5665b5274eaSAleksandr Urakov   // with 0b00, 0b01, 0b10, or 0b11
5675b5274eaSAleksandr Urakov   // for 1, 2, 8 (if supported), or 4 bytes, respectively
5685b5274eaSAleksandr Urakov   uint32_t size_bits = (size == 8 ? 0x2 : size - 1) << (18 + 4 * wp_index);
5695b5274eaSAleksandr Urakov 
5705b5274eaSAleksandr Urakov   uint32_t bit_mask = (0x3 << (2 * wp_index)) | (0xF << (16 + 4 * wp_index));
5715b5274eaSAleksandr Urakov 
5725b5274eaSAleksandr Urakov   uint32_t control_bits = reg_value.GetAsUInt32() & ~bit_mask;
5735b5274eaSAleksandr Urakov   control_bits |= enable_bit | rw_bits | size_bits;
5745b5274eaSAleksandr Urakov 
5755b5274eaSAleksandr Urakov   error = DRWrite(lldb_dr7_i386, RegisterValue(control_bits));
5765b5274eaSAleksandr Urakov   if (error.Fail())
5775b5274eaSAleksandr Urakov     return error;
5785b5274eaSAleksandr Urakov 
5795b5274eaSAleksandr Urakov   error = DRWrite(lldb_dr0_i386 + wp_index, RegisterValue(addr));
5805b5274eaSAleksandr Urakov   if (error.Fail())
5815b5274eaSAleksandr Urakov     return error;
5825b5274eaSAleksandr Urakov 
5835b5274eaSAleksandr Urakov   return {};
5845b5274eaSAleksandr Urakov }
5855b5274eaSAleksandr Urakov 
5865146a9eaSAaron Smith lldb::addr_t
GetWatchpointAddress(uint32_t wp_index)5875146a9eaSAaron Smith NativeRegisterContextWindows_WoW64::GetWatchpointAddress(uint32_t wp_index) {
5885b5274eaSAleksandr Urakov   if (wp_index >= NumSupportedHardwareWatchpoints())
5895146a9eaSAaron Smith     return LLDB_INVALID_ADDRESS;
5905b5274eaSAleksandr Urakov 
5915b5274eaSAleksandr Urakov   RegisterValue reg_value;
5925b5274eaSAleksandr Urakov   if (DRRead(lldb_dr0_i386 + wp_index, reg_value).Fail())
5935b5274eaSAleksandr Urakov     return LLDB_INVALID_ADDRESS;
5945b5274eaSAleksandr Urakov 
5955b5274eaSAleksandr Urakov   return reg_value.GetAsUInt32();
5965146a9eaSAaron Smith }
5975146a9eaSAaron Smith 
NumSupportedHardwareWatchpoints()5985146a9eaSAaron Smith uint32_t NativeRegisterContextWindows_WoW64::NumSupportedHardwareWatchpoints() {
5995b5274eaSAleksandr Urakov   return 4;
6005146a9eaSAaron Smith }
6015146a9eaSAaron Smith 
602900f9ba2STatyana Krasnukha #endif // defined(__x86_64__) || defined(_M_X64)
603