xref: /openbsd-src/gnu/llvm/lldb/source/Plugins/ABI/X86/ABIX86.cpp (revision f6aab3d83b51b91c24247ad2c2573574de475a82)
1*f6aab3d8Srobert //===-- ABIX86.cpp --------------------------------------------------------===//
2dda28197Spatrick //
3dda28197Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4dda28197Spatrick // See https://llvm.org/LICENSE.txt for license information.
5dda28197Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6dda28197Spatrick //
7dda28197Spatrick //===----------------------------------------------------------------------===//
8dda28197Spatrick 
9dda28197Spatrick #include "ABIMacOSX_i386.h"
10dda28197Spatrick #include "ABISysV_i386.h"
11dda28197Spatrick #include "ABISysV_x86_64.h"
12dda28197Spatrick #include "ABIWindows_x86_64.h"
13*f6aab3d8Srobert #include "ABIX86.h"
14dda28197Spatrick #include "lldb/Core/PluginManager.h"
15*f6aab3d8Srobert #include "lldb/Target/Process.h"
16*f6aab3d8Srobert #include <optional>
17*f6aab3d8Srobert 
18*f6aab3d8Srobert using namespace lldb;
19*f6aab3d8Srobert using namespace lldb_private;
20dda28197Spatrick 
LLDB_PLUGIN_DEFINE(ABIX86)21dda28197Spatrick LLDB_PLUGIN_DEFINE(ABIX86)
22dda28197Spatrick 
23dda28197Spatrick void ABIX86::Initialize() {
24dda28197Spatrick   ABIMacOSX_i386::Initialize();
25dda28197Spatrick   ABISysV_i386::Initialize();
26dda28197Spatrick   ABISysV_x86_64::Initialize();
27dda28197Spatrick   ABIWindows_x86_64::Initialize();
28dda28197Spatrick }
29dda28197Spatrick 
Terminate()30dda28197Spatrick void ABIX86::Terminate() {
31dda28197Spatrick   ABIMacOSX_i386::Terminate();
32dda28197Spatrick   ABISysV_i386::Terminate();
33dda28197Spatrick   ABISysV_x86_64::Terminate();
34dda28197Spatrick   ABIWindows_x86_64::Terminate();
35dda28197Spatrick }
36dda28197Spatrick 
37*f6aab3d8Srobert namespace {
38*f6aab3d8Srobert enum RegKind {
39*f6aab3d8Srobert   GPR32,
40*f6aab3d8Srobert   GPR16,
41*f6aab3d8Srobert   GPR8h,
42*f6aab3d8Srobert   GPR8,
43*f6aab3d8Srobert   MM,
44*f6aab3d8Srobert   YMM_YMMh,
45*f6aab3d8Srobert   YMM_XMM,
46*f6aab3d8Srobert 
47*f6aab3d8Srobert   RegKindCount
48*f6aab3d8Srobert };
49*f6aab3d8Srobert }
50*f6aab3d8Srobert 
51*f6aab3d8Srobert struct RegData {
52*f6aab3d8Srobert   RegKind subreg_kind;
53*f6aab3d8Srobert   llvm::StringRef subreg_name;
54*f6aab3d8Srobert   std::optional<uint32_t> base_index;
55*f6aab3d8Srobert };
56*f6aab3d8Srobert 
57*f6aab3d8Srobert static void
addPartialRegisters(std::vector<DynamicRegisterInfo::Register> & regs,llvm::ArrayRef<RegData * > subregs,uint32_t base_size,lldb::Encoding encoding,lldb::Format format,uint32_t subreg_size,uint32_t subreg_offset=0)58*f6aab3d8Srobert addPartialRegisters(std::vector<DynamicRegisterInfo::Register> &regs,
59*f6aab3d8Srobert                     llvm::ArrayRef<RegData *> subregs, uint32_t base_size,
60*f6aab3d8Srobert                     lldb::Encoding encoding, lldb::Format format,
61*f6aab3d8Srobert                     uint32_t subreg_size, uint32_t subreg_offset = 0) {
62*f6aab3d8Srobert   for (const RegData *subreg : subregs) {
63*f6aab3d8Srobert     assert(subreg);
64*f6aab3d8Srobert     uint32_t base_index = *subreg->base_index;
65*f6aab3d8Srobert     DynamicRegisterInfo::Register &full_reg = regs[base_index];
66*f6aab3d8Srobert     if (full_reg.byte_size != base_size)
67*f6aab3d8Srobert       continue;
68*f6aab3d8Srobert 
69*f6aab3d8Srobert     lldb_private::DynamicRegisterInfo::Register new_reg{
70*f6aab3d8Srobert         lldb_private::ConstString(subreg->subreg_name),
71*f6aab3d8Srobert         lldb_private::ConstString(),
72*f6aab3d8Srobert         lldb_private::ConstString("supplementary registers"),
73*f6aab3d8Srobert         subreg_size,
74*f6aab3d8Srobert         LLDB_INVALID_INDEX32,
75*f6aab3d8Srobert         encoding,
76*f6aab3d8Srobert         format,
77*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
78*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
79*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
80*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
81*f6aab3d8Srobert         {base_index},
82*f6aab3d8Srobert         {},
83*f6aab3d8Srobert         subreg_offset};
84*f6aab3d8Srobert 
85*f6aab3d8Srobert     addSupplementaryRegister(regs, new_reg);
86*f6aab3d8Srobert   }
87*f6aab3d8Srobert }
88*f6aab3d8Srobert 
89*f6aab3d8Srobert static void
addCombinedRegisters(std::vector<DynamicRegisterInfo::Register> & regs,llvm::ArrayRef<RegData * > subregs1,llvm::ArrayRef<RegData * > subregs2,uint32_t base_size,lldb::Encoding encoding,lldb::Format format)90*f6aab3d8Srobert addCombinedRegisters(std::vector<DynamicRegisterInfo::Register> &regs,
91*f6aab3d8Srobert                      llvm::ArrayRef<RegData *> subregs1,
92*f6aab3d8Srobert                      llvm::ArrayRef<RegData *> subregs2, uint32_t base_size,
93*f6aab3d8Srobert                      lldb::Encoding encoding, lldb::Format format) {
94*f6aab3d8Srobert   for (auto it : llvm::zip(subregs1, subregs2)) {
95*f6aab3d8Srobert     RegData *regdata1, *regdata2;
96*f6aab3d8Srobert     std::tie(regdata1, regdata2) = it;
97*f6aab3d8Srobert     assert(regdata1);
98*f6aab3d8Srobert     assert(regdata2);
99*f6aab3d8Srobert 
100*f6aab3d8Srobert     // verify that we've got matching target registers
101*f6aab3d8Srobert     if (regdata1->subreg_name != regdata2->subreg_name)
102*f6aab3d8Srobert       continue;
103*f6aab3d8Srobert 
104*f6aab3d8Srobert     uint32_t base_index1 = *regdata1->base_index;
105*f6aab3d8Srobert     uint32_t base_index2 = *regdata2->base_index;
106*f6aab3d8Srobert     if (regs[base_index1].byte_size != base_size ||
107*f6aab3d8Srobert         regs[base_index2].byte_size != base_size)
108*f6aab3d8Srobert       continue;
109*f6aab3d8Srobert 
110*f6aab3d8Srobert     lldb_private::DynamicRegisterInfo::Register new_reg{
111*f6aab3d8Srobert         lldb_private::ConstString(regdata1->subreg_name),
112*f6aab3d8Srobert         lldb_private::ConstString(),
113*f6aab3d8Srobert         lldb_private::ConstString("supplementary registers"),
114*f6aab3d8Srobert         base_size * 2,
115*f6aab3d8Srobert         LLDB_INVALID_INDEX32,
116*f6aab3d8Srobert         encoding,
117*f6aab3d8Srobert         format,
118*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
119*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
120*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
121*f6aab3d8Srobert         LLDB_INVALID_REGNUM,
122*f6aab3d8Srobert         {base_index1, base_index2},
123*f6aab3d8Srobert         {}};
124*f6aab3d8Srobert 
125*f6aab3d8Srobert     addSupplementaryRegister(regs, new_reg);
126*f6aab3d8Srobert   }
127*f6aab3d8Srobert }
128*f6aab3d8Srobert 
129*f6aab3d8Srobert typedef llvm::SmallDenseMap<llvm::StringRef, llvm::SmallVector<RegData, 4>, 64>
130*f6aab3d8Srobert     BaseRegToRegsMap;
131*f6aab3d8Srobert 
132*f6aab3d8Srobert #define GPRh(l)                                                                \
133*f6aab3d8Srobert   {                                                                            \
134*f6aab3d8Srobert     is64bit ? BaseRegToRegsMap::value_type("r" l "x",                          \
135*f6aab3d8Srobert                                            {{GPR32, "e" l "x", std::nullopt},  \
136*f6aab3d8Srobert                                             {GPR16, l "x", std::nullopt},      \
137*f6aab3d8Srobert                                             {GPR8h, l "h", std::nullopt},      \
138*f6aab3d8Srobert                                             {GPR8, l "l", std::nullopt}})      \
139*f6aab3d8Srobert             : BaseRegToRegsMap::value_type("e" l "x",                          \
140*f6aab3d8Srobert                                            {{GPR16, l "x", std::nullopt},      \
141*f6aab3d8Srobert                                             {GPR8h, l "h", std::nullopt},      \
142*f6aab3d8Srobert                                             {GPR8, l "l", std::nullopt}})      \
143*f6aab3d8Srobert   }
144*f6aab3d8Srobert 
145*f6aab3d8Srobert #define GPR(r16)                                                               \
146*f6aab3d8Srobert   {                                                                            \
147*f6aab3d8Srobert     is64bit ? BaseRegToRegsMap::value_type("r" r16,                            \
148*f6aab3d8Srobert                                            {{GPR32, "e" r16, std::nullopt},    \
149*f6aab3d8Srobert                                             {GPR16, r16, std::nullopt},        \
150*f6aab3d8Srobert                                             {GPR8, r16 "l", std::nullopt}})    \
151*f6aab3d8Srobert             : BaseRegToRegsMap::value_type(                                    \
152*f6aab3d8Srobert                   "e" r16,                                                     \
153*f6aab3d8Srobert                   {{GPR16, r16, std::nullopt}, {GPR8, r16 "l", std::nullopt}}) \
154*f6aab3d8Srobert   }
155*f6aab3d8Srobert 
156*f6aab3d8Srobert #define GPR64(n)                                                               \
157*f6aab3d8Srobert   {                                                                            \
158*f6aab3d8Srobert     BaseRegToRegsMap::value_type("r" #n, {{GPR32, "r" #n "d", std::nullopt},   \
159*f6aab3d8Srobert                                           {GPR16, "r" #n "w", std::nullopt},   \
160*f6aab3d8Srobert                                           {GPR8, "r" #n "l", std::nullopt}})   \
161*f6aab3d8Srobert   }
162*f6aab3d8Srobert 
163*f6aab3d8Srobert #define STMM(n)                                                                \
164*f6aab3d8Srobert   { BaseRegToRegsMap::value_type("st" #n, {{MM, "mm" #n, std::nullopt}}) }
165*f6aab3d8Srobert 
166*f6aab3d8Srobert #define YMM(n)                                                                 \
167*f6aab3d8Srobert   {BaseRegToRegsMap::value_type("ymm" #n "h",                                  \
168*f6aab3d8Srobert                                 {{YMM_YMMh, "ymm" #n, std::nullopt}})},        \
169*f6aab3d8Srobert   {                                                                            \
170*f6aab3d8Srobert     BaseRegToRegsMap::value_type("xmm" #n,                                     \
171*f6aab3d8Srobert                                  {{YMM_XMM, "ymm" #n, std::nullopt}})          \
172*f6aab3d8Srobert   }
173*f6aab3d8Srobert 
makeBaseRegMap(bool is64bit)174*f6aab3d8Srobert BaseRegToRegsMap makeBaseRegMap(bool is64bit) {
175*f6aab3d8Srobert   BaseRegToRegsMap out{
176*f6aab3d8Srobert       {// GPRs common to amd64 & i386
177*f6aab3d8Srobert        GPRh("a"), GPRh("b"), GPRh("c"), GPRh("d"), GPR("si"), GPR("di"),
178*f6aab3d8Srobert        GPR("bp"), GPR("sp"),
179*f6aab3d8Srobert 
180*f6aab3d8Srobert        // ST/MM registers
181*f6aab3d8Srobert        STMM(0), STMM(1), STMM(2), STMM(3), STMM(4), STMM(5), STMM(6), STMM(7),
182*f6aab3d8Srobert 
183*f6aab3d8Srobert        // lower YMM registers (common to amd64 & i386)
184*f6aab3d8Srobert        YMM(0), YMM(1), YMM(2), YMM(3), YMM(4), YMM(5), YMM(6), YMM(7)}};
185*f6aab3d8Srobert 
186*f6aab3d8Srobert   if (is64bit) {
187*f6aab3d8Srobert     BaseRegToRegsMap amd64_regs{{// GPRs specific to amd64
188*f6aab3d8Srobert                                  GPR64(8), GPR64(9), GPR64(10), GPR64(11),
189*f6aab3d8Srobert                                  GPR64(12), GPR64(13), GPR64(14), GPR64(15),
190*f6aab3d8Srobert 
191*f6aab3d8Srobert                                  // higher YMM registers (specific to amd64)
192*f6aab3d8Srobert                                  YMM(8), YMM(9), YMM(10), YMM(11), YMM(12),
193*f6aab3d8Srobert                                  YMM(13), YMM(14), YMM(15)}};
194*f6aab3d8Srobert     out.insert(amd64_regs.begin(), amd64_regs.end());
195*f6aab3d8Srobert   }
196*f6aab3d8Srobert 
197*f6aab3d8Srobert   return out;
198*f6aab3d8Srobert }
199*f6aab3d8Srobert 
AugmentRegisterInfo(std::vector<DynamicRegisterInfo::Register> & regs)200*f6aab3d8Srobert void ABIX86::AugmentRegisterInfo(
201*f6aab3d8Srobert     std::vector<DynamicRegisterInfo::Register> &regs) {
202*f6aab3d8Srobert   MCBasedABI::AugmentRegisterInfo(regs);
203*f6aab3d8Srobert 
204*f6aab3d8Srobert   ProcessSP process_sp = GetProcessSP();
205*f6aab3d8Srobert   if (!process_sp)
206*f6aab3d8Srobert     return;
207*f6aab3d8Srobert 
208*f6aab3d8Srobert   uint32_t gpr_base_size =
209*f6aab3d8Srobert       process_sp->GetTarget().GetArchitecture().GetAddressByteSize();
210*f6aab3d8Srobert 
211*f6aab3d8Srobert   // primary map from a base register to its subregisters
212*f6aab3d8Srobert   BaseRegToRegsMap base_reg_map = makeBaseRegMap(gpr_base_size == 8);
213*f6aab3d8Srobert   // set used for fast matching of register names to subregisters
214*f6aab3d8Srobert   llvm::SmallDenseSet<llvm::StringRef, 64> subreg_name_set;
215*f6aab3d8Srobert   // convenience array providing access to all subregisters of given kind,
216*f6aab3d8Srobert   // sorted by base register index
217*f6aab3d8Srobert   std::array<llvm::SmallVector<RegData *, 16>, RegKindCount> subreg_by_kind;
218*f6aab3d8Srobert 
219*f6aab3d8Srobert   // prepare the set of all known subregisters
220*f6aab3d8Srobert   for (const auto &x : base_reg_map) {
221*f6aab3d8Srobert     for (const auto &subreg : x.second)
222*f6aab3d8Srobert       subreg_name_set.insert(subreg.subreg_name);
223*f6aab3d8Srobert   }
224*f6aab3d8Srobert 
225*f6aab3d8Srobert   // iterate over all registers
226*f6aab3d8Srobert   for (const auto &x : llvm::enumerate(regs)) {
227*f6aab3d8Srobert     llvm::StringRef reg_name = x.value().name.GetStringRef();
228*f6aab3d8Srobert     // abort if at least one sub-register is already present
229*f6aab3d8Srobert     if (llvm::is_contained(subreg_name_set, reg_name))
230*f6aab3d8Srobert       return;
231*f6aab3d8Srobert 
232*f6aab3d8Srobert     auto found = base_reg_map.find(reg_name);
233*f6aab3d8Srobert     if (found == base_reg_map.end())
234*f6aab3d8Srobert       continue;
235*f6aab3d8Srobert 
236*f6aab3d8Srobert     for (auto &subreg : found->second) {
237*f6aab3d8Srobert       // fill in base register indices
238*f6aab3d8Srobert       subreg.base_index = x.index();
239*f6aab3d8Srobert       // fill subreg_by_kind map-array
240*f6aab3d8Srobert       subreg_by_kind[static_cast<size_t>(subreg.subreg_kind)].push_back(
241*f6aab3d8Srobert           &subreg);
242*f6aab3d8Srobert     }
243*f6aab3d8Srobert   }
244*f6aab3d8Srobert 
245*f6aab3d8Srobert   // now add registers by kind
246*f6aab3d8Srobert   addPartialRegisters(regs, subreg_by_kind[GPR32], gpr_base_size, eEncodingUint,
247*f6aab3d8Srobert                       eFormatHex, 4);
248*f6aab3d8Srobert   addPartialRegisters(regs, subreg_by_kind[GPR16], gpr_base_size, eEncodingUint,
249*f6aab3d8Srobert                       eFormatHex, 2);
250*f6aab3d8Srobert   addPartialRegisters(regs, subreg_by_kind[GPR8h], gpr_base_size, eEncodingUint,
251*f6aab3d8Srobert                       eFormatHex, 1, 1);
252*f6aab3d8Srobert   addPartialRegisters(regs, subreg_by_kind[GPR8], gpr_base_size, eEncodingUint,
253*f6aab3d8Srobert                       eFormatHex, 1);
254*f6aab3d8Srobert 
255*f6aab3d8Srobert   addPartialRegisters(regs, subreg_by_kind[MM], 10, eEncodingUint, eFormatHex,
256*f6aab3d8Srobert                       8);
257*f6aab3d8Srobert 
258*f6aab3d8Srobert   addCombinedRegisters(regs, subreg_by_kind[YMM_XMM], subreg_by_kind[YMM_YMMh],
259*f6aab3d8Srobert                        16, eEncodingVector, eFormatVectorOfUInt8);
260dda28197Spatrick }
261