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
18#include "TargetThreadWindows.h"
19
20#include "llvm/ADT/STLExtras.h"
21
22using namespace lldb;
23using namespace lldb_private;
24
25#define GPR_OFFSET(idx) 0
26#define FPU_OFFSET(idx) 0
27#define FPSCR_OFFSET 0
28#define TLS_OFFSET 0
29#define EXC_OFFSET(reg) 0
30#define DBG_OFFSET_NAME(reg) 0
31
32#define DEFINE_DBG(reg, i) \
33 #reg, NULL, \
34 0, DBG_OFFSET_NAME(reg[i]), eEncodingUint, eFormatHex, \
35 {LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, \
36 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, \
37 LLDB_INVALID_REGNUM }, \
38 NULL, NULL, NULL
39
40// Include RegisterInfos_arm to declare our g_register_infos_arm structure.
41#define DECLARE_REGISTER_INFOS_ARM_STRUCT
43#undef DECLARE_REGISTER_INFOS_ARM_STRUCT
44
45static size_t k_num_register_infos = std::size(g_register_infos_arm);
46
47// Array of lldb register numbers used to define the set of all General Purpose
48// Registers
49uint32_t g_gpr_reg_indices[] = {
52};
53
54uint32_t g_fpu_reg_indices[] = {
59
60 fpu_d0, fpu_d1, fpu_d2, fpu_d3, fpu_d4, fpu_d5, fpu_d6, fpu_d7,
61 fpu_d8, fpu_d9, fpu_d10, fpu_d11, fpu_d12, fpu_d13, fpu_d14, fpu_d15,
62 fpu_d16, fpu_d17, fpu_d18, fpu_d19, fpu_d20, fpu_d21, fpu_d22, fpu_d23,
63 fpu_d24, fpu_d25, fpu_d26, fpu_d27, fpu_d28, fpu_d29, fpu_d30, fpu_d31,
64
65 fpu_q0, fpu_q1, fpu_q2, fpu_q3, fpu_q4, fpu_q5, fpu_q6, fpu_q7,
66 fpu_q8, fpu_q9, fpu_q10, fpu_q11, fpu_q12, fpu_q13, fpu_q14, fpu_q15,
67
69};
70
71RegisterSet g_register_sets[] = {
72 {"General Purpose Registers", "gpr", std::size(g_gpr_reg_indices),
73 g_gpr_reg_indices},
74 {"Floating Point Registers", "fpu", std::size(g_fpu_reg_indices),
75 g_fpu_reg_indices},
76};
77
78// Constructors and Destructors
79RegisterContextWindows_arm::RegisterContextWindows_arm(
80 Thread &thread, uint32_t concrete_frame_idx)
81 : RegisterContextWindows(thread, concrete_frame_idx) {}
82
83RegisterContextWindows_arm::~RegisterContextWindows_arm() {}
84
85size_t RegisterContextWindows_arm::GetRegisterCount() {
86 return std::size(g_register_infos_arm);
87}
88
89const RegisterInfo *
90RegisterContextWindows_arm::GetRegisterInfoAtIndex(size_t reg) {
91 if (reg < k_num_register_infos)
92 return &g_register_infos_arm[reg];
93 return NULL;
94}
95
96size_t RegisterContextWindows_arm::GetRegisterSetCount() {
97 return std::size(g_register_sets);
98}
99
100const RegisterSet *RegisterContextWindows_arm::GetRegisterSet(size_t reg_set) {
101 return &g_register_sets[reg_set];
102}
103
104bool RegisterContextWindows_arm::ReadRegister(const RegisterInfo *reg_info,
105 RegisterValue &reg_value) {
106 if (!CacheAllRegisterValues())
107 return false;
108
109 if (reg_info == nullptr)
110 return false;
111
112 const uint32_t reg = reg_info->kinds[eRegisterKindLLDB];
113
114 switch (reg) {
115 case gpr_r0:
116 reg_value.SetUInt32(m_context.R0);
117 break;
118 case gpr_r1:
119 reg_value.SetUInt32(m_context.R1);
120 break;
121 case gpr_r2:
122 reg_value.SetUInt32(m_context.R2);
123 break;
124 case gpr_r3:
125 reg_value.SetUInt32(m_context.R3);
126 break;
127 case gpr_r4:
128 reg_value.SetUInt32(m_context.R4);
129 break;
130 case gpr_r5:
131 reg_value.SetUInt32(m_context.R5);
132 break;
133 case gpr_r6:
134 reg_value.SetUInt32(m_context.R6);
135 break;
136 case gpr_r7:
137 reg_value.SetUInt32(m_context.R7);
138 break;
139 case gpr_r8:
140 reg_value.SetUInt32(m_context.R8);
141 break;
142 case gpr_r9:
143 reg_value.SetUInt32(m_context.R9);
144 break;
145 case gpr_r10:
146 reg_value.SetUInt32(m_context.R10);
147 break;
148 case gpr_r11:
149 reg_value.SetUInt32(m_context.R11);
150 break;
151 case gpr_r12:
152 reg_value.SetUInt32(m_context.R12);
153 break;
154 case gpr_sp:
155 reg_value.SetUInt32(m_context.Sp);
156 break;
157 case gpr_lr:
158 reg_value.SetUInt32(m_context.Lr);
159 break;
160 case gpr_pc:
161 reg_value.SetUInt32(m_context.Pc);
162 break;
163 case gpr_cpsr:
164 reg_value.SetUInt32(m_context.Cpsr);
165 break;
166
167 case fpu_s0:
168 case fpu_s1:
169 case fpu_s2:
170 case fpu_s3:
171 case fpu_s4:
172 case fpu_s5:
173 case fpu_s6:
174 case fpu_s7:
175 case fpu_s8:
176 case fpu_s9:
177 case fpu_s10:
178 case fpu_s11:
179 case fpu_s12:
180 case fpu_s13:
181 case fpu_s14:
182 case fpu_s15:
183 case fpu_s16:
184 case fpu_s17:
185 case fpu_s18:
186 case fpu_s19:
187 case fpu_s20:
188 case fpu_s21:
189 case fpu_s22:
190 case fpu_s23:
191 case fpu_s24:
192 case fpu_s25:
193 case fpu_s26:
194 case fpu_s27:
195 case fpu_s28:
196 case fpu_s29:
197 case fpu_s30:
198 case fpu_s31:
199 reg_value.SetUInt32(m_context.S[reg - fpu_s0], RegisterValue::eTypeFloat);
200 break;
201
202 case fpu_d0:
203 case fpu_d1:
204 case fpu_d2:
205 case fpu_d3:
206 case fpu_d4:
207 case fpu_d5:
208 case fpu_d6:
209 case fpu_d7:
210 case fpu_d8:
211 case fpu_d9:
212 case fpu_d10:
213 case fpu_d11:
214 case fpu_d12:
215 case fpu_d13:
216 case fpu_d14:
217 case fpu_d15:
218 case fpu_d16:
219 case fpu_d17:
220 case fpu_d18:
221 case fpu_d19:
222 case fpu_d20:
223 case fpu_d21:
224 case fpu_d22:
225 case fpu_d23:
226 case fpu_d24:
227 case fpu_d25:
228 case fpu_d26:
229 case fpu_d27:
230 case fpu_d28:
231 case fpu_d29:
232 case fpu_d30:
233 case fpu_d31:
234 reg_value.SetUInt64(m_context.D[reg - fpu_d0], RegisterValue::eTypeDouble);
235 break;
236
237 case fpu_q0:
238 case fpu_q1:
239 case fpu_q2:
240 case fpu_q3:
241 case fpu_q4:
242 case fpu_q5:
243 case fpu_q6:
244 case fpu_q7:
245 case fpu_q8:
246 case fpu_q9:
247 case fpu_q10:
248 case fpu_q11:
249 case fpu_q12:
250 case fpu_q13:
251 case fpu_q14:
252 case fpu_q15:
253 reg_value.SetBytes(&m_context.Q[reg - fpu_q0], reg_info->byte_size,
255 break;
256
257 case fpu_fpscr:
258 reg_value.SetUInt32(m_context.Fpscr);
259 break;
260
261 default:
262 reg_value.SetValueToInvalid();
263 return false;
264 }
265 return true;
266}
267
268bool RegisterContextWindows_arm::WriteRegister(const RegisterInfo *reg_info,
269 const RegisterValue &reg_value) {
270 // Since we cannot only write a single register value to the inferior, we
271 // need to make sure our cached copy of the register values are fresh.
272 // Otherwise when writing EAX, for example, we may also overwrite some other
273 // register with a stale value.
274 if (!CacheAllRegisterValues())
275 return false;
276
277 const uint32_t reg = reg_info->kinds[eRegisterKindLLDB];
278
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 return false;
427 }
428
429 // Physically update the registers in the target process.
430 return ApplyAllRegisterValues();
431}
432
433#endif // defined(__arm__) || defined(_M_ARM)
static const uint32_t k_num_register_infos
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.
@ 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.