Coverage Report

Created: 2023-09-21 18:56

/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> &regs,
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> &regs,
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> &regs) {
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
}