1*9c7fbc3fSMichał Górny //===-- RegisterContextFreeBSDKernel_arm64.cpp ----------------------------===//
2*9c7fbc3fSMichał Górny //
3*9c7fbc3fSMichał Górny // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*9c7fbc3fSMichał Górny // See https://llvm.org/LICENSE.txt for license information.
5*9c7fbc3fSMichał Górny // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*9c7fbc3fSMichał Górny //
7*9c7fbc3fSMichał Górny //===----------------------------------------------------------------------===//
8*9c7fbc3fSMichał Górny 
9*9c7fbc3fSMichał Górny #include "RegisterContextFreeBSDKernel_arm64.h"
10*9c7fbc3fSMichał Górny #include "Plugins/Process/Utility/lldb-arm64-register-enums.h"
11*9c7fbc3fSMichał Górny 
12*9c7fbc3fSMichał Górny #include "lldb/Target/Process.h"
13*9c7fbc3fSMichał Górny #include "lldb/Target/Thread.h"
14*9c7fbc3fSMichał Górny #include "lldb/Utility/RegisterValue.h"
15*9c7fbc3fSMichał Górny #include "llvm/Support/Endian.h"
16*9c7fbc3fSMichał Górny 
17*9c7fbc3fSMichał Górny using namespace lldb;
18*9c7fbc3fSMichał Górny using namespace lldb_private;
19*9c7fbc3fSMichał Górny 
RegisterContextFreeBSDKernel_arm64(Thread & thread,std::unique_ptr<RegisterInfoPOSIX_arm64> register_info_up,lldb::addr_t pcb_addr)20*9c7fbc3fSMichał Górny RegisterContextFreeBSDKernel_arm64::RegisterContextFreeBSDKernel_arm64(
21*9c7fbc3fSMichał Górny     Thread &thread, std::unique_ptr<RegisterInfoPOSIX_arm64> register_info_up,
22*9c7fbc3fSMichał Górny     lldb::addr_t pcb_addr)
23*9c7fbc3fSMichał Górny     : RegisterContextPOSIX_arm64(thread, std::move(register_info_up)),
24*9c7fbc3fSMichał Górny       m_pcb_addr(pcb_addr) {}
25*9c7fbc3fSMichał Górny 
ReadGPR()26*9c7fbc3fSMichał Górny bool RegisterContextFreeBSDKernel_arm64::ReadGPR() { return true; }
27*9c7fbc3fSMichał Górny 
ReadFPR()28*9c7fbc3fSMichał Górny bool RegisterContextFreeBSDKernel_arm64::ReadFPR() { return true; }
29*9c7fbc3fSMichał Górny 
WriteGPR()30*9c7fbc3fSMichał Górny bool RegisterContextFreeBSDKernel_arm64::WriteGPR() {
31*9c7fbc3fSMichał Górny   assert(0);
32*9c7fbc3fSMichał Górny   return false;
33*9c7fbc3fSMichał Górny }
34*9c7fbc3fSMichał Górny 
WriteFPR()35*9c7fbc3fSMichał Górny bool RegisterContextFreeBSDKernel_arm64::WriteFPR() {
36*9c7fbc3fSMichał Górny   assert(0);
37*9c7fbc3fSMichał Górny   return false;
38*9c7fbc3fSMichał Górny }
39*9c7fbc3fSMichał Górny 
ReadRegister(const RegisterInfo * reg_info,RegisterValue & value)40*9c7fbc3fSMichał Górny bool RegisterContextFreeBSDKernel_arm64::ReadRegister(
41*9c7fbc3fSMichał Górny     const RegisterInfo *reg_info, RegisterValue &value) {
42*9c7fbc3fSMichał Górny   if (m_pcb_addr == LLDB_INVALID_ADDRESS)
43*9c7fbc3fSMichał Górny     return false;
44*9c7fbc3fSMichał Górny 
45*9c7fbc3fSMichał Górny   struct {
46*9c7fbc3fSMichał Górny     llvm::support::ulittle64_t x[30];
47*9c7fbc3fSMichał Górny     llvm::support::ulittle64_t lr;
48*9c7fbc3fSMichał Górny     llvm::support::ulittle64_t _reserved;
49*9c7fbc3fSMichał Górny     llvm::support::ulittle64_t sp;
50*9c7fbc3fSMichał Górny   } pcb;
51*9c7fbc3fSMichał Górny 
52*9c7fbc3fSMichał Górny   Status error;
53*9c7fbc3fSMichał Górny   size_t rd =
54*9c7fbc3fSMichał Górny       m_thread.GetProcess()->ReadMemory(m_pcb_addr, &pcb, sizeof(pcb), error);
55*9c7fbc3fSMichał Górny   if (rd != sizeof(pcb))
56*9c7fbc3fSMichał Górny     return false;
57*9c7fbc3fSMichał Górny 
58*9c7fbc3fSMichał Górny   uint32_t reg = reg_info->kinds[lldb::eRegisterKindLLDB];
59*9c7fbc3fSMichał Górny   switch (reg) {
60*9c7fbc3fSMichał Górny   case gpr_x0_arm64:
61*9c7fbc3fSMichał Górny   case gpr_x1_arm64:
62*9c7fbc3fSMichał Górny   case gpr_x2_arm64:
63*9c7fbc3fSMichał Górny   case gpr_x3_arm64:
64*9c7fbc3fSMichał Górny   case gpr_x4_arm64:
65*9c7fbc3fSMichał Górny   case gpr_x5_arm64:
66*9c7fbc3fSMichał Górny   case gpr_x6_arm64:
67*9c7fbc3fSMichał Górny   case gpr_x7_arm64:
68*9c7fbc3fSMichał Górny   case gpr_x8_arm64:
69*9c7fbc3fSMichał Górny   case gpr_x9_arm64:
70*9c7fbc3fSMichał Górny   case gpr_x10_arm64:
71*9c7fbc3fSMichał Górny   case gpr_x11_arm64:
72*9c7fbc3fSMichał Górny   case gpr_x12_arm64:
73*9c7fbc3fSMichał Górny   case gpr_x13_arm64:
74*9c7fbc3fSMichał Górny   case gpr_x14_arm64:
75*9c7fbc3fSMichał Górny   case gpr_x15_arm64:
76*9c7fbc3fSMichał Górny   case gpr_x16_arm64:
77*9c7fbc3fSMichał Górny   case gpr_x17_arm64:
78*9c7fbc3fSMichał Górny   case gpr_x18_arm64:
79*9c7fbc3fSMichał Górny   case gpr_x19_arm64:
80*9c7fbc3fSMichał Górny   case gpr_x20_arm64:
81*9c7fbc3fSMichał Górny   case gpr_x21_arm64:
82*9c7fbc3fSMichał Górny   case gpr_x22_arm64:
83*9c7fbc3fSMichał Górny   case gpr_x23_arm64:
84*9c7fbc3fSMichał Górny   case gpr_x24_arm64:
85*9c7fbc3fSMichał Górny   case gpr_x25_arm64:
86*9c7fbc3fSMichał Górny   case gpr_x26_arm64:
87*9c7fbc3fSMichał Górny   case gpr_x27_arm64:
88*9c7fbc3fSMichał Górny   case gpr_x28_arm64:
89*9c7fbc3fSMichał Górny   case gpr_fp_arm64:
90*9c7fbc3fSMichał Górny     static_assert(gpr_fp_arm64 - gpr_x0_arm64 == 29,
91*9c7fbc3fSMichał Górny                   "nonconsecutive arm64 register numbers");
92*9c7fbc3fSMichał Górny     value = pcb.x[reg - gpr_x0_arm64];
93*9c7fbc3fSMichał Górny     break;
94*9c7fbc3fSMichał Górny   case gpr_sp_arm64:
95*9c7fbc3fSMichał Górny     value = pcb.sp;
96*9c7fbc3fSMichał Górny     break;
97*9c7fbc3fSMichał Górny   case gpr_pc_arm64:
98*9c7fbc3fSMichał Górny     // The pc of crashing thread is stored in lr.
99*9c7fbc3fSMichał Górny     value = pcb.lr;
100*9c7fbc3fSMichał Górny     break;
101*9c7fbc3fSMichał Górny   default:
102*9c7fbc3fSMichał Górny     return false;
103*9c7fbc3fSMichał Górny   }
104*9c7fbc3fSMichał Górny   return true;
105*9c7fbc3fSMichał Górny }
106*9c7fbc3fSMichał Górny 
WriteRegister(const RegisterInfo * reg_info,const RegisterValue & value)107*9c7fbc3fSMichał Górny bool RegisterContextFreeBSDKernel_arm64::WriteRegister(
108*9c7fbc3fSMichał Górny     const RegisterInfo *reg_info, const RegisterValue &value) {
109*9c7fbc3fSMichał Górny   return false;
110*9c7fbc3fSMichał Górny }
111