1 //===---- MachO_arm64.cpp - JIT linker implementation for MachO/arm64 -----===// 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/arm64 jit-link implementation. 10 // 11 //===----------------------------------------------------------------------===// 12 13 #include "llvm/ExecutionEngine/JITLink/MachO_arm64.h" 14 #include "llvm/ExecutionEngine/JITLink/DWARFRecordSectionSplitter.h" 15 #include "llvm/ExecutionEngine/JITLink/aarch64.h" 16 17 #include "MachOLinkGraphBuilder.h" 18 19 #define DEBUG_TYPE "jitlink" 20 21 using namespace llvm; 22 using namespace llvm::jitlink; 23 24 namespace { 25 26 class MachOLinkGraphBuilder_arm64 : public MachOLinkGraphBuilder { 27 public: 28 MachOLinkGraphBuilder_arm64(const object::MachOObjectFile &Obj) 29 : MachOLinkGraphBuilder(Obj, Triple("arm64-apple-darwin"), 30 aarch64::getEdgeKindName), 31 NumSymbols(Obj.getSymtabLoadCommand().nsyms) {} 32 33 private: 34 enum MachOARM64RelocationKind : Edge::Kind { 35 MachOBranch26 = Edge::FirstRelocation, 36 MachOPointer32, 37 MachOPointer64, 38 MachOPointer64Anon, 39 MachOPage21, 40 MachOPageOffset12, 41 MachOGOTPage21, 42 MachOGOTPageOffset12, 43 MachOTLVPage21, 44 MachOTLVPageOffset12, 45 MachOPointerToGOT, 46 MachOPairedAddend, 47 MachOLDRLiteral19, 48 MachODelta32, 49 MachODelta64, 50 MachONegDelta32, 51 MachONegDelta64, 52 }; 53 54 static Expected<MachOARM64RelocationKind> 55 getRelocationKind(const MachO::relocation_info &RI) { 56 switch (RI.r_type) { 57 case MachO::ARM64_RELOC_UNSIGNED: 58 if (!RI.r_pcrel) { 59 if (RI.r_length == 3) 60 return RI.r_extern ? MachOPointer64 : MachOPointer64Anon; 61 else if (RI.r_length == 2) 62 return MachOPointer32; 63 } 64 break; 65 case MachO::ARM64_RELOC_SUBTRACTOR: 66 // SUBTRACTOR must be non-pc-rel, extern, with length 2 or 3. 67 // Initially represent SUBTRACTOR relocations with 'Delta<W>'. 68 // They may be turned into NegDelta<W> by parsePairRelocation. 69 if (!RI.r_pcrel && RI.r_extern) { 70 if (RI.r_length == 2) 71 return MachODelta32; 72 else if (RI.r_length == 3) 73 return MachODelta64; 74 } 75 break; 76 case MachO::ARM64_RELOC_BRANCH26: 77 if (RI.r_pcrel && RI.r_extern && RI.r_length == 2) 78 return MachOBranch26; 79 break; 80 case MachO::ARM64_RELOC_PAGE21: 81 if (RI.r_pcrel && RI.r_extern && RI.r_length == 2) 82 return MachOPage21; 83 break; 84 case MachO::ARM64_RELOC_PAGEOFF12: 85 if (!RI.r_pcrel && RI.r_extern && RI.r_length == 2) 86 return MachOPageOffset12; 87 break; 88 case MachO::ARM64_RELOC_GOT_LOAD_PAGE21: 89 if (RI.r_pcrel && RI.r_extern && RI.r_length == 2) 90 return MachOGOTPage21; 91 break; 92 case MachO::ARM64_RELOC_GOT_LOAD_PAGEOFF12: 93 if (!RI.r_pcrel && RI.r_extern && RI.r_length == 2) 94 return MachOGOTPageOffset12; 95 break; 96 case MachO::ARM64_RELOC_POINTER_TO_GOT: 97 if (RI.r_pcrel && RI.r_extern && RI.r_length == 2) 98 return MachOPointerToGOT; 99 break; 100 case MachO::ARM64_RELOC_ADDEND: 101 if (!RI.r_pcrel && !RI.r_extern && RI.r_length == 2) 102 return MachOPairedAddend; 103 break; 104 case MachO::ARM64_RELOC_TLVP_LOAD_PAGE21: 105 if (RI.r_pcrel && RI.r_extern && RI.r_length == 2) 106 return MachOTLVPage21; 107 break; 108 case MachO::ARM64_RELOC_TLVP_LOAD_PAGEOFF12: 109 if (!RI.r_pcrel && RI.r_extern && RI.r_length == 2) 110 return MachOTLVPageOffset12; 111 break; 112 } 113 114 return make_error<JITLinkError>( 115 "Unsupported arm64 relocation: address=" + 116 formatv("{0:x8}", RI.r_address) + 117 ", symbolnum=" + formatv("{0:x6}", RI.r_symbolnum) + 118 ", kind=" + formatv("{0:x1}", RI.r_type) + 119 ", pc_rel=" + (RI.r_pcrel ? "true" : "false") + 120 ", extern=" + (RI.r_extern ? "true" : "false") + 121 ", length=" + formatv("{0:d}", RI.r_length)); 122 } 123 124 using PairRelocInfo = std::tuple<Edge::Kind, Symbol *, uint64_t>; 125 126 // Parses paired SUBTRACTOR/UNSIGNED relocations and, on success, 127 // returns the edge kind and addend to be used. 128 Expected<PairRelocInfo> 129 parsePairRelocation(Block &BlockToFix, Edge::Kind SubtractorKind, 130 const MachO::relocation_info &SubRI, 131 orc::ExecutorAddr FixupAddress, const char *FixupContent, 132 object::relocation_iterator &UnsignedRelItr, 133 object::relocation_iterator &RelEnd) { 134 using namespace support; 135 136 assert(((SubtractorKind == MachODelta32 && SubRI.r_length == 2) || 137 (SubtractorKind == MachODelta64 && SubRI.r_length == 3)) && 138 "Subtractor kind should match length"); 139 assert(SubRI.r_extern && "SUBTRACTOR reloc symbol should be extern"); 140 assert(!SubRI.r_pcrel && "SUBTRACTOR reloc should not be PCRel"); 141 142 if (UnsignedRelItr == RelEnd) 143 return make_error<JITLinkError>("arm64 SUBTRACTOR without paired " 144 "UNSIGNED relocation"); 145 146 auto UnsignedRI = getRelocationInfo(UnsignedRelItr); 147 148 if (SubRI.r_address != UnsignedRI.r_address) 149 return make_error<JITLinkError>("arm64 SUBTRACTOR and paired UNSIGNED " 150 "point to different addresses"); 151 152 if (SubRI.r_length != UnsignedRI.r_length) 153 return make_error<JITLinkError>("length of arm64 SUBTRACTOR and paired " 154 "UNSIGNED reloc must match"); 155 156 Symbol *FromSymbol; 157 if (auto FromSymbolOrErr = findSymbolByIndex(SubRI.r_symbolnum)) 158 FromSymbol = FromSymbolOrErr->GraphSymbol; 159 else 160 return FromSymbolOrErr.takeError(); 161 162 // Read the current fixup value. 163 uint64_t FixupValue = 0; 164 if (SubRI.r_length == 3) 165 FixupValue = *(const little64_t *)FixupContent; 166 else 167 FixupValue = *(const little32_t *)FixupContent; 168 169 // Find 'ToSymbol' using symbol number or address, depending on whether the 170 // paired UNSIGNED relocation is extern. 171 Symbol *ToSymbol = nullptr; 172 if (UnsignedRI.r_extern) { 173 // Find target symbol by symbol index. 174 if (auto ToSymbolOrErr = findSymbolByIndex(UnsignedRI.r_symbolnum)) 175 ToSymbol = ToSymbolOrErr->GraphSymbol; 176 else 177 return ToSymbolOrErr.takeError(); 178 } else { 179 auto ToSymbolSec = findSectionByIndex(UnsignedRI.r_symbolnum - 1); 180 if (!ToSymbolSec) 181 return ToSymbolSec.takeError(); 182 ToSymbol = getSymbolByAddress(*ToSymbolSec, ToSymbolSec->Address); 183 assert(ToSymbol && "No symbol for section"); 184 FixupValue -= ToSymbol->getAddress().getValue(); 185 } 186 187 Edge::Kind DeltaKind; 188 Symbol *TargetSymbol; 189 uint64_t Addend; 190 if (&BlockToFix == &FromSymbol->getAddressable()) { 191 TargetSymbol = ToSymbol; 192 DeltaKind = (SubRI.r_length == 3) ? aarch64::Delta64 : aarch64::Delta32; 193 Addend = FixupValue + (FixupAddress - FromSymbol->getAddress()); 194 // FIXME: handle extern 'from'. 195 } else if (&BlockToFix == &ToSymbol->getAddressable()) { 196 TargetSymbol = &*FromSymbol; 197 DeltaKind = 198 (SubRI.r_length == 3) ? aarch64::NegDelta64 : aarch64::NegDelta32; 199 Addend = FixupValue - (FixupAddress - ToSymbol->getAddress()); 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 return PairRelocInfo(DeltaKind, TargetSymbol, Addend); 208 } 209 210 Error addRelocations() override { 211 using namespace support; 212 auto &Obj = getObject(); 213 214 LLVM_DEBUG(dbgs() << "Processing relocations:\n"); 215 216 for (auto &S : Obj.sections()) { 217 218 orc::ExecutorAddr SectionAddress(S.getAddress()); 219 220 // Skip relocations virtual sections. 221 if (S.isVirtual()) { 222 if (S.relocation_begin() != S.relocation_end()) 223 return make_error<JITLinkError>("Virtual section contains " 224 "relocations"); 225 continue; 226 } 227 228 auto NSec = 229 findSectionByIndex(Obj.getSectionIndex(S.getRawDataRefImpl())); 230 if (!NSec) 231 return NSec.takeError(); 232 233 // Skip relocations for MachO sections without corresponding graph 234 // sections. 235 { 236 if (!NSec->GraphSection) { 237 LLVM_DEBUG({ 238 dbgs() << " Skipping relocations for MachO section " 239 << NSec->SegName << "/" << NSec->SectName 240 << " which has no associated graph section\n"; 241 }); 242 continue; 243 } 244 } 245 246 for (auto RelItr = S.relocation_begin(), RelEnd = S.relocation_end(); 247 RelItr != RelEnd; ++RelItr) { 248 249 MachO::relocation_info RI = getRelocationInfo(RelItr); 250 251 // Validate the relocation kind. 252 auto MachORelocKind = getRelocationKind(RI); 253 if (!MachORelocKind) 254 return MachORelocKind.takeError(); 255 256 // Find the address of the value to fix up. 257 orc::ExecutorAddr FixupAddress = 258 SectionAddress + (uint32_t)RI.r_address; 259 LLVM_DEBUG({ 260 dbgs() << " " << NSec->SectName << " + " 261 << formatv("{0:x8}", RI.r_address) << ":\n"; 262 }); 263 264 // Find the block that the fixup points to. 265 Block *BlockToFix = nullptr; 266 { 267 auto SymbolToFixOrErr = findSymbolByAddress(*NSec, FixupAddress); 268 if (!SymbolToFixOrErr) 269 return SymbolToFixOrErr.takeError(); 270 BlockToFix = &SymbolToFixOrErr->getBlock(); 271 } 272 273 if (FixupAddress + orc::ExecutorAddrDiff(1ULL << RI.r_length) > 274 BlockToFix->getAddress() + BlockToFix->getContent().size()) 275 return make_error<JITLinkError>( 276 "Relocation content extends past end of fixup block"); 277 278 Edge::Kind Kind = Edge::Invalid; 279 280 // Get a pointer to the fixup content. 281 const char *FixupContent = BlockToFix->getContent().data() + 282 (FixupAddress - BlockToFix->getAddress()); 283 284 // The target symbol and addend will be populated by the switch below. 285 Symbol *TargetSymbol = nullptr; 286 uint64_t Addend = 0; 287 288 if (*MachORelocKind == MachOPairedAddend) { 289 // If this is an Addend relocation then process it and move to the 290 // paired reloc. 291 292 Addend = SignExtend64(RI.r_symbolnum, 24); 293 294 if (RelItr == RelEnd) 295 return make_error<JITLinkError>("Unpaired Addend reloc at " + 296 formatv("{0:x16}", FixupAddress)); 297 ++RelItr; 298 RI = getRelocationInfo(RelItr); 299 300 MachORelocKind = getRelocationKind(RI); 301 if (!MachORelocKind) 302 return MachORelocKind.takeError(); 303 304 if (*MachORelocKind != MachOBranch26 && 305 *MachORelocKind != MachOPage21 && 306 *MachORelocKind != MachOPageOffset12) 307 return make_error<JITLinkError>( 308 "Invalid relocation pair: Addend + " + 309 StringRef(getMachOARM64RelocationKindName(*MachORelocKind))); 310 311 LLVM_DEBUG({ 312 dbgs() << " Addend: value = " << formatv("{0:x6}", Addend) 313 << ", pair is " 314 << getMachOARM64RelocationKindName(*MachORelocKind) << "\n"; 315 }); 316 317 // Find the address of the value to fix up. 318 orc::ExecutorAddr PairedFixupAddress = 319 SectionAddress + (uint32_t)RI.r_address; 320 if (PairedFixupAddress != FixupAddress) 321 return make_error<JITLinkError>("Paired relocation points at " 322 "different target"); 323 } 324 325 switch (*MachORelocKind) { 326 case MachOBranch26: { 327 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 328 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 329 else 330 return TargetSymbolOrErr.takeError(); 331 uint32_t Instr = *(const ulittle32_t *)FixupContent; 332 if ((Instr & 0x7fffffff) != 0x14000000) 333 return make_error<JITLinkError>("BRANCH26 target is not a B or BL " 334 "instruction with a zero addend"); 335 Kind = aarch64::Branch26; 336 break; 337 } 338 case MachOPointer32: 339 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 340 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 341 else 342 return TargetSymbolOrErr.takeError(); 343 Addend = *(const ulittle32_t *)FixupContent; 344 Kind = aarch64::Pointer32; 345 break; 346 case MachOPointer64: 347 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 348 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 349 else 350 return TargetSymbolOrErr.takeError(); 351 Addend = *(const ulittle64_t *)FixupContent; 352 Kind = aarch64::Pointer64; 353 break; 354 case MachOPointer64Anon: { 355 orc::ExecutorAddr TargetAddress(*(const ulittle64_t *)FixupContent); 356 auto TargetNSec = findSectionByIndex(RI.r_symbolnum - 1); 357 if (!TargetNSec) 358 return TargetNSec.takeError(); 359 if (auto TargetSymbolOrErr = 360 findSymbolByAddress(*TargetNSec, TargetAddress)) 361 TargetSymbol = &*TargetSymbolOrErr; 362 else 363 return TargetSymbolOrErr.takeError(); 364 Addend = TargetAddress - TargetSymbol->getAddress(); 365 Kind = aarch64::Pointer64Anon; 366 break; 367 } 368 case MachOPage21: 369 case MachOTLVPage21: 370 case MachOGOTPage21: { 371 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 372 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 373 else 374 return TargetSymbolOrErr.takeError(); 375 uint32_t Instr = *(const ulittle32_t *)FixupContent; 376 if ((Instr & 0xffffffe0) != 0x90000000) 377 return make_error<JITLinkError>("PAGE21/GOTPAGE21 target is not an " 378 "ADRP instruction with a zero " 379 "addend"); 380 381 if (*MachORelocKind == MachOPage21) { 382 Kind = aarch64::Page21; 383 } else if (*MachORelocKind == MachOTLVPage21) { 384 Kind = aarch64::TLVPage21; 385 } else if (*MachORelocKind == MachOGOTPage21) { 386 Kind = aarch64::GOTPage21; 387 } 388 break; 389 } 390 case MachOPageOffset12: { 391 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 392 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 393 else 394 return TargetSymbolOrErr.takeError(); 395 uint32_t Instr = *(const ulittle32_t *)FixupContent; 396 uint32_t EncodedAddend = (Instr & 0x003FFC00) >> 10; 397 if (EncodedAddend != 0) 398 return make_error<JITLinkError>("GOTPAGEOFF12 target has non-zero " 399 "encoded addend"); 400 Kind = aarch64::PageOffset12; 401 break; 402 } 403 case MachOTLVPageOffset12: 404 case MachOGOTPageOffset12: { 405 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 406 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 407 else 408 return TargetSymbolOrErr.takeError(); 409 uint32_t Instr = *(const ulittle32_t *)FixupContent; 410 if ((Instr & 0xfffffc00) != 0xf9400000) 411 return make_error<JITLinkError>("GOTPAGEOFF12 target is not an LDR " 412 "immediate instruction with a zero " 413 "addend"); 414 415 if (*MachORelocKind == MachOTLVPageOffset12) { 416 Kind = aarch64::TLVPageOffset12; 417 } else if (*MachORelocKind == MachOGOTPageOffset12) { 418 Kind = aarch64::GOTPageOffset12; 419 } 420 break; 421 } 422 case MachOPointerToGOT: 423 if (auto TargetSymbolOrErr = findSymbolByIndex(RI.r_symbolnum)) 424 TargetSymbol = TargetSymbolOrErr->GraphSymbol; 425 else 426 return TargetSymbolOrErr.takeError(); 427 428 Kind = aarch64::Delta32ToGOT; 429 break; 430 case MachODelta32: 431 case MachODelta64: { 432 // We use Delta32/Delta64 to represent SUBTRACTOR relocations. 433 // parsePairRelocation handles the paired reloc, and returns the 434 // edge kind to be used (either Delta32/Delta64, or 435 // NegDelta32/NegDelta64, depending on the direction of the 436 // subtraction) along with the addend. 437 auto PairInfo = 438 parsePairRelocation(*BlockToFix, *MachORelocKind, RI, 439 FixupAddress, FixupContent, ++RelItr, RelEnd); 440 if (!PairInfo) 441 return PairInfo.takeError(); 442 std::tie(Kind, TargetSymbol, Addend) = *PairInfo; 443 assert(TargetSymbol && "No target symbol from parsePairRelocation?"); 444 break; 445 } 446 default: 447 llvm_unreachable("Special relocation kind should not appear in " 448 "mach-o file"); 449 } 450 451 LLVM_DEBUG({ 452 dbgs() << " "; 453 Edge GE(Kind, FixupAddress - BlockToFix->getAddress(), *TargetSymbol, 454 Addend); 455 printEdge(dbgs(), *BlockToFix, GE, aarch64::getEdgeKindName(Kind)); 456 dbgs() << "\n"; 457 }); 458 BlockToFix->addEdge(Kind, FixupAddress - BlockToFix->getAddress(), 459 *TargetSymbol, Addend); 460 } 461 } 462 return Error::success(); 463 } 464 465 /// Return the string name of the given MachO arm64 edge kind. 466 const char *getMachOARM64RelocationKindName(Edge::Kind R) { 467 switch (R) { 468 case MachOBranch26: 469 return "MachOBranch26"; 470 case MachOPointer64: 471 return "MachOPointer64"; 472 case MachOPointer64Anon: 473 return "MachOPointer64Anon"; 474 case MachOPage21: 475 return "MachOPage21"; 476 case MachOPageOffset12: 477 return "MachOPageOffset12"; 478 case MachOGOTPage21: 479 return "MachOGOTPage21"; 480 case MachOGOTPageOffset12: 481 return "MachOGOTPageOffset12"; 482 case MachOTLVPage21: 483 return "MachOTLVPage21"; 484 case MachOTLVPageOffset12: 485 return "MachOTLVPageOffset12"; 486 case MachOPointerToGOT: 487 return "MachOPointerToGOT"; 488 case MachOPairedAddend: 489 return "MachOPairedAddend"; 490 case MachOLDRLiteral19: 491 return "MachOLDRLiteral19"; 492 case MachODelta32: 493 return "MachODelta32"; 494 case MachODelta64: 495 return "MachODelta64"; 496 case MachONegDelta32: 497 return "MachONegDelta32"; 498 case MachONegDelta64: 499 return "MachONegDelta64"; 500 default: 501 return getGenericEdgeKindName(static_cast<Edge::Kind>(R)); 502 } 503 } 504 505 unsigned NumSymbols = 0; 506 }; 507 508 } // namespace 509 510 namespace llvm { 511 namespace jitlink { 512 513 Error buildTables_MachO_arm64(LinkGraph &G) { 514 LLVM_DEBUG(dbgs() << "Visiting edges in graph:\n"); 515 516 aarch64::GOTTableManager GOT; 517 aarch64::PLTTableManager PLT(GOT); 518 visitExistingEdges(G, GOT, PLT); 519 return Error::success(); 520 } 521 522 class MachOJITLinker_arm64 : public JITLinker<MachOJITLinker_arm64> { 523 friend class JITLinker<MachOJITLinker_arm64>; 524 525 public: 526 MachOJITLinker_arm64(std::unique_ptr<JITLinkContext> Ctx, 527 std::unique_ptr<LinkGraph> G, 528 PassConfiguration PassConfig) 529 : JITLinker(std::move(Ctx), std::move(G), std::move(PassConfig)) {} 530 531 private: 532 Error applyFixup(LinkGraph &G, Block &B, const Edge &E) const { 533 return aarch64::applyFixup(G, B, E); 534 } 535 536 uint64_t NullValue = 0; 537 }; 538 539 Expected<std::unique_ptr<LinkGraph>> 540 createLinkGraphFromMachOObject_arm64(MemoryBufferRef ObjectBuffer) { 541 auto MachOObj = object::ObjectFile::createMachOObjectFile(ObjectBuffer); 542 if (!MachOObj) 543 return MachOObj.takeError(); 544 return MachOLinkGraphBuilder_arm64(**MachOObj).buildGraph(); 545 } 546 547 void link_MachO_arm64(std::unique_ptr<LinkGraph> G, 548 std::unique_ptr<JITLinkContext> Ctx) { 549 550 PassConfiguration Config; 551 552 if (Ctx->shouldAddDefaultTargetPasses(G->getTargetTriple())) { 553 // Add a mark-live pass. 554 if (auto MarkLive = Ctx->getMarkLivePass(G->getTargetTriple())) 555 Config.PrePrunePasses.push_back(std::move(MarkLive)); 556 else 557 Config.PrePrunePasses.push_back(markAllSymbolsLive); 558 559 // Add compact unwind splitter pass. 560 Config.PrePrunePasses.push_back( 561 CompactUnwindSplitter("__LD,__compact_unwind")); 562 563 // Add eh-frame passses. 564 // FIXME: Prune eh-frames for which compact-unwind is available once 565 // we support compact-unwind registration with libunwind. 566 Config.PrePrunePasses.push_back( 567 DWARFRecordSectionSplitter("__TEXT,__eh_frame")); 568 Config.PrePrunePasses.push_back(EHFrameEdgeFixer( 569 "__TEXT,__eh_frame", 8, aarch64::Pointer32, aarch64::Pointer64, 570 aarch64::Delta32, aarch64::Delta64, aarch64::NegDelta32)); 571 572 // Add an in-place GOT/Stubs pass. 573 Config.PostPrunePasses.push_back(buildTables_MachO_arm64); 574 } 575 576 if (auto Err = Ctx->modifyPassConfig(*G, Config)) 577 return Ctx->notifyFailed(std::move(Err)); 578 579 // Construct a JITLinker and run the link function. 580 MachOJITLinker_arm64::link(std::move(Ctx), std::move(G), std::move(Config)); 581 } 582 583 } // end namespace jitlink 584 } // end namespace llvm 585