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