180814287SRaphael Isemann //===-- NativeRegisterContextWindows_arm64.cpp ----------------------------===//
2b1f6ba2aSMartin Storsjo //
3b1f6ba2aSMartin Storsjo // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4b1f6ba2aSMartin Storsjo // See https://llvm.org/LICENSE.txt for license information.
5b1f6ba2aSMartin Storsjo // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6b1f6ba2aSMartin Storsjo //
7b1f6ba2aSMartin Storsjo //===----------------------------------------------------------------------===//
8b1f6ba2aSMartin Storsjo
9b1f6ba2aSMartin Storsjo #if defined(__aarch64__) || defined(_M_ARM64)
10b1f6ba2aSMartin Storsjo
11b1f6ba2aSMartin Storsjo #include "NativeRegisterContextWindows_arm64.h"
12b1f6ba2aSMartin Storsjo #include "NativeThreadWindows.h"
13b1f6ba2aSMartin Storsjo #include "Plugins/Process/Utility/RegisterInfoPOSIX_arm64.h"
14b1f6ba2aSMartin Storsjo #include "ProcessWindowsLog.h"
15b1f6ba2aSMartin Storsjo #include "lldb/Host/HostInfo.h"
16b1f6ba2aSMartin Storsjo #include "lldb/Host/HostThread.h"
17b1f6ba2aSMartin Storsjo #include "lldb/Host/windows/HostThreadWindows.h"
18b1f6ba2aSMartin Storsjo #include "lldb/Host/windows/windows.h"
19b1f6ba2aSMartin Storsjo
20b1f6ba2aSMartin Storsjo #include "lldb/Utility/Log.h"
21b1f6ba2aSMartin Storsjo #include "lldb/Utility/RegisterValue.h"
22b1f6ba2aSMartin Storsjo #include "llvm/ADT/STLExtras.h"
23b1f6ba2aSMartin Storsjo
24b1f6ba2aSMartin Storsjo using namespace lldb;
25b1f6ba2aSMartin Storsjo using namespace lldb_private;
26b1f6ba2aSMartin Storsjo
27b1f6ba2aSMartin Storsjo #define REG_CONTEXT_SIZE sizeof(::CONTEXT)
28b1f6ba2aSMartin Storsjo
29b1f6ba2aSMartin Storsjo namespace {
30b1f6ba2aSMartin Storsjo static const uint32_t g_gpr_regnums_arm64[] = {
31b1f6ba2aSMartin Storsjo gpr_x0_arm64, gpr_x1_arm64, gpr_x2_arm64, gpr_x3_arm64,
32b1f6ba2aSMartin Storsjo gpr_x4_arm64, gpr_x5_arm64, gpr_x6_arm64, gpr_x7_arm64,
33b1f6ba2aSMartin Storsjo gpr_x8_arm64, gpr_x9_arm64, gpr_x10_arm64, gpr_x11_arm64,
34b1f6ba2aSMartin Storsjo gpr_x12_arm64, gpr_x13_arm64, gpr_x14_arm64, gpr_x15_arm64,
35b1f6ba2aSMartin Storsjo gpr_x16_arm64, gpr_x17_arm64, gpr_x18_arm64, gpr_x19_arm64,
36b1f6ba2aSMartin Storsjo gpr_x20_arm64, gpr_x21_arm64, gpr_x22_arm64, gpr_x23_arm64,
37b1f6ba2aSMartin Storsjo gpr_x24_arm64, gpr_x25_arm64, gpr_x26_arm64, gpr_x27_arm64,
38b1f6ba2aSMartin Storsjo gpr_x28_arm64, gpr_fp_arm64, gpr_lr_arm64, gpr_sp_arm64,
39b1f6ba2aSMartin Storsjo gpr_pc_arm64, gpr_cpsr_arm64, gpr_w0_arm64, gpr_w1_arm64,
40b1f6ba2aSMartin Storsjo gpr_w2_arm64, gpr_w3_arm64, gpr_w4_arm64, gpr_w5_arm64,
41b1f6ba2aSMartin Storsjo gpr_w6_arm64, gpr_w7_arm64, gpr_w8_arm64, gpr_w9_arm64,
42b1f6ba2aSMartin Storsjo gpr_w10_arm64, gpr_w11_arm64, gpr_w12_arm64, gpr_w13_arm64,
43b1f6ba2aSMartin Storsjo gpr_w14_arm64, gpr_w15_arm64, gpr_w16_arm64, gpr_w17_arm64,
44b1f6ba2aSMartin Storsjo gpr_w18_arm64, gpr_w19_arm64, gpr_w20_arm64, gpr_w21_arm64,
45b1f6ba2aSMartin Storsjo gpr_w22_arm64, gpr_w23_arm64, gpr_w24_arm64, gpr_w25_arm64,
46b1f6ba2aSMartin Storsjo gpr_w26_arm64, gpr_w27_arm64, gpr_w28_arm64,
47b1f6ba2aSMartin Storsjo LLDB_INVALID_REGNUM // Register set must be terminated with this flag
48b1f6ba2aSMartin Storsjo };
49b1f6ba2aSMartin Storsjo static_assert(((sizeof g_gpr_regnums_arm64 / sizeof g_gpr_regnums_arm64[0]) -
50b1f6ba2aSMartin Storsjo 1) == k_num_gpr_registers_arm64,
51b1f6ba2aSMartin Storsjo "g_gpr_regnums_arm64 has wrong number of register infos");
52b1f6ba2aSMartin Storsjo
53b1f6ba2aSMartin Storsjo static const uint32_t g_fpr_regnums_arm64[] = {
54b1f6ba2aSMartin Storsjo fpu_v0_arm64, fpu_v1_arm64, fpu_v2_arm64, fpu_v3_arm64,
55b1f6ba2aSMartin Storsjo fpu_v4_arm64, fpu_v5_arm64, fpu_v6_arm64, fpu_v7_arm64,
56b1f6ba2aSMartin Storsjo fpu_v8_arm64, fpu_v9_arm64, fpu_v10_arm64, fpu_v11_arm64,
57b1f6ba2aSMartin Storsjo fpu_v12_arm64, fpu_v13_arm64, fpu_v14_arm64, fpu_v15_arm64,
58b1f6ba2aSMartin Storsjo fpu_v16_arm64, fpu_v17_arm64, fpu_v18_arm64, fpu_v19_arm64,
59b1f6ba2aSMartin Storsjo fpu_v20_arm64, fpu_v21_arm64, fpu_v22_arm64, fpu_v23_arm64,
60b1f6ba2aSMartin Storsjo fpu_v24_arm64, fpu_v25_arm64, fpu_v26_arm64, fpu_v27_arm64,
61b1f6ba2aSMartin Storsjo fpu_v28_arm64, fpu_v29_arm64, fpu_v30_arm64, fpu_v31_arm64,
62b1f6ba2aSMartin Storsjo fpu_s0_arm64, fpu_s1_arm64, fpu_s2_arm64, fpu_s3_arm64,
63b1f6ba2aSMartin Storsjo fpu_s4_arm64, fpu_s5_arm64, fpu_s6_arm64, fpu_s7_arm64,
64b1f6ba2aSMartin Storsjo fpu_s8_arm64, fpu_s9_arm64, fpu_s10_arm64, fpu_s11_arm64,
65b1f6ba2aSMartin Storsjo fpu_s12_arm64, fpu_s13_arm64, fpu_s14_arm64, fpu_s15_arm64,
66b1f6ba2aSMartin Storsjo fpu_s16_arm64, fpu_s17_arm64, fpu_s18_arm64, fpu_s19_arm64,
67b1f6ba2aSMartin Storsjo fpu_s20_arm64, fpu_s21_arm64, fpu_s22_arm64, fpu_s23_arm64,
68b1f6ba2aSMartin Storsjo fpu_s24_arm64, fpu_s25_arm64, fpu_s26_arm64, fpu_s27_arm64,
69b1f6ba2aSMartin Storsjo fpu_s28_arm64, fpu_s29_arm64, fpu_s30_arm64, fpu_s31_arm64,
70b1f6ba2aSMartin Storsjo
71b1f6ba2aSMartin Storsjo fpu_d0_arm64, fpu_d1_arm64, fpu_d2_arm64, fpu_d3_arm64,
72b1f6ba2aSMartin Storsjo fpu_d4_arm64, fpu_d5_arm64, fpu_d6_arm64, fpu_d7_arm64,
73b1f6ba2aSMartin Storsjo fpu_d8_arm64, fpu_d9_arm64, fpu_d10_arm64, fpu_d11_arm64,
74b1f6ba2aSMartin Storsjo fpu_d12_arm64, fpu_d13_arm64, fpu_d14_arm64, fpu_d15_arm64,
75b1f6ba2aSMartin Storsjo fpu_d16_arm64, fpu_d17_arm64, fpu_d18_arm64, fpu_d19_arm64,
76b1f6ba2aSMartin Storsjo fpu_d20_arm64, fpu_d21_arm64, fpu_d22_arm64, fpu_d23_arm64,
77b1f6ba2aSMartin Storsjo fpu_d24_arm64, fpu_d25_arm64, fpu_d26_arm64, fpu_d27_arm64,
78b1f6ba2aSMartin Storsjo fpu_d28_arm64, fpu_d29_arm64, fpu_d30_arm64, fpu_d31_arm64,
79b1f6ba2aSMartin Storsjo fpu_fpsr_arm64, fpu_fpcr_arm64,
80b1f6ba2aSMartin Storsjo LLDB_INVALID_REGNUM // Register set must be terminated with this flag
81b1f6ba2aSMartin Storsjo };
82b1f6ba2aSMartin Storsjo static_assert(((sizeof g_fpr_regnums_arm64 / sizeof g_fpr_regnums_arm64[0]) -
83b1f6ba2aSMartin Storsjo 1) == k_num_fpr_registers_arm64,
84b1f6ba2aSMartin Storsjo "g_fpu_regnums_arm64 has wrong number of register infos");
85b1f6ba2aSMartin Storsjo
86b1f6ba2aSMartin Storsjo static const RegisterSet g_reg_sets_arm64[] = {
87b1f6ba2aSMartin Storsjo {"General Purpose Registers", "gpr",
88f15014ffSBenjamin Kramer llvm::array_lengthof(g_gpr_regnums_arm64) - 1, g_gpr_regnums_arm64},
89b1f6ba2aSMartin Storsjo {"Floating Point Registers", "fpr",
90f15014ffSBenjamin Kramer llvm::array_lengthof(g_fpr_regnums_arm64) - 1, g_fpr_regnums_arm64},
91b1f6ba2aSMartin Storsjo };
92b1f6ba2aSMartin Storsjo
93b1f6ba2aSMartin Storsjo enum { k_num_register_sets = 2 };
94b1f6ba2aSMartin Storsjo
95b1f6ba2aSMartin Storsjo } // namespace
96b1f6ba2aSMartin Storsjo
97b1f6ba2aSMartin Storsjo static RegisterInfoInterface *
CreateRegisterInfoInterface(const ArchSpec & target_arch)98b1f6ba2aSMartin Storsjo CreateRegisterInfoInterface(const ArchSpec &target_arch) {
99b1f6ba2aSMartin Storsjo assert((HostInfo::GetArchitecture().GetAddressByteSize() == 8) &&
100b1f6ba2aSMartin Storsjo "Register setting path assumes this is a 64-bit host");
101476e0879SMartin Storsjö return new RegisterInfoPOSIX_arm64(
102476e0879SMartin Storsjö target_arch, RegisterInfoPOSIX_arm64::eRegsetMaskDefault);
103b1f6ba2aSMartin Storsjo }
104b1f6ba2aSMartin Storsjo
GetThreadContextHelper(lldb::thread_t thread_handle,PCONTEXT context_ptr,const DWORD control_flag)105b1f6ba2aSMartin Storsjo static Status GetThreadContextHelper(lldb::thread_t thread_handle,
106b1f6ba2aSMartin Storsjo PCONTEXT context_ptr,
107b1f6ba2aSMartin Storsjo const DWORD control_flag) {
1086730df47SPavel Labath Log *log = GetLog(WindowsLog::Registers);
109b1f6ba2aSMartin Storsjo Status error;
110b1f6ba2aSMartin Storsjo
111b1f6ba2aSMartin Storsjo memset(context_ptr, 0, sizeof(::CONTEXT));
112b1f6ba2aSMartin Storsjo context_ptr->ContextFlags = control_flag;
113b1f6ba2aSMartin Storsjo if (!::GetThreadContext(thread_handle, context_ptr)) {
114b1f6ba2aSMartin Storsjo error.SetError(GetLastError(), eErrorTypeWin32);
115b1f6ba2aSMartin Storsjo LLDB_LOG(log, "{0} GetThreadContext failed with error {1}", __FUNCTION__,
116b1f6ba2aSMartin Storsjo error);
117b1f6ba2aSMartin Storsjo return error;
118b1f6ba2aSMartin Storsjo }
119b1f6ba2aSMartin Storsjo return Status();
120b1f6ba2aSMartin Storsjo }
121b1f6ba2aSMartin Storsjo
SetThreadContextHelper(lldb::thread_t thread_handle,PCONTEXT context_ptr)122b1f6ba2aSMartin Storsjo static Status SetThreadContextHelper(lldb::thread_t thread_handle,
123b1f6ba2aSMartin Storsjo PCONTEXT context_ptr) {
1246730df47SPavel Labath Log *log = GetLog(WindowsLog::Registers);
125b1f6ba2aSMartin Storsjo Status error;
126b1f6ba2aSMartin Storsjo // It's assumed that the thread has stopped.
127b1f6ba2aSMartin Storsjo if (!::SetThreadContext(thread_handle, context_ptr)) {
128b1f6ba2aSMartin Storsjo error.SetError(GetLastError(), eErrorTypeWin32);
129b1f6ba2aSMartin Storsjo LLDB_LOG(log, "{0} SetThreadContext failed with error {1}", __FUNCTION__,
130b1f6ba2aSMartin Storsjo error);
131b1f6ba2aSMartin Storsjo return error;
132b1f6ba2aSMartin Storsjo }
133b1f6ba2aSMartin Storsjo return Status();
134b1f6ba2aSMartin Storsjo }
135b1f6ba2aSMartin Storsjo
136b1f6ba2aSMartin Storsjo std::unique_ptr<NativeRegisterContextWindows>
CreateHostNativeRegisterContextWindows(const ArchSpec & target_arch,NativeThreadProtocol & native_thread)137b1f6ba2aSMartin Storsjo NativeRegisterContextWindows::CreateHostNativeRegisterContextWindows(
138b1f6ba2aSMartin Storsjo const ArchSpec &target_arch, NativeThreadProtocol &native_thread) {
139b1f6ba2aSMartin Storsjo // Register context for a native 64-bit application.
140b1f6ba2aSMartin Storsjo return std::make_unique<NativeRegisterContextWindows_arm64>(target_arch,
141b1f6ba2aSMartin Storsjo native_thread);
142b1f6ba2aSMartin Storsjo }
143b1f6ba2aSMartin Storsjo
NativeRegisterContextWindows_arm64(const ArchSpec & target_arch,NativeThreadProtocol & native_thread)144b1f6ba2aSMartin Storsjo NativeRegisterContextWindows_arm64::NativeRegisterContextWindows_arm64(
145b1f6ba2aSMartin Storsjo const ArchSpec &target_arch, NativeThreadProtocol &native_thread)
146b1f6ba2aSMartin Storsjo : NativeRegisterContextWindows(native_thread,
147b1f6ba2aSMartin Storsjo CreateRegisterInfoInterface(target_arch)) {}
148b1f6ba2aSMartin Storsjo
IsGPR(uint32_t reg_index) const149b1f6ba2aSMartin Storsjo bool NativeRegisterContextWindows_arm64::IsGPR(uint32_t reg_index) const {
150b1f6ba2aSMartin Storsjo return (reg_index >= k_first_gpr_arm64 && reg_index <= k_last_gpr_arm64);
151b1f6ba2aSMartin Storsjo }
152b1f6ba2aSMartin Storsjo
IsFPR(uint32_t reg_index) const153b1f6ba2aSMartin Storsjo bool NativeRegisterContextWindows_arm64::IsFPR(uint32_t reg_index) const {
154b1f6ba2aSMartin Storsjo return (reg_index >= k_first_fpr_arm64 && reg_index <= k_last_fpr_arm64);
155b1f6ba2aSMartin Storsjo }
156b1f6ba2aSMartin Storsjo
GetRegisterSetCount() const157b1f6ba2aSMartin Storsjo uint32_t NativeRegisterContextWindows_arm64::GetRegisterSetCount() const {
158b1f6ba2aSMartin Storsjo return k_num_register_sets;
159b1f6ba2aSMartin Storsjo }
160b1f6ba2aSMartin Storsjo
161b1f6ba2aSMartin Storsjo const RegisterSet *
GetRegisterSet(uint32_t set_index) const162b1f6ba2aSMartin Storsjo NativeRegisterContextWindows_arm64::GetRegisterSet(uint32_t set_index) const {
163b1f6ba2aSMartin Storsjo if (set_index >= k_num_register_sets)
164b1f6ba2aSMartin Storsjo return nullptr;
165b1f6ba2aSMartin Storsjo return &g_reg_sets_arm64[set_index];
166b1f6ba2aSMartin Storsjo }
167b1f6ba2aSMartin Storsjo
GPRRead(const uint32_t reg,RegisterValue & reg_value)168b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::GPRRead(const uint32_t reg,
169b1f6ba2aSMartin Storsjo RegisterValue ®_value) {
170b1f6ba2aSMartin Storsjo ::CONTEXT tls_context;
171b1f6ba2aSMartin Storsjo DWORD context_flag = CONTEXT_CONTROL | CONTEXT_INTEGER;
172b1f6ba2aSMartin Storsjo Status error =
173b1f6ba2aSMartin Storsjo GetThreadContextHelper(GetThreadHandle(), &tls_context, context_flag);
174b1f6ba2aSMartin Storsjo if (error.Fail())
175b1f6ba2aSMartin Storsjo return error;
176b1f6ba2aSMartin Storsjo
177b1f6ba2aSMartin Storsjo switch (reg) {
178b1f6ba2aSMartin Storsjo case gpr_x0_arm64:
179b1f6ba2aSMartin Storsjo case gpr_x1_arm64:
180b1f6ba2aSMartin Storsjo case gpr_x2_arm64:
181b1f6ba2aSMartin Storsjo case gpr_x3_arm64:
182b1f6ba2aSMartin Storsjo case gpr_x4_arm64:
183b1f6ba2aSMartin Storsjo case gpr_x5_arm64:
184b1f6ba2aSMartin Storsjo case gpr_x6_arm64:
185b1f6ba2aSMartin Storsjo case gpr_x7_arm64:
186b1f6ba2aSMartin Storsjo case gpr_x8_arm64:
187b1f6ba2aSMartin Storsjo case gpr_x9_arm64:
188b1f6ba2aSMartin Storsjo case gpr_x10_arm64:
189b1f6ba2aSMartin Storsjo case gpr_x11_arm64:
190b1f6ba2aSMartin Storsjo case gpr_x12_arm64:
191b1f6ba2aSMartin Storsjo case gpr_x13_arm64:
192b1f6ba2aSMartin Storsjo case gpr_x14_arm64:
193b1f6ba2aSMartin Storsjo case gpr_x15_arm64:
194b1f6ba2aSMartin Storsjo case gpr_x16_arm64:
195b1f6ba2aSMartin Storsjo case gpr_x17_arm64:
196b1f6ba2aSMartin Storsjo case gpr_x18_arm64:
197b1f6ba2aSMartin Storsjo case gpr_x19_arm64:
198b1f6ba2aSMartin Storsjo case gpr_x20_arm64:
199b1f6ba2aSMartin Storsjo case gpr_x21_arm64:
200b1f6ba2aSMartin Storsjo case gpr_x22_arm64:
201b1f6ba2aSMartin Storsjo case gpr_x23_arm64:
202b1f6ba2aSMartin Storsjo case gpr_x24_arm64:
203b1f6ba2aSMartin Storsjo case gpr_x25_arm64:
204b1f6ba2aSMartin Storsjo case gpr_x26_arm64:
205b1f6ba2aSMartin Storsjo case gpr_x27_arm64:
206b1f6ba2aSMartin Storsjo case gpr_x28_arm64:
207b1f6ba2aSMartin Storsjo reg_value.SetUInt64(tls_context.X[reg - gpr_x0_arm64]);
208b1f6ba2aSMartin Storsjo break;
209b1f6ba2aSMartin Storsjo
210b1f6ba2aSMartin Storsjo case gpr_fp_arm64:
211b1f6ba2aSMartin Storsjo reg_value.SetUInt64(tls_context.Fp);
212b1f6ba2aSMartin Storsjo break;
213b1f6ba2aSMartin Storsjo case gpr_sp_arm64:
214b1f6ba2aSMartin Storsjo reg_value.SetUInt64(tls_context.Sp);
215b1f6ba2aSMartin Storsjo break;
216b1f6ba2aSMartin Storsjo case gpr_lr_arm64:
217b1f6ba2aSMartin Storsjo reg_value.SetUInt64(tls_context.Lr);
218b1f6ba2aSMartin Storsjo break;
219b1f6ba2aSMartin Storsjo case gpr_pc_arm64:
220b1f6ba2aSMartin Storsjo reg_value.SetUInt64(tls_context.Pc);
221b1f6ba2aSMartin Storsjo break;
222b1f6ba2aSMartin Storsjo case gpr_cpsr_arm64:
223b595137fSMuhammad Omair Javaid reg_value.SetUInt32(tls_context.Cpsr);
224b1f6ba2aSMartin Storsjo break;
225b1f6ba2aSMartin Storsjo
226b1f6ba2aSMartin Storsjo case gpr_w0_arm64:
227b1f6ba2aSMartin Storsjo case gpr_w1_arm64:
228b1f6ba2aSMartin Storsjo case gpr_w2_arm64:
229b1f6ba2aSMartin Storsjo case gpr_w3_arm64:
230b1f6ba2aSMartin Storsjo case gpr_w4_arm64:
231b1f6ba2aSMartin Storsjo case gpr_w5_arm64:
232b1f6ba2aSMartin Storsjo case gpr_w6_arm64:
233b1f6ba2aSMartin Storsjo case gpr_w7_arm64:
234b1f6ba2aSMartin Storsjo case gpr_w8_arm64:
235b1f6ba2aSMartin Storsjo case gpr_w9_arm64:
236b1f6ba2aSMartin Storsjo case gpr_w10_arm64:
237b1f6ba2aSMartin Storsjo case gpr_w11_arm64:
238b1f6ba2aSMartin Storsjo case gpr_w12_arm64:
239b1f6ba2aSMartin Storsjo case gpr_w13_arm64:
240b1f6ba2aSMartin Storsjo case gpr_w14_arm64:
241b1f6ba2aSMartin Storsjo case gpr_w15_arm64:
242b1f6ba2aSMartin Storsjo case gpr_w16_arm64:
243b1f6ba2aSMartin Storsjo case gpr_w17_arm64:
244b1f6ba2aSMartin Storsjo case gpr_w18_arm64:
245b1f6ba2aSMartin Storsjo case gpr_w19_arm64:
246b1f6ba2aSMartin Storsjo case gpr_w20_arm64:
247b1f6ba2aSMartin Storsjo case gpr_w21_arm64:
248b1f6ba2aSMartin Storsjo case gpr_w22_arm64:
249b1f6ba2aSMartin Storsjo case gpr_w23_arm64:
250b1f6ba2aSMartin Storsjo case gpr_w24_arm64:
251b1f6ba2aSMartin Storsjo case gpr_w25_arm64:
252b1f6ba2aSMartin Storsjo case gpr_w26_arm64:
253b1f6ba2aSMartin Storsjo case gpr_w27_arm64:
254b1f6ba2aSMartin Storsjo case gpr_w28_arm64:
255b1f6ba2aSMartin Storsjo reg_value.SetUInt32(
256b1f6ba2aSMartin Storsjo static_cast<uint32_t>(tls_context.X[reg - gpr_w0_arm64] & 0xffffffff));
257b1f6ba2aSMartin Storsjo break;
258b1f6ba2aSMartin Storsjo }
259b1f6ba2aSMartin Storsjo
260b1f6ba2aSMartin Storsjo return error;
261b1f6ba2aSMartin Storsjo }
262b1f6ba2aSMartin Storsjo
263b1f6ba2aSMartin Storsjo Status
GPRWrite(const uint32_t reg,const RegisterValue & reg_value)264b1f6ba2aSMartin Storsjo NativeRegisterContextWindows_arm64::GPRWrite(const uint32_t reg,
265b1f6ba2aSMartin Storsjo const RegisterValue ®_value) {
266b1f6ba2aSMartin Storsjo ::CONTEXT tls_context;
267b1f6ba2aSMartin Storsjo DWORD context_flag = CONTEXT_CONTROL | CONTEXT_INTEGER;
268b1f6ba2aSMartin Storsjo auto thread_handle = GetThreadHandle();
269b1f6ba2aSMartin Storsjo Status error =
270b1f6ba2aSMartin Storsjo GetThreadContextHelper(thread_handle, &tls_context, context_flag);
271b1f6ba2aSMartin Storsjo if (error.Fail())
272b1f6ba2aSMartin Storsjo return error;
273b1f6ba2aSMartin Storsjo
274b1f6ba2aSMartin Storsjo switch (reg) {
275b1f6ba2aSMartin Storsjo case gpr_x0_arm64:
276b1f6ba2aSMartin Storsjo case gpr_x1_arm64:
277b1f6ba2aSMartin Storsjo case gpr_x2_arm64:
278b1f6ba2aSMartin Storsjo case gpr_x3_arm64:
279b1f6ba2aSMartin Storsjo case gpr_x4_arm64:
280b1f6ba2aSMartin Storsjo case gpr_x5_arm64:
281b1f6ba2aSMartin Storsjo case gpr_x6_arm64:
282b1f6ba2aSMartin Storsjo case gpr_x7_arm64:
283b1f6ba2aSMartin Storsjo case gpr_x8_arm64:
284b1f6ba2aSMartin Storsjo case gpr_x9_arm64:
285b1f6ba2aSMartin Storsjo case gpr_x10_arm64:
286b1f6ba2aSMartin Storsjo case gpr_x11_arm64:
287b1f6ba2aSMartin Storsjo case gpr_x12_arm64:
288b1f6ba2aSMartin Storsjo case gpr_x13_arm64:
289b1f6ba2aSMartin Storsjo case gpr_x14_arm64:
290b1f6ba2aSMartin Storsjo case gpr_x15_arm64:
291b1f6ba2aSMartin Storsjo case gpr_x16_arm64:
292b1f6ba2aSMartin Storsjo case gpr_x17_arm64:
293b1f6ba2aSMartin Storsjo case gpr_x18_arm64:
294b1f6ba2aSMartin Storsjo case gpr_x19_arm64:
295b1f6ba2aSMartin Storsjo case gpr_x20_arm64:
296b1f6ba2aSMartin Storsjo case gpr_x21_arm64:
297b1f6ba2aSMartin Storsjo case gpr_x22_arm64:
298b1f6ba2aSMartin Storsjo case gpr_x23_arm64:
299b1f6ba2aSMartin Storsjo case gpr_x24_arm64:
300b1f6ba2aSMartin Storsjo case gpr_x25_arm64:
301b1f6ba2aSMartin Storsjo case gpr_x26_arm64:
302b1f6ba2aSMartin Storsjo case gpr_x27_arm64:
303b1f6ba2aSMartin Storsjo case gpr_x28_arm64:
304b1f6ba2aSMartin Storsjo tls_context.X[reg - gpr_x0_arm64] = reg_value.GetAsUInt64();
305b1f6ba2aSMartin Storsjo break;
306b1f6ba2aSMartin Storsjo
307b1f6ba2aSMartin Storsjo case gpr_fp_arm64:
308b1f6ba2aSMartin Storsjo tls_context.Fp = reg_value.GetAsUInt64();
309b1f6ba2aSMartin Storsjo break;
310b1f6ba2aSMartin Storsjo case gpr_sp_arm64:
311b1f6ba2aSMartin Storsjo tls_context.Sp = reg_value.GetAsUInt64();
312b1f6ba2aSMartin Storsjo break;
313b1f6ba2aSMartin Storsjo case gpr_lr_arm64:
314b1f6ba2aSMartin Storsjo tls_context.Lr = reg_value.GetAsUInt64();
315b1f6ba2aSMartin Storsjo break;
316b1f6ba2aSMartin Storsjo case gpr_pc_arm64:
317b1f6ba2aSMartin Storsjo tls_context.Pc = reg_value.GetAsUInt64();
318b1f6ba2aSMartin Storsjo break;
319b1f6ba2aSMartin Storsjo case gpr_cpsr_arm64:
320b595137fSMuhammad Omair Javaid tls_context.Cpsr = reg_value.GetAsUInt32();
321b1f6ba2aSMartin Storsjo break;
322b1f6ba2aSMartin Storsjo
323b1f6ba2aSMartin Storsjo case gpr_w0_arm64:
324b1f6ba2aSMartin Storsjo case gpr_w1_arm64:
325b1f6ba2aSMartin Storsjo case gpr_w2_arm64:
326b1f6ba2aSMartin Storsjo case gpr_w3_arm64:
327b1f6ba2aSMartin Storsjo case gpr_w4_arm64:
328b1f6ba2aSMartin Storsjo case gpr_w5_arm64:
329b1f6ba2aSMartin Storsjo case gpr_w6_arm64:
330b1f6ba2aSMartin Storsjo case gpr_w7_arm64:
331b1f6ba2aSMartin Storsjo case gpr_w8_arm64:
332b1f6ba2aSMartin Storsjo case gpr_w9_arm64:
333b1f6ba2aSMartin Storsjo case gpr_w10_arm64:
334b1f6ba2aSMartin Storsjo case gpr_w11_arm64:
335b1f6ba2aSMartin Storsjo case gpr_w12_arm64:
336b1f6ba2aSMartin Storsjo case gpr_w13_arm64:
337b1f6ba2aSMartin Storsjo case gpr_w14_arm64:
338b1f6ba2aSMartin Storsjo case gpr_w15_arm64:
339b1f6ba2aSMartin Storsjo case gpr_w16_arm64:
340b1f6ba2aSMartin Storsjo case gpr_w17_arm64:
341b1f6ba2aSMartin Storsjo case gpr_w18_arm64:
342b1f6ba2aSMartin Storsjo case gpr_w19_arm64:
343b1f6ba2aSMartin Storsjo case gpr_w20_arm64:
344b1f6ba2aSMartin Storsjo case gpr_w21_arm64:
345b1f6ba2aSMartin Storsjo case gpr_w22_arm64:
346b1f6ba2aSMartin Storsjo case gpr_w23_arm64:
347b1f6ba2aSMartin Storsjo case gpr_w24_arm64:
348b1f6ba2aSMartin Storsjo case gpr_w25_arm64:
349b1f6ba2aSMartin Storsjo case gpr_w26_arm64:
350b1f6ba2aSMartin Storsjo case gpr_w27_arm64:
351b1f6ba2aSMartin Storsjo case gpr_w28_arm64:
352b1f6ba2aSMartin Storsjo tls_context.X[reg - gpr_w0_arm64] = reg_value.GetAsUInt32();
353b1f6ba2aSMartin Storsjo break;
354b1f6ba2aSMartin Storsjo }
355b1f6ba2aSMartin Storsjo
356b1f6ba2aSMartin Storsjo return SetThreadContextHelper(thread_handle, &tls_context);
357b1f6ba2aSMartin Storsjo }
358b1f6ba2aSMartin Storsjo
FPRRead(const uint32_t reg,RegisterValue & reg_value)359b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::FPRRead(const uint32_t reg,
360b1f6ba2aSMartin Storsjo RegisterValue ®_value) {
361b1f6ba2aSMartin Storsjo ::CONTEXT tls_context;
362b1f6ba2aSMartin Storsjo DWORD context_flag = CONTEXT_CONTROL | CONTEXT_FLOATING_POINT;
363b1f6ba2aSMartin Storsjo Status error =
364b1f6ba2aSMartin Storsjo GetThreadContextHelper(GetThreadHandle(), &tls_context, context_flag);
365b1f6ba2aSMartin Storsjo if (error.Fail())
366b1f6ba2aSMartin Storsjo return error;
367b1f6ba2aSMartin Storsjo
368b1f6ba2aSMartin Storsjo switch (reg) {
369b1f6ba2aSMartin Storsjo case fpu_v0_arm64:
370b1f6ba2aSMartin Storsjo case fpu_v1_arm64:
371b1f6ba2aSMartin Storsjo case fpu_v2_arm64:
372b1f6ba2aSMartin Storsjo case fpu_v3_arm64:
373b1f6ba2aSMartin Storsjo case fpu_v4_arm64:
374b1f6ba2aSMartin Storsjo case fpu_v5_arm64:
375b1f6ba2aSMartin Storsjo case fpu_v6_arm64:
376b1f6ba2aSMartin Storsjo case fpu_v7_arm64:
377b1f6ba2aSMartin Storsjo case fpu_v8_arm64:
378b1f6ba2aSMartin Storsjo case fpu_v9_arm64:
379b1f6ba2aSMartin Storsjo case fpu_v10_arm64:
380b1f6ba2aSMartin Storsjo case fpu_v11_arm64:
381b1f6ba2aSMartin Storsjo case fpu_v12_arm64:
382b1f6ba2aSMartin Storsjo case fpu_v13_arm64:
383b1f6ba2aSMartin Storsjo case fpu_v14_arm64:
384b1f6ba2aSMartin Storsjo case fpu_v15_arm64:
385b1f6ba2aSMartin Storsjo case fpu_v16_arm64:
386b1f6ba2aSMartin Storsjo case fpu_v17_arm64:
387b1f6ba2aSMartin Storsjo case fpu_v18_arm64:
388b1f6ba2aSMartin Storsjo case fpu_v19_arm64:
389b1f6ba2aSMartin Storsjo case fpu_v20_arm64:
390b1f6ba2aSMartin Storsjo case fpu_v21_arm64:
391b1f6ba2aSMartin Storsjo case fpu_v22_arm64:
392b1f6ba2aSMartin Storsjo case fpu_v23_arm64:
393b1f6ba2aSMartin Storsjo case fpu_v24_arm64:
394b1f6ba2aSMartin Storsjo case fpu_v25_arm64:
395b1f6ba2aSMartin Storsjo case fpu_v26_arm64:
396b1f6ba2aSMartin Storsjo case fpu_v27_arm64:
397b1f6ba2aSMartin Storsjo case fpu_v28_arm64:
398b1f6ba2aSMartin Storsjo case fpu_v29_arm64:
399b1f6ba2aSMartin Storsjo case fpu_v30_arm64:
400b1f6ba2aSMartin Storsjo case fpu_v31_arm64:
401b1f6ba2aSMartin Storsjo reg_value.SetBytes(tls_context.V[reg - fpu_v0_arm64].B, 16,
402b1f6ba2aSMartin Storsjo endian::InlHostByteOrder());
403b1f6ba2aSMartin Storsjo break;
404b1f6ba2aSMartin Storsjo
405b1f6ba2aSMartin Storsjo case fpu_s0_arm64:
406b1f6ba2aSMartin Storsjo case fpu_s1_arm64:
407b1f6ba2aSMartin Storsjo case fpu_s2_arm64:
408b1f6ba2aSMartin Storsjo case fpu_s3_arm64:
409b1f6ba2aSMartin Storsjo case fpu_s4_arm64:
410b1f6ba2aSMartin Storsjo case fpu_s5_arm64:
411b1f6ba2aSMartin Storsjo case fpu_s6_arm64:
412b1f6ba2aSMartin Storsjo case fpu_s7_arm64:
413b1f6ba2aSMartin Storsjo case fpu_s8_arm64:
414b1f6ba2aSMartin Storsjo case fpu_s9_arm64:
415b1f6ba2aSMartin Storsjo case fpu_s10_arm64:
416b1f6ba2aSMartin Storsjo case fpu_s11_arm64:
417b1f6ba2aSMartin Storsjo case fpu_s12_arm64:
418b1f6ba2aSMartin Storsjo case fpu_s13_arm64:
419b1f6ba2aSMartin Storsjo case fpu_s14_arm64:
420b1f6ba2aSMartin Storsjo case fpu_s15_arm64:
421b1f6ba2aSMartin Storsjo case fpu_s16_arm64:
422b1f6ba2aSMartin Storsjo case fpu_s17_arm64:
423b1f6ba2aSMartin Storsjo case fpu_s18_arm64:
424b1f6ba2aSMartin Storsjo case fpu_s19_arm64:
425b1f6ba2aSMartin Storsjo case fpu_s20_arm64:
426b1f6ba2aSMartin Storsjo case fpu_s21_arm64:
427b1f6ba2aSMartin Storsjo case fpu_s22_arm64:
428b1f6ba2aSMartin Storsjo case fpu_s23_arm64:
429b1f6ba2aSMartin Storsjo case fpu_s24_arm64:
430b1f6ba2aSMartin Storsjo case fpu_s25_arm64:
431b1f6ba2aSMartin Storsjo case fpu_s26_arm64:
432b1f6ba2aSMartin Storsjo case fpu_s27_arm64:
433b1f6ba2aSMartin Storsjo case fpu_s28_arm64:
434b1f6ba2aSMartin Storsjo case fpu_s29_arm64:
435b1f6ba2aSMartin Storsjo case fpu_s30_arm64:
436b1f6ba2aSMartin Storsjo case fpu_s31_arm64:
437b1f6ba2aSMartin Storsjo reg_value.SetFloat(tls_context.V[reg - fpu_s0_arm64].S[0]);
438b1f6ba2aSMartin Storsjo break;
439b1f6ba2aSMartin Storsjo
440b1f6ba2aSMartin Storsjo case fpu_d0_arm64:
441b1f6ba2aSMartin Storsjo case fpu_d1_arm64:
442b1f6ba2aSMartin Storsjo case fpu_d2_arm64:
443b1f6ba2aSMartin Storsjo case fpu_d3_arm64:
444b1f6ba2aSMartin Storsjo case fpu_d4_arm64:
445b1f6ba2aSMartin Storsjo case fpu_d5_arm64:
446b1f6ba2aSMartin Storsjo case fpu_d6_arm64:
447b1f6ba2aSMartin Storsjo case fpu_d7_arm64:
448b1f6ba2aSMartin Storsjo case fpu_d8_arm64:
449b1f6ba2aSMartin Storsjo case fpu_d9_arm64:
450b1f6ba2aSMartin Storsjo case fpu_d10_arm64:
451b1f6ba2aSMartin Storsjo case fpu_d11_arm64:
452b1f6ba2aSMartin Storsjo case fpu_d12_arm64:
453b1f6ba2aSMartin Storsjo case fpu_d13_arm64:
454b1f6ba2aSMartin Storsjo case fpu_d14_arm64:
455b1f6ba2aSMartin Storsjo case fpu_d15_arm64:
456b1f6ba2aSMartin Storsjo case fpu_d16_arm64:
457b1f6ba2aSMartin Storsjo case fpu_d17_arm64:
458b1f6ba2aSMartin Storsjo case fpu_d18_arm64:
459b1f6ba2aSMartin Storsjo case fpu_d19_arm64:
460b1f6ba2aSMartin Storsjo case fpu_d20_arm64:
461b1f6ba2aSMartin Storsjo case fpu_d21_arm64:
462b1f6ba2aSMartin Storsjo case fpu_d22_arm64:
463b1f6ba2aSMartin Storsjo case fpu_d23_arm64:
464b1f6ba2aSMartin Storsjo case fpu_d24_arm64:
465b1f6ba2aSMartin Storsjo case fpu_d25_arm64:
466b1f6ba2aSMartin Storsjo case fpu_d26_arm64:
467b1f6ba2aSMartin Storsjo case fpu_d27_arm64:
468b1f6ba2aSMartin Storsjo case fpu_d28_arm64:
469b1f6ba2aSMartin Storsjo case fpu_d29_arm64:
470b1f6ba2aSMartin Storsjo case fpu_d30_arm64:
471b1f6ba2aSMartin Storsjo case fpu_d31_arm64:
472b1f6ba2aSMartin Storsjo reg_value.SetDouble(tls_context.V[reg - fpu_d0_arm64].D[0]);
473b1f6ba2aSMartin Storsjo break;
474b1f6ba2aSMartin Storsjo
475b1f6ba2aSMartin Storsjo case fpu_fpsr_arm64:
476b1f6ba2aSMartin Storsjo reg_value.SetUInt32(tls_context.Fpsr);
477b1f6ba2aSMartin Storsjo break;
478b1f6ba2aSMartin Storsjo
479b1f6ba2aSMartin Storsjo case fpu_fpcr_arm64:
480b1f6ba2aSMartin Storsjo reg_value.SetUInt32(tls_context.Fpcr);
481b1f6ba2aSMartin Storsjo break;
482b1f6ba2aSMartin Storsjo }
483b1f6ba2aSMartin Storsjo
484b1f6ba2aSMartin Storsjo return error;
485b1f6ba2aSMartin Storsjo }
486b1f6ba2aSMartin Storsjo
487b1f6ba2aSMartin Storsjo Status
FPRWrite(const uint32_t reg,const RegisterValue & reg_value)488b1f6ba2aSMartin Storsjo NativeRegisterContextWindows_arm64::FPRWrite(const uint32_t reg,
489b1f6ba2aSMartin Storsjo const RegisterValue ®_value) {
490b1f6ba2aSMartin Storsjo ::CONTEXT tls_context;
491b1f6ba2aSMartin Storsjo DWORD context_flag = CONTEXT_CONTROL | CONTEXT_FLOATING_POINT;
492b1f6ba2aSMartin Storsjo auto thread_handle = GetThreadHandle();
493b1f6ba2aSMartin Storsjo Status error =
494b1f6ba2aSMartin Storsjo GetThreadContextHelper(thread_handle, &tls_context, context_flag);
495b1f6ba2aSMartin Storsjo if (error.Fail())
496b1f6ba2aSMartin Storsjo return error;
497b1f6ba2aSMartin Storsjo
498b1f6ba2aSMartin Storsjo switch (reg) {
499b1f6ba2aSMartin Storsjo case fpu_v0_arm64:
500b1f6ba2aSMartin Storsjo case fpu_v1_arm64:
501b1f6ba2aSMartin Storsjo case fpu_v2_arm64:
502b1f6ba2aSMartin Storsjo case fpu_v3_arm64:
503b1f6ba2aSMartin Storsjo case fpu_v4_arm64:
504b1f6ba2aSMartin Storsjo case fpu_v5_arm64:
505b1f6ba2aSMartin Storsjo case fpu_v6_arm64:
506b1f6ba2aSMartin Storsjo case fpu_v7_arm64:
507b1f6ba2aSMartin Storsjo case fpu_v8_arm64:
508b1f6ba2aSMartin Storsjo case fpu_v9_arm64:
509b1f6ba2aSMartin Storsjo case fpu_v10_arm64:
510b1f6ba2aSMartin Storsjo case fpu_v11_arm64:
511b1f6ba2aSMartin Storsjo case fpu_v12_arm64:
512b1f6ba2aSMartin Storsjo case fpu_v13_arm64:
513b1f6ba2aSMartin Storsjo case fpu_v14_arm64:
514b1f6ba2aSMartin Storsjo case fpu_v15_arm64:
515b1f6ba2aSMartin Storsjo case fpu_v16_arm64:
516b1f6ba2aSMartin Storsjo case fpu_v17_arm64:
517b1f6ba2aSMartin Storsjo case fpu_v18_arm64:
518b1f6ba2aSMartin Storsjo case fpu_v19_arm64:
519b1f6ba2aSMartin Storsjo case fpu_v20_arm64:
520b1f6ba2aSMartin Storsjo case fpu_v21_arm64:
521b1f6ba2aSMartin Storsjo case fpu_v22_arm64:
522b1f6ba2aSMartin Storsjo case fpu_v23_arm64:
523b1f6ba2aSMartin Storsjo case fpu_v24_arm64:
524b1f6ba2aSMartin Storsjo case fpu_v25_arm64:
525b1f6ba2aSMartin Storsjo case fpu_v26_arm64:
526b1f6ba2aSMartin Storsjo case fpu_v27_arm64:
527b1f6ba2aSMartin Storsjo case fpu_v28_arm64:
528b1f6ba2aSMartin Storsjo case fpu_v29_arm64:
529b1f6ba2aSMartin Storsjo case fpu_v30_arm64:
530b1f6ba2aSMartin Storsjo case fpu_v31_arm64:
531b1f6ba2aSMartin Storsjo memcpy(tls_context.V[reg - fpu_v0_arm64].B, reg_value.GetBytes(), 16);
532b1f6ba2aSMartin Storsjo break;
533b1f6ba2aSMartin Storsjo
534b1f6ba2aSMartin Storsjo case fpu_s0_arm64:
535b1f6ba2aSMartin Storsjo case fpu_s1_arm64:
536b1f6ba2aSMartin Storsjo case fpu_s2_arm64:
537b1f6ba2aSMartin Storsjo case fpu_s3_arm64:
538b1f6ba2aSMartin Storsjo case fpu_s4_arm64:
539b1f6ba2aSMartin Storsjo case fpu_s5_arm64:
540b1f6ba2aSMartin Storsjo case fpu_s6_arm64:
541b1f6ba2aSMartin Storsjo case fpu_s7_arm64:
542b1f6ba2aSMartin Storsjo case fpu_s8_arm64:
543b1f6ba2aSMartin Storsjo case fpu_s9_arm64:
544b1f6ba2aSMartin Storsjo case fpu_s10_arm64:
545b1f6ba2aSMartin Storsjo case fpu_s11_arm64:
546b1f6ba2aSMartin Storsjo case fpu_s12_arm64:
547b1f6ba2aSMartin Storsjo case fpu_s13_arm64:
548b1f6ba2aSMartin Storsjo case fpu_s14_arm64:
549b1f6ba2aSMartin Storsjo case fpu_s15_arm64:
550b1f6ba2aSMartin Storsjo case fpu_s16_arm64:
551b1f6ba2aSMartin Storsjo case fpu_s17_arm64:
552b1f6ba2aSMartin Storsjo case fpu_s18_arm64:
553b1f6ba2aSMartin Storsjo case fpu_s19_arm64:
554b1f6ba2aSMartin Storsjo case fpu_s20_arm64:
555b1f6ba2aSMartin Storsjo case fpu_s21_arm64:
556b1f6ba2aSMartin Storsjo case fpu_s22_arm64:
557b1f6ba2aSMartin Storsjo case fpu_s23_arm64:
558b1f6ba2aSMartin Storsjo case fpu_s24_arm64:
559b1f6ba2aSMartin Storsjo case fpu_s25_arm64:
560b1f6ba2aSMartin Storsjo case fpu_s26_arm64:
561b1f6ba2aSMartin Storsjo case fpu_s27_arm64:
562b1f6ba2aSMartin Storsjo case fpu_s28_arm64:
563b1f6ba2aSMartin Storsjo case fpu_s29_arm64:
564b1f6ba2aSMartin Storsjo case fpu_s30_arm64:
565b1f6ba2aSMartin Storsjo case fpu_s31_arm64:
566b1f6ba2aSMartin Storsjo tls_context.V[reg - fpu_s0_arm64].S[0] = reg_value.GetAsFloat();
567b1f6ba2aSMartin Storsjo break;
568b1f6ba2aSMartin Storsjo
569b1f6ba2aSMartin Storsjo case fpu_d0_arm64:
570b1f6ba2aSMartin Storsjo case fpu_d1_arm64:
571b1f6ba2aSMartin Storsjo case fpu_d2_arm64:
572b1f6ba2aSMartin Storsjo case fpu_d3_arm64:
573b1f6ba2aSMartin Storsjo case fpu_d4_arm64:
574b1f6ba2aSMartin Storsjo case fpu_d5_arm64:
575b1f6ba2aSMartin Storsjo case fpu_d6_arm64:
576b1f6ba2aSMartin Storsjo case fpu_d7_arm64:
577b1f6ba2aSMartin Storsjo case fpu_d8_arm64:
578b1f6ba2aSMartin Storsjo case fpu_d9_arm64:
579b1f6ba2aSMartin Storsjo case fpu_d10_arm64:
580b1f6ba2aSMartin Storsjo case fpu_d11_arm64:
581b1f6ba2aSMartin Storsjo case fpu_d12_arm64:
582b1f6ba2aSMartin Storsjo case fpu_d13_arm64:
583b1f6ba2aSMartin Storsjo case fpu_d14_arm64:
584b1f6ba2aSMartin Storsjo case fpu_d15_arm64:
585b1f6ba2aSMartin Storsjo case fpu_d16_arm64:
586b1f6ba2aSMartin Storsjo case fpu_d17_arm64:
587b1f6ba2aSMartin Storsjo case fpu_d18_arm64:
588b1f6ba2aSMartin Storsjo case fpu_d19_arm64:
589b1f6ba2aSMartin Storsjo case fpu_d20_arm64:
590b1f6ba2aSMartin Storsjo case fpu_d21_arm64:
591b1f6ba2aSMartin Storsjo case fpu_d22_arm64:
592b1f6ba2aSMartin Storsjo case fpu_d23_arm64:
593b1f6ba2aSMartin Storsjo case fpu_d24_arm64:
594b1f6ba2aSMartin Storsjo case fpu_d25_arm64:
595b1f6ba2aSMartin Storsjo case fpu_d26_arm64:
596b1f6ba2aSMartin Storsjo case fpu_d27_arm64:
597b1f6ba2aSMartin Storsjo case fpu_d28_arm64:
598b1f6ba2aSMartin Storsjo case fpu_d29_arm64:
599b1f6ba2aSMartin Storsjo case fpu_d30_arm64:
600b1f6ba2aSMartin Storsjo case fpu_d31_arm64:
601b1f6ba2aSMartin Storsjo tls_context.V[reg - fpu_d0_arm64].D[0] = reg_value.GetAsDouble();
602b1f6ba2aSMartin Storsjo break;
603b1f6ba2aSMartin Storsjo
604b1f6ba2aSMartin Storsjo case fpu_fpsr_arm64:
605b1f6ba2aSMartin Storsjo tls_context.Fpsr = reg_value.GetAsUInt32();
606b1f6ba2aSMartin Storsjo break;
607b1f6ba2aSMartin Storsjo
608b1f6ba2aSMartin Storsjo case fpu_fpcr_arm64:
609b1f6ba2aSMartin Storsjo tls_context.Fpcr = reg_value.GetAsUInt32();
610b1f6ba2aSMartin Storsjo break;
611b1f6ba2aSMartin Storsjo }
612b1f6ba2aSMartin Storsjo
613b1f6ba2aSMartin Storsjo return SetThreadContextHelper(thread_handle, &tls_context);
614b1f6ba2aSMartin Storsjo }
615b1f6ba2aSMartin Storsjo
616b1f6ba2aSMartin Storsjo Status
ReadRegister(const RegisterInfo * reg_info,RegisterValue & reg_value)617b1f6ba2aSMartin Storsjo NativeRegisterContextWindows_arm64::ReadRegister(const RegisterInfo *reg_info,
618b1f6ba2aSMartin Storsjo RegisterValue ®_value) {
619b1f6ba2aSMartin Storsjo Status error;
620b1f6ba2aSMartin Storsjo if (!reg_info) {
621b1f6ba2aSMartin Storsjo error.SetErrorString("reg_info NULL");
622b1f6ba2aSMartin Storsjo return error;
623b1f6ba2aSMartin Storsjo }
624b1f6ba2aSMartin Storsjo
625b1f6ba2aSMartin Storsjo const uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
626b1f6ba2aSMartin Storsjo if (reg == LLDB_INVALID_REGNUM) {
627b1f6ba2aSMartin Storsjo // This is likely an internal register for lldb use only and should not be
628b1f6ba2aSMartin Storsjo // directly queried.
629b1f6ba2aSMartin Storsjo error.SetErrorStringWithFormat("register \"%s\" is an internal-only lldb "
630b1f6ba2aSMartin Storsjo "register, cannot read directly",
631b1f6ba2aSMartin Storsjo reg_info->name);
632b1f6ba2aSMartin Storsjo return error;
633b1f6ba2aSMartin Storsjo }
634b1f6ba2aSMartin Storsjo
635b1f6ba2aSMartin Storsjo if (IsGPR(reg))
636b1f6ba2aSMartin Storsjo return GPRRead(reg, reg_value);
637b1f6ba2aSMartin Storsjo
638b1f6ba2aSMartin Storsjo if (IsFPR(reg))
639b1f6ba2aSMartin Storsjo return FPRRead(reg, reg_value);
640b1f6ba2aSMartin Storsjo
641b1f6ba2aSMartin Storsjo return Status("unimplemented");
642b1f6ba2aSMartin Storsjo }
643b1f6ba2aSMartin Storsjo
WriteRegister(const RegisterInfo * reg_info,const RegisterValue & reg_value)644b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::WriteRegister(
645b1f6ba2aSMartin Storsjo const RegisterInfo *reg_info, const RegisterValue ®_value) {
646b1f6ba2aSMartin Storsjo Status error;
647b1f6ba2aSMartin Storsjo
648b1f6ba2aSMartin Storsjo if (!reg_info) {
649b1f6ba2aSMartin Storsjo error.SetErrorString("reg_info NULL");
650b1f6ba2aSMartin Storsjo return error;
651b1f6ba2aSMartin Storsjo }
652b1f6ba2aSMartin Storsjo
653b1f6ba2aSMartin Storsjo const uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
654b1f6ba2aSMartin Storsjo if (reg == LLDB_INVALID_REGNUM) {
655b1f6ba2aSMartin Storsjo // This is likely an internal register for lldb use only and should not be
656b1f6ba2aSMartin Storsjo // directly written.
657b1f6ba2aSMartin Storsjo error.SetErrorStringWithFormat("register \"%s\" is an internal-only lldb "
658b1f6ba2aSMartin Storsjo "register, cannot write directly",
659b1f6ba2aSMartin Storsjo reg_info->name);
660b1f6ba2aSMartin Storsjo return error;
661b1f6ba2aSMartin Storsjo }
662b1f6ba2aSMartin Storsjo
663b1f6ba2aSMartin Storsjo if (IsGPR(reg))
664b1f6ba2aSMartin Storsjo return GPRWrite(reg, reg_value);
665b1f6ba2aSMartin Storsjo
666b1f6ba2aSMartin Storsjo if (IsFPR(reg))
667b1f6ba2aSMartin Storsjo return FPRWrite(reg, reg_value);
668b1f6ba2aSMartin Storsjo
669b1f6ba2aSMartin Storsjo return Status("unimplemented");
670b1f6ba2aSMartin Storsjo }
671b1f6ba2aSMartin Storsjo
ReadAllRegisterValues(lldb::WritableDataBufferSP & data_sp)672b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::ReadAllRegisterValues(
673*c2f64601SJonas Devlieghere lldb::WritableDataBufferSP &data_sp) {
674b1f6ba2aSMartin Storsjo const size_t data_size = REG_CONTEXT_SIZE;
675b1f6ba2aSMartin Storsjo data_sp = std::make_shared<DataBufferHeap>(data_size, 0);
676b1f6ba2aSMartin Storsjo ::CONTEXT tls_context;
677b1f6ba2aSMartin Storsjo Status error =
678b1f6ba2aSMartin Storsjo GetThreadContextHelper(GetThreadHandle(), &tls_context, CONTEXT_ALL);
679b1f6ba2aSMartin Storsjo if (error.Fail())
680b1f6ba2aSMartin Storsjo return error;
681b1f6ba2aSMartin Storsjo
682b1f6ba2aSMartin Storsjo uint8_t *dst = data_sp->GetBytes();
683b1f6ba2aSMartin Storsjo ::memcpy(dst, &tls_context, data_size);
684b1f6ba2aSMartin Storsjo return error;
685b1f6ba2aSMartin Storsjo }
686b1f6ba2aSMartin Storsjo
WriteAllRegisterValues(const lldb::DataBufferSP & data_sp)687b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::WriteAllRegisterValues(
688b1f6ba2aSMartin Storsjo const lldb::DataBufferSP &data_sp) {
689b1f6ba2aSMartin Storsjo Status error;
690b1f6ba2aSMartin Storsjo const size_t data_size = REG_CONTEXT_SIZE;
691b1f6ba2aSMartin Storsjo if (!data_sp) {
692b1f6ba2aSMartin Storsjo error.SetErrorStringWithFormat(
693b1f6ba2aSMartin Storsjo "NativeRegisterContextWindows_arm64::%s invalid data_sp provided",
694b1f6ba2aSMartin Storsjo __FUNCTION__);
695b1f6ba2aSMartin Storsjo return error;
696b1f6ba2aSMartin Storsjo }
697b1f6ba2aSMartin Storsjo
698b1f6ba2aSMartin Storsjo if (data_sp->GetByteSize() != data_size) {
699b1f6ba2aSMartin Storsjo error.SetErrorStringWithFormatv(
700b1f6ba2aSMartin Storsjo "data_sp contained mismatched data size, expected {0}, actual {1}",
701b1f6ba2aSMartin Storsjo data_size, data_sp->GetByteSize());
702b1f6ba2aSMartin Storsjo return error;
703b1f6ba2aSMartin Storsjo }
704b1f6ba2aSMartin Storsjo
705b1f6ba2aSMartin Storsjo ::CONTEXT tls_context;
706b1f6ba2aSMartin Storsjo memcpy(&tls_context, data_sp->GetBytes(), data_size);
707b1f6ba2aSMartin Storsjo return SetThreadContextHelper(GetThreadHandle(), &tls_context);
708b1f6ba2aSMartin Storsjo }
709b1f6ba2aSMartin Storsjo
IsWatchpointHit(uint32_t wp_index,bool & is_hit)710b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::IsWatchpointHit(uint32_t wp_index,
711b1f6ba2aSMartin Storsjo bool &is_hit) {
712b1f6ba2aSMartin Storsjo return Status("unimplemented");
713b1f6ba2aSMartin Storsjo }
714b1f6ba2aSMartin Storsjo
GetWatchpointHitIndex(uint32_t & wp_index,lldb::addr_t trap_addr)715b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::GetWatchpointHitIndex(
716b1f6ba2aSMartin Storsjo uint32_t &wp_index, lldb::addr_t trap_addr) {
717b1f6ba2aSMartin Storsjo return Status("unimplemented");
718b1f6ba2aSMartin Storsjo }
719b1f6ba2aSMartin Storsjo
IsWatchpointVacant(uint32_t wp_index,bool & is_vacant)720b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::IsWatchpointVacant(uint32_t wp_index,
721b1f6ba2aSMartin Storsjo bool &is_vacant) {
722b1f6ba2aSMartin Storsjo return Status("unimplemented");
723b1f6ba2aSMartin Storsjo }
724b1f6ba2aSMartin Storsjo
SetHardwareWatchpointWithIndex(lldb::addr_t addr,size_t size,uint32_t watch_flags,uint32_t wp_index)725b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::SetHardwareWatchpointWithIndex(
726b1f6ba2aSMartin Storsjo lldb::addr_t addr, size_t size, uint32_t watch_flags, uint32_t wp_index) {
727b1f6ba2aSMartin Storsjo return Status("unimplemented");
728b1f6ba2aSMartin Storsjo }
729b1f6ba2aSMartin Storsjo
ClearHardwareWatchpoint(uint32_t wp_index)730b1f6ba2aSMartin Storsjo bool NativeRegisterContextWindows_arm64::ClearHardwareWatchpoint(
731b1f6ba2aSMartin Storsjo uint32_t wp_index) {
732b1f6ba2aSMartin Storsjo return false;
733b1f6ba2aSMartin Storsjo }
734b1f6ba2aSMartin Storsjo
ClearAllHardwareWatchpoints()735b1f6ba2aSMartin Storsjo Status NativeRegisterContextWindows_arm64::ClearAllHardwareWatchpoints() {
736b1f6ba2aSMartin Storsjo return Status("unimplemented");
737b1f6ba2aSMartin Storsjo }
738b1f6ba2aSMartin Storsjo
SetHardwareWatchpoint(lldb::addr_t addr,size_t size,uint32_t watch_flags)739b1f6ba2aSMartin Storsjo uint32_t NativeRegisterContextWindows_arm64::SetHardwareWatchpoint(
740b1f6ba2aSMartin Storsjo lldb::addr_t addr, size_t size, uint32_t watch_flags) {
741b1f6ba2aSMartin Storsjo return LLDB_INVALID_INDEX32;
742b1f6ba2aSMartin Storsjo }
743b1f6ba2aSMartin Storsjo
744b1f6ba2aSMartin Storsjo lldb::addr_t
GetWatchpointAddress(uint32_t wp_index)745b1f6ba2aSMartin Storsjo NativeRegisterContextWindows_arm64::GetWatchpointAddress(uint32_t wp_index) {
746b1f6ba2aSMartin Storsjo return LLDB_INVALID_ADDRESS;
747b1f6ba2aSMartin Storsjo }
748b1f6ba2aSMartin Storsjo
NumSupportedHardwareWatchpoints()749b1f6ba2aSMartin Storsjo uint32_t NativeRegisterContextWindows_arm64::NumSupportedHardwareWatchpoints() {
750b1f6ba2aSMartin Storsjo // Not implemented
751b1f6ba2aSMartin Storsjo return 0;
752b1f6ba2aSMartin Storsjo }
753b1f6ba2aSMartin Storsjo
754b1f6ba2aSMartin Storsjo #endif // defined(__aarch64__) || defined(_M_ARM64)
755