xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/Process/minidump/RegisterContextMinidump_x86_64.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1dda28197Spatrick //===-- RegisterContextMinidump_x86_64.cpp --------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick 
9061da546Spatrick #include "RegisterContextMinidump_x86_64.h"
10061da546Spatrick 
11061da546Spatrick #include "lldb/Utility/DataBufferHeap.h"
12061da546Spatrick 
13061da546Spatrick // C includes
14061da546Spatrick // C++ includes
15061da546Spatrick 
16061da546Spatrick using namespace lldb_private;
17061da546Spatrick using namespace minidump;
18061da546Spatrick 
getDestRegister(uint8_t * context,const RegisterInfo & reg)19061da546Spatrick static llvm::MutableArrayRef<uint8_t> getDestRegister(uint8_t *context,
20061da546Spatrick                                                       const RegisterInfo &reg) {
21061da546Spatrick   auto bytes = reg.mutable_data(context);
22061da546Spatrick 
23061da546Spatrick   switch (reg.kinds[lldb::eRegisterKindLLDB]) {
24061da546Spatrick   case lldb_cs_x86_64:
25061da546Spatrick   case lldb_ds_x86_64:
26061da546Spatrick   case lldb_es_x86_64:
27061da546Spatrick   case lldb_fs_x86_64:
28061da546Spatrick   case lldb_gs_x86_64:
29061da546Spatrick   case lldb_ss_x86_64:
30061da546Spatrick     return bytes.take_front(2);
31061da546Spatrick     break;
32061da546Spatrick   case lldb_rflags_x86_64:
33061da546Spatrick     return bytes.take_front(4);
34061da546Spatrick     break;
35061da546Spatrick   default:
36061da546Spatrick     return bytes.take_front(8);
37061da546Spatrick     break;
38061da546Spatrick   }
39061da546Spatrick }
40061da546Spatrick 
writeRegister(const void * reg_src,uint8_t * context,const RegisterInfo & reg)41061da546Spatrick static void writeRegister(const void *reg_src, uint8_t *context,
42061da546Spatrick                           const RegisterInfo &reg) {
43061da546Spatrick   llvm::MutableArrayRef<uint8_t> reg_dest = getDestRegister(context, reg);
44061da546Spatrick   memcpy(reg_dest.data(), reg_src, reg_dest.size());
45061da546Spatrick }
46061da546Spatrick 
ConvertMinidumpContext_x86_64(llvm::ArrayRef<uint8_t> source_data,RegisterInfoInterface * target_reg_interface)47061da546Spatrick lldb::DataBufferSP lldb_private::minidump::ConvertMinidumpContext_x86_64(
48061da546Spatrick     llvm::ArrayRef<uint8_t> source_data,
49061da546Spatrick     RegisterInfoInterface *target_reg_interface) {
50061da546Spatrick 
51061da546Spatrick   const RegisterInfo *reg_info = target_reg_interface->GetRegisterInfo();
52061da546Spatrick 
53*f6aab3d8Srobert   lldb::WritableDataBufferSP result_context_buf(
54061da546Spatrick       new DataBufferHeap(target_reg_interface->GetGPRSize(), 0));
55061da546Spatrick   uint8_t *result_base = result_context_buf->GetBytes();
56061da546Spatrick 
57061da546Spatrick   if (source_data.size() < sizeof(MinidumpContext_x86_64))
58061da546Spatrick     return nullptr;
59061da546Spatrick 
60061da546Spatrick   const MinidumpContext_x86_64 *context;
61061da546Spatrick   consumeObject(source_data, context);
62061da546Spatrick 
63061da546Spatrick   const MinidumpContext_x86_64_Flags context_flags =
64061da546Spatrick       static_cast<MinidumpContext_x86_64_Flags>(
65061da546Spatrick           static_cast<uint32_t>(context->context_flags));
66061da546Spatrick   auto x86_64_Flag = MinidumpContext_x86_64_Flags::x86_64_Flag;
67061da546Spatrick   auto ControlFlag = MinidumpContext_x86_64_Flags::Control;
68061da546Spatrick   auto IntegerFlag = MinidumpContext_x86_64_Flags::Integer;
69061da546Spatrick   auto SegmentsFlag = MinidumpContext_x86_64_Flags::Segments;
70061da546Spatrick 
71061da546Spatrick   if ((context_flags & x86_64_Flag) != x86_64_Flag)
72061da546Spatrick     return nullptr;
73061da546Spatrick 
74061da546Spatrick   if ((context_flags & ControlFlag) == ControlFlag) {
75061da546Spatrick     writeRegister(&context->cs, result_base, reg_info[lldb_cs_x86_64]);
76061da546Spatrick     writeRegister(&context->ss, result_base, reg_info[lldb_ss_x86_64]);
77061da546Spatrick     writeRegister(&context->eflags, result_base, reg_info[lldb_rflags_x86_64]);
78061da546Spatrick     writeRegister(&context->rsp, result_base, reg_info[lldb_rsp_x86_64]);
79061da546Spatrick     writeRegister(&context->rip, result_base, reg_info[lldb_rip_x86_64]);
80061da546Spatrick   }
81061da546Spatrick 
82061da546Spatrick   if ((context_flags & SegmentsFlag) == SegmentsFlag) {
83061da546Spatrick     writeRegister(&context->ds, result_base, reg_info[lldb_ds_x86_64]);
84061da546Spatrick     writeRegister(&context->es, result_base, reg_info[lldb_es_x86_64]);
85061da546Spatrick     writeRegister(&context->fs, result_base, reg_info[lldb_fs_x86_64]);
86061da546Spatrick     writeRegister(&context->gs, result_base, reg_info[lldb_gs_x86_64]);
87061da546Spatrick   }
88061da546Spatrick 
89061da546Spatrick   if ((context_flags & IntegerFlag) == IntegerFlag) {
90061da546Spatrick     writeRegister(&context->rax, result_base, reg_info[lldb_rax_x86_64]);
91061da546Spatrick     writeRegister(&context->rcx, result_base, reg_info[lldb_rcx_x86_64]);
92061da546Spatrick     writeRegister(&context->rdx, result_base, reg_info[lldb_rdx_x86_64]);
93061da546Spatrick     writeRegister(&context->rbx, result_base, reg_info[lldb_rbx_x86_64]);
94061da546Spatrick     writeRegister(&context->rbp, result_base, reg_info[lldb_rbp_x86_64]);
95061da546Spatrick     writeRegister(&context->rsi, result_base, reg_info[lldb_rsi_x86_64]);
96061da546Spatrick     writeRegister(&context->rdi, result_base, reg_info[lldb_rdi_x86_64]);
97061da546Spatrick     writeRegister(&context->r8, result_base, reg_info[lldb_r8_x86_64]);
98061da546Spatrick     writeRegister(&context->r9, result_base, reg_info[lldb_r9_x86_64]);
99061da546Spatrick     writeRegister(&context->r10, result_base, reg_info[lldb_r10_x86_64]);
100061da546Spatrick     writeRegister(&context->r11, result_base, reg_info[lldb_r11_x86_64]);
101061da546Spatrick     writeRegister(&context->r12, result_base, reg_info[lldb_r12_x86_64]);
102061da546Spatrick     writeRegister(&context->r13, result_base, reg_info[lldb_r13_x86_64]);
103061da546Spatrick     writeRegister(&context->r14, result_base, reg_info[lldb_r14_x86_64]);
104061da546Spatrick     writeRegister(&context->r15, result_base, reg_info[lldb_r15_x86_64]);
105061da546Spatrick   }
106061da546Spatrick 
107061da546Spatrick   // TODO parse the floating point registers
108061da546Spatrick 
109061da546Spatrick   return result_context_buf;
110061da546Spatrick }
111