Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/freebsd-src
Path: blob/main/contrib/llvm-project/llvm/lib/ExecutionEngine/JITLink/MachO_x86_64.cpp
35271 views
1
//===---- MachO_x86_64.cpp -JIT linker implementation for MachO/x86-64 ----===//
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
// MachO/x86-64 jit-link implementation.
10
//
11
//===----------------------------------------------------------------------===//
12
13
#include "llvm/ExecutionEngine/JITLink/MachO_x86_64.h"
14
#include "llvm/ExecutionEngine/JITLink/DWARFRecordSectionSplitter.h"
15
#include "llvm/ExecutionEngine/JITLink/x86_64.h"
16
17
#include "DefineExternalSectionStartAndEndSymbols.h"
18
#include "MachOLinkGraphBuilder.h"
19
20
#define DEBUG_TYPE "jitlink"
21
22
using namespace llvm;
23
using namespace llvm::jitlink;
24
25
namespace {
26
27
class MachOLinkGraphBuilder_x86_64 : public MachOLinkGraphBuilder {
28
public:
29
MachOLinkGraphBuilder_x86_64(const object::MachOObjectFile &Obj,
30
SubtargetFeatures Features)
31
: MachOLinkGraphBuilder(Obj, Triple("x86_64-apple-darwin"),
32
std::move(Features), x86_64::getEdgeKindName) {}
33
34
private:
35
enum MachONormalizedRelocationType : unsigned {
36
MachOBranch32,
37
MachOPointer32,
38
MachOPointer64,
39
MachOPointer64Anon,
40
MachOPCRel32,
41
MachOPCRel32Minus1,
42
MachOPCRel32Minus2,
43
MachOPCRel32Minus4,
44
MachOPCRel32Anon,
45
MachOPCRel32Minus1Anon,
46
MachOPCRel32Minus2Anon,
47
MachOPCRel32Minus4Anon,
48
MachOPCRel32GOTLoad,
49
MachOPCRel32GOT,
50
MachOPCRel32TLV,
51
MachOSubtractor32,
52
MachOSubtractor64,
53
};
54
55
static Expected<MachONormalizedRelocationType>
56
getRelocKind(const MachO::relocation_info &RI) {
57
switch (RI.r_type) {
58
case MachO::X86_64_RELOC_UNSIGNED:
59
if (!RI.r_pcrel) {
60
if (RI.r_length == 3)
61
return RI.r_extern ? MachOPointer64 : MachOPointer64Anon;
62
else if (RI.r_extern && RI.r_length == 2)
63
return MachOPointer32;
64
}
65
break;
66
case MachO::X86_64_RELOC_SIGNED:
67
if (RI.r_pcrel && RI.r_length == 2)
68
return RI.r_extern ? MachOPCRel32 : MachOPCRel32Anon;
69
break;
70
case MachO::X86_64_RELOC_BRANCH:
71
if (RI.r_pcrel && RI.r_extern && RI.r_length == 2)
72
return MachOBranch32;
73
break;
74
case MachO::X86_64_RELOC_GOT_LOAD:
75
if (RI.r_pcrel && RI.r_extern && RI.r_length == 2)
76
return MachOPCRel32GOTLoad;
77
break;
78
case MachO::X86_64_RELOC_GOT:
79
if (RI.r_pcrel && RI.r_extern && RI.r_length == 2)
80
return MachOPCRel32GOT;
81
break;
82
case MachO::X86_64_RELOC_SUBTRACTOR:
83
if (!RI.r_pcrel && RI.r_extern) {
84
if (RI.r_length == 2)
85
return MachOSubtractor32;
86
else if (RI.r_length == 3)
87
return MachOSubtractor64;
88
}
89
break;
90
case MachO::X86_64_RELOC_SIGNED_1:
91
if (RI.r_pcrel && RI.r_length == 2)
92
return RI.r_extern ? MachOPCRel32Minus1 : MachOPCRel32Minus1Anon;
93
break;
94
case MachO::X86_64_RELOC_SIGNED_2:
95
if (RI.r_pcrel && RI.r_length == 2)
96
return RI.r_extern ? MachOPCRel32Minus2 : MachOPCRel32Minus2Anon;
97
break;
98
case MachO::X86_64_RELOC_SIGNED_4:
99
if (RI.r_pcrel && RI.r_length == 2)
100
return RI.r_extern ? MachOPCRel32Minus4 : MachOPCRel32Minus4Anon;
101
break;
102
case MachO::X86_64_RELOC_TLV:
103
if (RI.r_pcrel && RI.r_extern && RI.r_length == 2)
104
return MachOPCRel32TLV;
105
break;
106
}
107
108
return make_error<JITLinkError>(
109
"Unsupported x86-64 relocation: address=" +
110
formatv("{0:x8}", RI.r_address) +
111
", symbolnum=" + formatv("{0:x6}", RI.r_symbolnum) +
112
", kind=" + formatv("{0:x1}", RI.r_type) +
113
", pc_rel=" + (RI.r_pcrel ? "true" : "false") +
114
", extern=" + (RI.r_extern ? "true" : "false") +
115
", length=" + formatv("{0:d}", RI.r_length));
116
}
117
118
using PairRelocInfo = std::tuple<Edge::Kind, Symbol *, uint64_t>;
119
120
// Parses paired SUBTRACTOR/UNSIGNED relocations and, on success,
121
// returns the edge kind and addend to be used.
122
Expected<PairRelocInfo> parsePairRelocation(
123
Block &BlockToFix, MachONormalizedRelocationType SubtractorKind,
124
const MachO::relocation_info &SubRI, orc::ExecutorAddr FixupAddress,
125
const char *FixupContent, object::relocation_iterator &UnsignedRelItr,
126
object::relocation_iterator &RelEnd) {
127
using namespace support;
128
129
assert(((SubtractorKind == MachOSubtractor32 && SubRI.r_length == 2) ||
130
(SubtractorKind == MachOSubtractor64 && SubRI.r_length == 3)) &&
131
"Subtractor kind should match length");
132
assert(SubRI.r_extern && "SUBTRACTOR reloc symbol should be extern");
133
assert(!SubRI.r_pcrel && "SUBTRACTOR reloc should not be PCRel");
134
135
if (UnsignedRelItr == RelEnd)
136
return make_error<JITLinkError>("x86_64 SUBTRACTOR without paired "
137
"UNSIGNED relocation");
138
139
auto UnsignedRI = getRelocationInfo(UnsignedRelItr);
140
141
if (SubRI.r_address != UnsignedRI.r_address)
142
return make_error<JITLinkError>("x86_64 SUBTRACTOR and paired UNSIGNED "
143
"point to different addresses");
144
145
if (SubRI.r_length != UnsignedRI.r_length)
146
return make_error<JITLinkError>("length of x86_64 SUBTRACTOR and paired "
147
"UNSIGNED reloc must match");
148
149
Symbol *FromSymbol;
150
if (auto FromSymbolOrErr = findSymbolByIndex(SubRI.r_symbolnum))
151
FromSymbol = FromSymbolOrErr->GraphSymbol;
152
else
153
return FromSymbolOrErr.takeError();
154
155
// Read the current fixup value.
156
uint64_t FixupValue = 0;
157
if (SubRI.r_length == 3)
158
FixupValue = *(const little64_t *)FixupContent;
159
else
160
FixupValue = *(const little32_t *)FixupContent;
161
162
// Find 'ToSymbol' using symbol number or address, depending on whether the
163
// paired UNSIGNED relocation is extern.
164
Symbol *ToSymbol = nullptr;
165
if (UnsignedRI.r_extern) {
166
// Find target symbol by symbol index.
167
if (auto ToSymbolOrErr = findSymbolByIndex(UnsignedRI.r_symbolnum))
168
ToSymbol = ToSymbolOrErr->GraphSymbol;
169
else
170
return ToSymbolOrErr.takeError();
171
} else {
172
auto ToSymbolSec = findSectionByIndex(UnsignedRI.r_symbolnum - 1);
173
if (!ToSymbolSec)
174
return ToSymbolSec.takeError();
175
ToSymbol = getSymbolByAddress(*ToSymbolSec, ToSymbolSec->Address);
176
assert(ToSymbol && "No symbol for section");
177
FixupValue -= ToSymbol->getAddress().getValue();
178
}
179
180
Edge::Kind DeltaKind;
181
Symbol *TargetSymbol;
182
uint64_t Addend;
183
184
bool FixingFromSymbol = true;
185
if (&BlockToFix == &FromSymbol->getAddressable()) {
186
if (LLVM_UNLIKELY(&BlockToFix == &ToSymbol->getAddressable())) {
187
// From and To are symbols in the same block. Decide direction by offset
188
// instead.
189
if (ToSymbol->getAddress() > FixupAddress)
190
FixingFromSymbol = true;
191
else if (FromSymbol->getAddress() > FixupAddress)
192
FixingFromSymbol = false;
193
else
194
FixingFromSymbol = FromSymbol->getAddress() >= ToSymbol->getAddress();
195
} else
196
FixingFromSymbol = true;
197
} else {
198
if (&BlockToFix == &ToSymbol->getAddressable())
199
FixingFromSymbol = false;
200
else {
201
// BlockToFix was neither FromSymbol nor ToSymbol.
202
return make_error<JITLinkError>("SUBTRACTOR relocation must fix up "
203
"either 'A' or 'B' (or a symbol in one "
204
"of their alt-entry groups)");
205
}
206
}
207
208
if (FixingFromSymbol) {
209
TargetSymbol = ToSymbol;
210
DeltaKind = (SubRI.r_length == 3) ? x86_64::Delta64 : x86_64::Delta32;
211
Addend = FixupValue + (FixupAddress - FromSymbol->getAddress());
212
// FIXME: handle extern 'from'.
213
} else {
214
TargetSymbol = FromSymbol;
215
DeltaKind =
216
(SubRI.r_length == 3) ? x86_64::NegDelta64 : x86_64::NegDelta32;
217
Addend = FixupValue - (FixupAddress - ToSymbol->getAddress());
218
}
219
220
return PairRelocInfo(DeltaKind, TargetSymbol, Addend);
221
}
222
223
Error addRelocations() override {
224
using namespace support;
225
auto &Obj = getObject();
226
227
LLVM_DEBUG(dbgs() << "Processing relocations:\n");
228
229
for (const auto &S : Obj.sections()) {
230
231
orc::ExecutorAddr SectionAddress(S.getAddress());
232
233
// Skip relocations virtual sections.
234
if (S.isVirtual()) {
235
if (S.relocation_begin() != S.relocation_end())
236
return make_error<JITLinkError>("Virtual section contains "
237
"relocations");
238
continue;
239
}
240
241
auto NSec =
242
findSectionByIndex(Obj.getSectionIndex(S.getRawDataRefImpl()));
243
if (!NSec)
244
return NSec.takeError();
245
246
// Skip relocations for MachO sections without corresponding graph
247
// sections.
248
{
249
if (!NSec->GraphSection) {
250
LLVM_DEBUG({
251
dbgs() << " Skipping relocations for MachO section "
252
<< NSec->SegName << "/" << NSec->SectName
253
<< " which has no associated graph section\n";
254
});
255
continue;
256
}
257
}
258
259
// Add relocations for section.
260
for (auto RelItr = S.relocation_begin(), RelEnd = S.relocation_end();
261
RelItr != RelEnd; ++RelItr) {
262
263
MachO::relocation_info RI = getRelocationInfo(RelItr);
264
265
// Find the address of the value to fix up.
266
auto FixupAddress = SectionAddress + (uint32_t)RI.r_address;
267
268
LLVM_DEBUG({
269
dbgs() << " " << NSec->SectName << " + "
270
<< formatv("{0:x8}", RI.r_address) << ":\n";
271
});
272
273
// Find the block that the fixup points to.
274
Block *BlockToFix = nullptr;
275
{
276
auto SymbolToFixOrErr = findSymbolByAddress(*NSec, FixupAddress);
277
if (!SymbolToFixOrErr)
278
return SymbolToFixOrErr.takeError();
279
BlockToFix = &SymbolToFixOrErr->getBlock();
280
}
281
282
if (FixupAddress + orc::ExecutorAddrDiff(1ULL << RI.r_length) >
283
BlockToFix->getAddress() + BlockToFix->getContent().size())
284
return make_error<JITLinkError>(
285
"Relocation extends past end of fixup block");
286
287
// Get a pointer to the fixup content.
288
const char *FixupContent = BlockToFix->getContent().data() +
289
(FixupAddress - BlockToFix->getAddress());
290
291
size_t FixupOffset = FixupAddress - BlockToFix->getAddress();
292
293
// The target symbol and addend will be populated by the switch below.
294
Symbol *TargetSymbol = nullptr;
295
uint64_t Addend = 0;
296
297
// Validate the relocation kind.
298
auto MachORelocKind = getRelocKind(RI);
299
if (!MachORelocKind)
300
return MachORelocKind.takeError();
301
302
Edge::Kind Kind = Edge::Invalid;
303
304
switch (*MachORelocKind) {
305
case MachOBranch32:
306
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
307
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
308
else
309
return TargetSymbolOrErr.takeError();
310
Addend = *(const little32_t *)FixupContent;
311
Kind = x86_64::BranchPCRel32;
312
break;
313
case MachOPCRel32:
314
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
315
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
316
else
317
return TargetSymbolOrErr.takeError();
318
Addend = *(const little32_t *)FixupContent - 4;
319
Kind = x86_64::Delta32;
320
break;
321
case MachOPCRel32GOTLoad:
322
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
323
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
324
else
325
return TargetSymbolOrErr.takeError();
326
Addend = *(const little32_t *)FixupContent;
327
Kind = x86_64::RequestGOTAndTransformToPCRel32GOTLoadREXRelaxable;
328
if (FixupOffset < 3)
329
return make_error<JITLinkError>("GOTLD at invalid offset " +
330
formatv("{0}", FixupOffset));
331
break;
332
case MachOPCRel32GOT:
333
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
334
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
335
else
336
return TargetSymbolOrErr.takeError();
337
Addend = *(const little32_t *)FixupContent - 4;
338
Kind = x86_64::RequestGOTAndTransformToDelta32;
339
break;
340
case MachOPCRel32TLV:
341
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
342
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
343
else
344
return TargetSymbolOrErr.takeError();
345
Addend = *(const little32_t *)FixupContent;
346
Kind = x86_64::RequestTLVPAndTransformToPCRel32TLVPLoadREXRelaxable;
347
if (FixupOffset < 3)
348
return make_error<JITLinkError>("TLV at invalid offset " +
349
formatv("{0}", FixupOffset));
350
break;
351
case MachOPointer32:
352
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
353
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
354
else
355
return TargetSymbolOrErr.takeError();
356
Addend = *(const ulittle32_t *)FixupContent;
357
Kind = x86_64::Pointer32;
358
break;
359
case MachOPointer64:
360
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
361
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
362
else
363
return TargetSymbolOrErr.takeError();
364
Addend = *(const ulittle64_t *)FixupContent;
365
Kind = x86_64::Pointer64;
366
break;
367
case MachOPointer64Anon: {
368
orc::ExecutorAddr TargetAddress(*(const ulittle64_t *)FixupContent);
369
auto TargetNSec = findSectionByIndex(RI.r_symbolnum - 1);
370
if (!TargetNSec)
371
return TargetNSec.takeError();
372
if (auto TargetSymbolOrErr =
373
findSymbolByAddress(*TargetNSec, TargetAddress))
374
TargetSymbol = &*TargetSymbolOrErr;
375
else
376
return TargetSymbolOrErr.takeError();
377
Addend = TargetAddress - TargetSymbol->getAddress();
378
Kind = x86_64::Pointer64;
379
break;
380
}
381
case MachOPCRel32Minus1:
382
case MachOPCRel32Minus2:
383
case MachOPCRel32Minus4:
384
if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum))
385
TargetSymbol = TargetSymbolOrErr->GraphSymbol;
386
else
387
return TargetSymbolOrErr.takeError();
388
Addend = *(const little32_t *)FixupContent - 4;
389
Kind = x86_64::Delta32;
390
break;
391
case MachOPCRel32Anon: {
392
orc::ExecutorAddr TargetAddress(FixupAddress + 4 +
393
*(const little32_t *)FixupContent);
394
auto TargetNSec = findSectionByIndex(RI.r_symbolnum - 1);
395
if (!TargetNSec)
396
return TargetNSec.takeError();
397
if (auto TargetSymbolOrErr =
398
findSymbolByAddress(*TargetNSec, TargetAddress))
399
TargetSymbol = &*TargetSymbolOrErr;
400
else
401
return TargetSymbolOrErr.takeError();
402
Addend = TargetAddress - TargetSymbol->getAddress() - 4;
403
Kind = x86_64::Delta32;
404
break;
405
}
406
case MachOPCRel32Minus1Anon:
407
case MachOPCRel32Minus2Anon:
408
case MachOPCRel32Minus4Anon: {
409
orc::ExecutorAddrDiff Delta =
410
4 + orc::ExecutorAddrDiff(
411
1ULL << (*MachORelocKind - MachOPCRel32Minus1Anon));
412
orc::ExecutorAddr TargetAddress =
413
FixupAddress + Delta + *(const little32_t *)FixupContent;
414
auto TargetNSec = findSectionByIndex(RI.r_symbolnum - 1);
415
if (!TargetNSec)
416
return TargetNSec.takeError();
417
if (auto TargetSymbolOrErr =
418
findSymbolByAddress(*TargetNSec, TargetAddress))
419
TargetSymbol = &*TargetSymbolOrErr;
420
else
421
return TargetSymbolOrErr.takeError();
422
Addend = TargetAddress - TargetSymbol->getAddress() - Delta;
423
Kind = x86_64::Delta32;
424
break;
425
}
426
case MachOSubtractor32:
427
case MachOSubtractor64: {
428
// We use Delta32/Delta64 to represent SUBTRACTOR relocations.
429
// parsePairRelocation handles the paired reloc, and returns the
430
// edge kind to be used (either Delta32/Delta64, or
431
// NegDelta32/NegDelta64, depending on the direction of the
432
// subtraction) along with the addend.
433
auto PairInfo =
434
parsePairRelocation(*BlockToFix, *MachORelocKind, RI,
435
FixupAddress, FixupContent, ++RelItr, RelEnd);
436
if (!PairInfo)
437
return PairInfo.takeError();
438
std::tie(Kind, TargetSymbol, Addend) = *PairInfo;
439
assert(TargetSymbol && "No target symbol from parsePairRelocation?");
440
break;
441
}
442
}
443
444
LLVM_DEBUG({
445
dbgs() << " ";
446
Edge GE(Kind, FixupAddress - BlockToFix->getAddress(), *TargetSymbol,
447
Addend);
448
printEdge(dbgs(), *BlockToFix, GE, x86_64::getEdgeKindName(Kind));
449
dbgs() << "\n";
450
});
451
BlockToFix->addEdge(Kind, FixupAddress - BlockToFix->getAddress(),
452
*TargetSymbol, Addend);
453
}
454
}
455
return Error::success();
456
}
457
};
458
459
Error buildGOTAndStubs_MachO_x86_64(LinkGraph &G) {
460
x86_64::GOTTableManager GOT;
461
x86_64::PLTTableManager PLT(GOT);
462
visitExistingEdges(G, GOT, PLT);
463
return Error::success();
464
}
465
466
} // namespace
467
468
namespace llvm {
469
namespace jitlink {
470
471
class MachOJITLinker_x86_64 : public JITLinker<MachOJITLinker_x86_64> {
472
friend class JITLinker<MachOJITLinker_x86_64>;
473
474
public:
475
MachOJITLinker_x86_64(std::unique_ptr<JITLinkContext> Ctx,
476
std::unique_ptr<LinkGraph> G,
477
PassConfiguration PassConfig)
478
: JITLinker(std::move(Ctx), std::move(G), std::move(PassConfig)) {}
479
480
private:
481
Error applyFixup(LinkGraph &G, Block &B, const Edge &E) const {
482
return x86_64::applyFixup(G, B, E, nullptr);
483
}
484
};
485
486
Expected<std::unique_ptr<LinkGraph>>
487
createLinkGraphFromMachOObject_x86_64(MemoryBufferRef ObjectBuffer) {
488
auto MachOObj = object::ObjectFile::createMachOObjectFile(ObjectBuffer);
489
if (!MachOObj)
490
return MachOObj.takeError();
491
492
auto Features = (*MachOObj)->getFeatures();
493
if (!Features)
494
return Features.takeError();
495
496
return MachOLinkGraphBuilder_x86_64(**MachOObj, std::move(*Features))
497
.buildGraph();
498
}
499
500
void link_MachO_x86_64(std::unique_ptr<LinkGraph> G,
501
std::unique_ptr<JITLinkContext> Ctx) {
502
503
PassConfiguration Config;
504
505
if (Ctx->shouldAddDefaultTargetPasses(G->getTargetTriple())) {
506
// Add eh-frame passes.
507
Config.PrePrunePasses.push_back(createEHFrameSplitterPass_MachO_x86_64());
508
Config.PrePrunePasses.push_back(createEHFrameEdgeFixerPass_MachO_x86_64());
509
510
// Add compact unwind splitter pass.
511
Config.PrePrunePasses.push_back(
512
CompactUnwindSplitter("__LD,__compact_unwind"));
513
514
// Add a mark-live pass.
515
if (auto MarkLive = Ctx->getMarkLivePass(G->getTargetTriple()))
516
Config.PrePrunePasses.push_back(std::move(MarkLive));
517
else
518
Config.PrePrunePasses.push_back(markAllSymbolsLive);
519
520
// Resolve any external section start / end symbols.
521
Config.PostAllocationPasses.push_back(
522
createDefineExternalSectionStartAndEndSymbolsPass(
523
identifyMachOSectionStartAndEndSymbols));
524
525
// Add an in-place GOT/Stubs pass.
526
Config.PostPrunePasses.push_back(buildGOTAndStubs_MachO_x86_64);
527
528
// Add GOT/Stubs optimizer pass.
529
Config.PreFixupPasses.push_back(x86_64::optimizeGOTAndStubAccesses);
530
}
531
532
if (auto Err = Ctx->modifyPassConfig(*G, Config))
533
return Ctx->notifyFailed(std::move(Err));
534
535
// Construct a JITLinker and run the link function.
536
MachOJITLinker_x86_64::link(std::move(Ctx), std::move(G), std::move(Config));
537
}
538
539
LinkGraphPassFunction createEHFrameSplitterPass_MachO_x86_64() {
540
return DWARFRecordSectionSplitter("__TEXT,__eh_frame");
541
}
542
543
LinkGraphPassFunction createEHFrameEdgeFixerPass_MachO_x86_64() {
544
return EHFrameEdgeFixer("__TEXT,__eh_frame", x86_64::PointerSize,
545
x86_64::Pointer32, x86_64::Pointer64, x86_64::Delta32,
546
x86_64::Delta64, x86_64::NegDelta32);
547
}
548
549
} // end namespace jitlink
550
} // end namespace llvm
551
552