LLDB mainline
RegisterContextWindows_arm.cpp
Go to the documentation of this file.
1//===-- RegisterContextWindows_arm.cpp ------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#if defined(__arm__) || defined(_M_ARM)
10
14#include "lldb/Utility/Status.h"
16
17#include "ProcessWindowsLog.h"
19#include "TargetThreadWindows.h"
20
21#include "llvm/ADT/STLExtras.h"
22
23using namespace lldb;
24using namespace lldb_private;
25
26#define GPR_OFFSET(idx) 0
27#define FPU_OFFSET(idx) 0
28#define FPSCR_OFFSET 0
29#define TLS_OFFSET 0
30#define EXC_OFFSET(reg) 0
31#define DBG_OFFSET_NAME(reg) 0
32
33#define DEFINE_DBG(reg, i) \
34 #reg, nullptr, 0, DBG_OFFSET_NAME(reg[i]), eEncodingUint, eFormatHex, \
35 {LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, \
36 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, \
37 nullptr, nullptr, nullptr
38
39// Include RegisterInfos_arm to declare our g_register_infos_arm structure.
40#define DECLARE_REGISTER_INFOS_ARM_STRUCT
42#undef DECLARE_REGISTER_INFOS_ARM_STRUCT
43
44static size_t k_num_register_infos = std::size(g_register_infos_arm);
45
46// Array of lldb register numbers used to define the set of all General Purpose
47// Registers
48uint32_t g_gpr_reg_indices[] = {
51};
52
53uint32_t g_fpu_reg_indices[] = {
58
59 fpu_d0, fpu_d1, fpu_d2, fpu_d3, fpu_d4, fpu_d5, fpu_d6, fpu_d7,
60 fpu_d8, fpu_d9, fpu_d10, fpu_d11, fpu_d12, fpu_d13, fpu_d14, fpu_d15,
61 fpu_d16, fpu_d17, fpu_d18, fpu_d19, fpu_d20, fpu_d21, fpu_d22, fpu_d23,
62 fpu_d24, fpu_d25, fpu_d26, fpu_d27, fpu_d28, fpu_d29, fpu_d30, fpu_d31,
63
64 fpu_q0, fpu_q1, fpu_q2, fpu_q3, fpu_q4, fpu_q5, fpu_q6, fpu_q7,
65 fpu_q8, fpu_q9, fpu_q10, fpu_q11, fpu_q12, fpu_q13, fpu_q14, fpu_q15,
66
68};
69
70RegisterSet g_register_sets[] = {
71 {"General Purpose Registers", "gpr", std::size(g_gpr_reg_indices),
72 g_gpr_reg_indices},
73 {"Floating Point Registers", "fpu", std::size(g_fpu_reg_indices),
74 g_fpu_reg_indices},
75};
76
77// Constructors and Destructors
78RegisterContextWindows_arm::RegisterContextWindows_arm(
79 Thread &thread, uint32_t concrete_frame_idx)
80 : RegisterContextWindows(thread, concrete_frame_idx) {}
81
82RegisterContextWindows_arm::~RegisterContextWindows_arm() = default;
83
84size_t RegisterContextWindows_arm::GetRegisterCount() {
85 return std::size(g_register_infos_arm);
86}
87
88const RegisterInfo *
89RegisterContextWindows_arm::GetRegisterInfoAtIndex(size_t reg) {
90 if (reg < k_num_register_infos)
91 return &g_register_infos_arm[reg];
92 return nullptr;
93}
94
95size_t RegisterContextWindows_arm::GetRegisterSetCount() {
96 return std::size(g_register_sets);
97}
98
99const RegisterSet *RegisterContextWindows_arm::GetRegisterSet(size_t reg_set) {
100 return &g_register_sets[reg_set];
101}
102
103bool RegisterContextWindows_arm::ReadRegister(const RegisterInfo *reg_info,
104 RegisterValue &reg_value) {
105 if (!CacheAllRegisterValues())
106 return false;
107
108 if (reg_info == nullptr)
109 return false;
110
111 const uint32_t reg = reg_info->kinds[eRegisterKindLLDB];
112
113 switch (reg) {
114 case gpr_r0:
115 reg_value.SetUInt32(m_context.R0);
116 break;
117 case gpr_r1:
118 reg_value.SetUInt32(m_context.R1);
119 break;
120 case gpr_r2:
121 reg_value.SetUInt32(m_context.R2);
122 break;
123 case gpr_r3:
124 reg_value.SetUInt32(m_context.R3);
125 break;
126 case gpr_r4:
127 reg_value.SetUInt32(m_context.R4);
128 break;
129 case gpr_r5:
130 reg_value.SetUInt32(m_context.R5);
131 break;
132 case gpr_r6:
133 reg_value.SetUInt32(m_context.R6);
134 break;
135 case gpr_r7:
136 reg_value.SetUInt32(m_context.R7);
137 break;
138 case gpr_r8:
139 reg_value.SetUInt32(m_context.R8);
140 break;
141 case gpr_r9:
142 reg_value.SetUInt32(m_context.R9);
143 break;
144 case gpr_r10:
145 reg_value.SetUInt32(m_context.R10);
146 break;
147 case gpr_r11:
148 reg_value.SetUInt32(m_context.R11);
149 break;
150 case gpr_r12:
151 reg_value.SetUInt32(m_context.R12);
152 break;
153 case gpr_sp:
154 reg_value.SetUInt32(m_context.Sp);
155 break;
156 case gpr_lr:
157 reg_value.SetUInt32(m_context.Lr);
158 break;
159 case gpr_pc:
160 reg_value.SetUInt32(m_context.Pc);
161 break;
162 case gpr_cpsr:
163 reg_value.SetUInt32(m_context.Cpsr);
164 break;
165
166 case fpu_s0:
167 case fpu_s1:
168 case fpu_s2:
169 case fpu_s3:
170 case fpu_s4:
171 case fpu_s5:
172 case fpu_s6:
173 case fpu_s7:
174 case fpu_s8:
175 case fpu_s9:
176 case fpu_s10:
177 case fpu_s11:
178 case fpu_s12:
179 case fpu_s13:
180 case fpu_s14:
181 case fpu_s15:
182 case fpu_s16:
183 case fpu_s17:
184 case fpu_s18:
185 case fpu_s19:
186 case fpu_s20:
187 case fpu_s21:
188 case fpu_s22:
189 case fpu_s23:
190 case fpu_s24:
191 case fpu_s25:
192 case fpu_s26:
193 case fpu_s27:
194 case fpu_s28:
195 case fpu_s29:
196 case fpu_s30:
197 case fpu_s31:
198 reg_value.SetUInt32(m_context.S[reg - fpu_s0], RegisterValue::eTypeFloat);
199 break;
200
201 case fpu_d0:
202 case fpu_d1:
203 case fpu_d2:
204 case fpu_d3:
205 case fpu_d4:
206 case fpu_d5:
207 case fpu_d6:
208 case fpu_d7:
209 case fpu_d8:
210 case fpu_d9:
211 case fpu_d10:
212 case fpu_d11:
213 case fpu_d12:
214 case fpu_d13:
215 case fpu_d14:
216 case fpu_d15:
217 case fpu_d16:
218 case fpu_d17:
219 case fpu_d18:
220 case fpu_d19:
221 case fpu_d20:
222 case fpu_d21:
223 case fpu_d22:
224 case fpu_d23:
225 case fpu_d24:
226 case fpu_d25:
227 case fpu_d26:
228 case fpu_d27:
229 case fpu_d28:
230 case fpu_d29:
231 case fpu_d30:
232 case fpu_d31:
233 reg_value.SetUInt64(m_context.D[reg - fpu_d0], RegisterValue::eTypeDouble);
234 break;
235
236 case fpu_q0:
237 case fpu_q1:
238 case fpu_q2:
239 case fpu_q3:
240 case fpu_q4:
241 case fpu_q5:
242 case fpu_q6:
243 case fpu_q7:
244 case fpu_q8:
245 case fpu_q9:
246 case fpu_q10:
247 case fpu_q11:
248 case fpu_q12:
249 case fpu_q13:
250 case fpu_q14:
251 case fpu_q15:
252 reg_value.SetBytes(&m_context.Q[reg - fpu_q0], reg_info->byte_size,
254 break;
255
256 case fpu_fpscr:
257 reg_value.SetUInt32(m_context.Fpscr);
258 break;
259
260 default:
261 reg_value.SetValueToInvalid();
262 return false;
263 }
264 return true;
265}
266
267bool RegisterContextWindows_arm::WriteRegister(const RegisterInfo *reg_info,
268 const RegisterValue &reg_value) {
269 // Since we cannot only write a single register value to the inferior, we
270 // need to make sure our cached copy of the register values are fresh.
271 // Otherwise when writing EAX, for example, we may also overwrite some other
272 // register with a stale value.
273 if (!CacheAllRegisterValues())
274 return false;
275
276 const uint32_t reg = reg_info->kinds[eRegisterKindLLDB];
277
279 switch (reg) {
280 case gpr_r0:
281 m_context.R0 = reg_value.GetAsUInt32();
282 break;
283 case gpr_r1:
284 m_context.R1 = reg_value.GetAsUInt32();
285 break;
286 case gpr_r2:
287 m_context.R2 = reg_value.GetAsUInt32();
288 break;
289 case gpr_r3:
290 m_context.R3 = reg_value.GetAsUInt32();
291 break;
292 case gpr_r4:
293 m_context.R4 = reg_value.GetAsUInt32();
294 break;
295 case gpr_r5:
296 m_context.R5 = reg_value.GetAsUInt32();
297 break;
298 case gpr_r6:
299 m_context.R6 = reg_value.GetAsUInt32();
300 break;
301 case gpr_r7:
302 m_context.R7 = reg_value.GetAsUInt32();
303 break;
304 case gpr_r8:
305 m_context.R8 = reg_value.GetAsUInt32();
306 break;
307 case gpr_r9:
308 m_context.R9 = reg_value.GetAsUInt32();
309 break;
310 case gpr_r10:
311 m_context.R10 = reg_value.GetAsUInt32();
312 break;
313 case gpr_r11:
314 m_context.R11 = reg_value.GetAsUInt32();
315 break;
316 case gpr_r12:
317 m_context.R12 = reg_value.GetAsUInt32();
318 break;
319 case gpr_sp:
320 m_context.Sp = reg_value.GetAsUInt32();
321 break;
322 case gpr_lr:
323 m_context.Lr = reg_value.GetAsUInt32();
324 break;
325 case gpr_pc:
326 m_context.Pc = reg_value.GetAsUInt32();
327 break;
328 case gpr_cpsr:
329 m_context.Cpsr = reg_value.GetAsUInt32();
330 break;
331
332 case fpu_s0:
333 case fpu_s1:
334 case fpu_s2:
335 case fpu_s3:
336 case fpu_s4:
337 case fpu_s5:
338 case fpu_s6:
339 case fpu_s7:
340 case fpu_s8:
341 case fpu_s9:
342 case fpu_s10:
343 case fpu_s11:
344 case fpu_s12:
345 case fpu_s13:
346 case fpu_s14:
347 case fpu_s15:
348 case fpu_s16:
349 case fpu_s17:
350 case fpu_s18:
351 case fpu_s19:
352 case fpu_s20:
353 case fpu_s21:
354 case fpu_s22:
355 case fpu_s23:
356 case fpu_s24:
357 case fpu_s25:
358 case fpu_s26:
359 case fpu_s27:
360 case fpu_s28:
361 case fpu_s29:
362 case fpu_s30:
363 case fpu_s31:
364 m_context.S[reg - fpu_s0] = reg_value.GetAsUInt32();
365 break;
366
367 case fpu_d0:
368 case fpu_d1:
369 case fpu_d2:
370 case fpu_d3:
371 case fpu_d4:
372 case fpu_d5:
373 case fpu_d6:
374 case fpu_d7:
375 case fpu_d8:
376 case fpu_d9:
377 case fpu_d10:
378 case fpu_d11:
379 case fpu_d12:
380 case fpu_d13:
381 case fpu_d14:
382 case fpu_d15:
383 case fpu_d16:
384 case fpu_d17:
385 case fpu_d18:
386 case fpu_d19:
387 case fpu_d20:
388 case fpu_d21:
389 case fpu_d22:
390 case fpu_d23:
391 case fpu_d24:
392 case fpu_d25:
393 case fpu_d26:
394 case fpu_d27:
395 case fpu_d28:
396 case fpu_d29:
397 case fpu_d30:
398 case fpu_d31:
399 m_context.D[reg - fpu_d0] = reg_value.GetAsUInt64();
400 break;
401
402 case fpu_q0:
403 case fpu_q1:
404 case fpu_q2:
405 case fpu_q3:
406 case fpu_q4:
407 case fpu_q5:
408 case fpu_q6:
409 case fpu_q7:
410 case fpu_q8:
411 case fpu_q9:
412 case fpu_q10:
413 case fpu_q11:
414 case fpu_q12:
415 case fpu_q13:
416 case fpu_q14:
417 case fpu_q15:
418 memcpy(&m_context.Q[reg - fpu_q0], reg_value.GetBytes(), 16);
419 break;
420
421 case fpu_fpscr:
422 m_context.Fpscr = reg_value.GetAsUInt32();
423 break;
424
425 default:
426 LLDB_LOG(log, "Write value {0:x} to unknown register {1}",
427 reg_value.GetAsUInt32(), reg);
428 return false;
429 }
430
431 // Physically update the registers in the target process.
432 return ApplyAllRegisterValues();
433}
434
435#endif // defined(__arm__) || defined(_M_ARM)
static const uint32_t k_num_register_infos
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
Definition Log.h:364
void SetUInt64(uint64_t uint, Type t=eTypeUInt64)
uint64_t GetAsUInt64(uint64_t fail_value=UINT64_MAX, bool *success_ptr=nullptr) const
void SetBytes(const void *bytes, size_t length, lldb::ByteOrder byte_order)
const void * GetBytes() const
uint32_t GetAsUInt32(uint32_t fail_value=UINT32_MAX, bool *success_ptr=nullptr) const
void SetUInt32(uint32_t uint, Type t=eTypeUInt32)
@ eTypeFloat
< This value is used when the (integer) register is larger than 64-bits.
lldb::ByteOrder InlHostByteOrder()
Definition Endian.h:25
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
Definition Log.h:327
@ eRegisterKindLLDB
lldb's internal register numbers
Every register is described in detail including its name, alternate name (optional),...
uint32_t byte_size
Size in bytes of the register.
uint32_t kinds[lldb::kNumRegisterKinds]
Holds all of the various register numbers for all register kinds.
Registers are grouped into register sets.