xref: /freebsd-src/contrib/llvm-project/llvm/lib/ExecutionEngine/JITLink/ELF_aarch32.cpp (revision 5f757f3ff9144b609b3c433dfd370cc6bdc191ad)
106c3fb27SDimitry Andric //===----- ELF_aarch32.cpp - JIT linker implementation for arm/thumb ------===//
206c3fb27SDimitry Andric //
306c3fb27SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
406c3fb27SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
506c3fb27SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
606c3fb27SDimitry Andric //
706c3fb27SDimitry Andric //===----------------------------------------------------------------------===//
806c3fb27SDimitry Andric //
906c3fb27SDimitry Andric // ELF/aarch32 jit-link implementation.
1006c3fb27SDimitry Andric //
1106c3fb27SDimitry Andric //===----------------------------------------------------------------------===//
1206c3fb27SDimitry Andric 
1306c3fb27SDimitry Andric #include "llvm/ExecutionEngine/JITLink/ELF_aarch32.h"
1406c3fb27SDimitry Andric 
1506c3fb27SDimitry Andric #include "llvm/BinaryFormat/ELF.h"
1606c3fb27SDimitry Andric #include "llvm/ExecutionEngine/JITLink/JITLink.h"
1706c3fb27SDimitry Andric #include "llvm/ExecutionEngine/JITLink/aarch32.h"
1806c3fb27SDimitry Andric #include "llvm/Object/ELF.h"
1906c3fb27SDimitry Andric #include "llvm/Object/ELFObjectFile.h"
2006c3fb27SDimitry Andric #include "llvm/Support/ErrorHandling.h"
2106c3fb27SDimitry Andric #include "llvm/TargetParser/ARMTargetParser.h"
2206c3fb27SDimitry Andric 
2306c3fb27SDimitry Andric #include "ELFLinkGraphBuilder.h"
2406c3fb27SDimitry Andric #include "JITLinkGeneric.h"
2506c3fb27SDimitry Andric 
2606c3fb27SDimitry Andric #define DEBUG_TYPE "jitlink"
2706c3fb27SDimitry Andric 
2806c3fb27SDimitry Andric using namespace llvm::object;
2906c3fb27SDimitry Andric 
3006c3fb27SDimitry Andric namespace llvm {
3106c3fb27SDimitry Andric namespace jitlink {
3206c3fb27SDimitry Andric 
3306c3fb27SDimitry Andric /// Translate from ELF relocation type to JITLink-internal edge kind.
3406c3fb27SDimitry Andric Expected<aarch32::EdgeKind_aarch32> getJITLinkEdgeKind(uint32_t ELFType) {
3506c3fb27SDimitry Andric   switch (ELFType) {
3606c3fb27SDimitry Andric   case ELF::R_ARM_ABS32:
3706c3fb27SDimitry Andric     return aarch32::Data_Pointer32;
3806c3fb27SDimitry Andric   case ELF::R_ARM_REL32:
3906c3fb27SDimitry Andric     return aarch32::Data_Delta32;
4006c3fb27SDimitry Andric   case ELF::R_ARM_CALL:
4106c3fb27SDimitry Andric     return aarch32::Arm_Call;
42*5f757f3fSDimitry Andric   case ELF::R_ARM_JUMP24:
43*5f757f3fSDimitry Andric     return aarch32::Arm_Jump24;
44*5f757f3fSDimitry Andric   case ELF::R_ARM_MOVW_ABS_NC:
45*5f757f3fSDimitry Andric     return aarch32::Arm_MovwAbsNC;
46*5f757f3fSDimitry Andric   case ELF::R_ARM_MOVT_ABS:
47*5f757f3fSDimitry Andric     return aarch32::Arm_MovtAbs;
4806c3fb27SDimitry Andric   case ELF::R_ARM_THM_CALL:
4906c3fb27SDimitry Andric     return aarch32::Thumb_Call;
5006c3fb27SDimitry Andric   case ELF::R_ARM_THM_JUMP24:
5106c3fb27SDimitry Andric     return aarch32::Thumb_Jump24;
5206c3fb27SDimitry Andric   case ELF::R_ARM_THM_MOVW_ABS_NC:
5306c3fb27SDimitry Andric     return aarch32::Thumb_MovwAbsNC;
5406c3fb27SDimitry Andric   case ELF::R_ARM_THM_MOVT_ABS:
5506c3fb27SDimitry Andric     return aarch32::Thumb_MovtAbs;
56*5f757f3fSDimitry Andric   case ELF::R_ARM_THM_MOVW_PREL_NC:
57*5f757f3fSDimitry Andric     return aarch32::Thumb_MovwPrelNC;
58*5f757f3fSDimitry Andric   case ELF::R_ARM_THM_MOVT_PREL:
59*5f757f3fSDimitry Andric     return aarch32::Thumb_MovtPrel;
6006c3fb27SDimitry Andric   }
6106c3fb27SDimitry Andric 
6206c3fb27SDimitry Andric   return make_error<JITLinkError>(
6306c3fb27SDimitry Andric       "Unsupported aarch32 relocation " + formatv("{0:d}: ", ELFType) +
6406c3fb27SDimitry Andric       object::getELFRelocationTypeName(ELF::EM_ARM, ELFType));
6506c3fb27SDimitry Andric }
6606c3fb27SDimitry Andric 
6706c3fb27SDimitry Andric /// Translate from JITLink-internal edge kind back to ELF relocation type.
6806c3fb27SDimitry Andric Expected<uint32_t> getELFRelocationType(Edge::Kind Kind) {
6906c3fb27SDimitry Andric   switch (static_cast<aarch32::EdgeKind_aarch32>(Kind)) {
7006c3fb27SDimitry Andric   case aarch32::Data_Delta32:
7106c3fb27SDimitry Andric     return ELF::R_ARM_REL32;
7206c3fb27SDimitry Andric   case aarch32::Data_Pointer32:
7306c3fb27SDimitry Andric     return ELF::R_ARM_ABS32;
7406c3fb27SDimitry Andric   case aarch32::Arm_Call:
7506c3fb27SDimitry Andric     return ELF::R_ARM_CALL;
76*5f757f3fSDimitry Andric   case aarch32::Arm_Jump24:
77*5f757f3fSDimitry Andric     return ELF::R_ARM_JUMP24;
78*5f757f3fSDimitry Andric   case aarch32::Arm_MovwAbsNC:
79*5f757f3fSDimitry Andric     return ELF::R_ARM_MOVW_ABS_NC;
80*5f757f3fSDimitry Andric   case aarch32::Arm_MovtAbs:
81*5f757f3fSDimitry Andric     return ELF::R_ARM_MOVT_ABS;
8206c3fb27SDimitry Andric   case aarch32::Thumb_Call:
8306c3fb27SDimitry Andric     return ELF::R_ARM_THM_CALL;
8406c3fb27SDimitry Andric   case aarch32::Thumb_Jump24:
8506c3fb27SDimitry Andric     return ELF::R_ARM_THM_JUMP24;
8606c3fb27SDimitry Andric   case aarch32::Thumb_MovwAbsNC:
8706c3fb27SDimitry Andric     return ELF::R_ARM_THM_MOVW_ABS_NC;
8806c3fb27SDimitry Andric   case aarch32::Thumb_MovtAbs:
8906c3fb27SDimitry Andric     return ELF::R_ARM_THM_MOVT_ABS;
90*5f757f3fSDimitry Andric   case aarch32::Thumb_MovwPrelNC:
91*5f757f3fSDimitry Andric     return ELF::R_ARM_THM_MOVW_PREL_NC;
92*5f757f3fSDimitry Andric   case aarch32::Thumb_MovtPrel:
93*5f757f3fSDimitry Andric     return ELF::R_ARM_THM_MOVT_PREL;
9406c3fb27SDimitry Andric   }
9506c3fb27SDimitry Andric 
9606c3fb27SDimitry Andric   return make_error<JITLinkError>(formatv("Invalid aarch32 edge {0:d}: ",
9706c3fb27SDimitry Andric                                           Kind));
9806c3fb27SDimitry Andric }
9906c3fb27SDimitry Andric 
10006c3fb27SDimitry Andric /// Get a human-readable name for the given ELF AArch32 edge kind.
10106c3fb27SDimitry Andric const char *getELFAArch32EdgeKindName(Edge::Kind R) {
10206c3fb27SDimitry Andric   // No ELF-specific edge kinds yet
10306c3fb27SDimitry Andric   return aarch32::getEdgeKindName(R);
10406c3fb27SDimitry Andric }
10506c3fb27SDimitry Andric 
10606c3fb27SDimitry Andric class ELFJITLinker_aarch32 : public JITLinker<ELFJITLinker_aarch32> {
10706c3fb27SDimitry Andric   friend class JITLinker<ELFJITLinker_aarch32>;
10806c3fb27SDimitry Andric 
10906c3fb27SDimitry Andric public:
11006c3fb27SDimitry Andric   ELFJITLinker_aarch32(std::unique_ptr<JITLinkContext> Ctx,
11106c3fb27SDimitry Andric                        std::unique_ptr<LinkGraph> G, PassConfiguration PassCfg,
11206c3fb27SDimitry Andric                        aarch32::ArmConfig ArmCfg)
11306c3fb27SDimitry Andric       : JITLinker(std::move(Ctx), std::move(G), std::move(PassCfg)),
11406c3fb27SDimitry Andric         ArmCfg(std::move(ArmCfg)) {}
11506c3fb27SDimitry Andric 
11606c3fb27SDimitry Andric private:
11706c3fb27SDimitry Andric   aarch32::ArmConfig ArmCfg;
11806c3fb27SDimitry Andric 
11906c3fb27SDimitry Andric   Error applyFixup(LinkGraph &G, Block &B, const Edge &E) const {
12006c3fb27SDimitry Andric     return aarch32::applyFixup(G, B, E, ArmCfg);
12106c3fb27SDimitry Andric   }
12206c3fb27SDimitry Andric };
12306c3fb27SDimitry Andric 
124*5f757f3fSDimitry Andric template <llvm::endianness DataEndianness>
12506c3fb27SDimitry Andric class ELFLinkGraphBuilder_aarch32
12606c3fb27SDimitry Andric     : public ELFLinkGraphBuilder<ELFType<DataEndianness, false>> {
12706c3fb27SDimitry Andric private:
12806c3fb27SDimitry Andric   using ELFT = ELFType<DataEndianness, false>;
12906c3fb27SDimitry Andric   using Base = ELFLinkGraphBuilder<ELFT>;
13006c3fb27SDimitry Andric 
13106c3fb27SDimitry Andric   bool excludeSection(const typename ELFT::Shdr &Sect) const override {
13206c3fb27SDimitry Andric     // TODO: An .ARM.exidx (Exception Index table) entry is 8-bytes in size and
13306c3fb27SDimitry Andric     // consists of 2 words. It might be sufficient to process only relocations
13406c3fb27SDimitry Andric     // in the the second word (offset 4). Please find more details in: Exception
13506c3fb27SDimitry Andric     // Handling ABI for the Arm® Architecture -> Index table entries
13606c3fb27SDimitry Andric     if (Sect.sh_type == ELF::SHT_ARM_EXIDX)
13706c3fb27SDimitry Andric       return true;
13806c3fb27SDimitry Andric     return false;
13906c3fb27SDimitry Andric   }
14006c3fb27SDimitry Andric 
14106c3fb27SDimitry Andric   Error addRelocations() override {
14206c3fb27SDimitry Andric     LLVM_DEBUG(dbgs() << "Processing relocations:\n");
14306c3fb27SDimitry Andric     using Self = ELFLinkGraphBuilder_aarch32<DataEndianness>;
14406c3fb27SDimitry Andric     for (const auto &RelSect : Base::Sections) {
14506c3fb27SDimitry Andric       if (Error Err = Base::forEachRelRelocation(RelSect, this,
14606c3fb27SDimitry Andric                                                  &Self::addSingleRelRelocation))
14706c3fb27SDimitry Andric         return Err;
14806c3fb27SDimitry Andric     }
14906c3fb27SDimitry Andric     return Error::success();
15006c3fb27SDimitry Andric   }
15106c3fb27SDimitry Andric 
15206c3fb27SDimitry Andric   Error addSingleRelRelocation(const typename ELFT::Rel &Rel,
15306c3fb27SDimitry Andric                                const typename ELFT::Shdr &FixupSect,
15406c3fb27SDimitry Andric                                Block &BlockToFix) {
15506c3fb27SDimitry Andric     uint32_t SymbolIndex = Rel.getSymbol(false);
15606c3fb27SDimitry Andric     auto ObjSymbol = Base::Obj.getRelocationSymbol(Rel, Base::SymTabSec);
15706c3fb27SDimitry Andric     if (!ObjSymbol)
15806c3fb27SDimitry Andric       return ObjSymbol.takeError();
15906c3fb27SDimitry Andric 
16006c3fb27SDimitry Andric     Symbol *GraphSymbol = Base::getGraphSymbol(SymbolIndex);
16106c3fb27SDimitry Andric     if (!GraphSymbol)
16206c3fb27SDimitry Andric       return make_error<StringError>(
16306c3fb27SDimitry Andric           formatv("Could not find symbol at given index, did you add it to "
16406c3fb27SDimitry Andric                   "JITSymbolTable? index: {0}, shndx: {1} Size of table: {2}",
16506c3fb27SDimitry Andric                   SymbolIndex, (*ObjSymbol)->st_shndx,
16606c3fb27SDimitry Andric                   Base::GraphSymbols.size()),
16706c3fb27SDimitry Andric           inconvertibleErrorCode());
16806c3fb27SDimitry Andric 
16906c3fb27SDimitry Andric     uint32_t Type = Rel.getType(false);
17006c3fb27SDimitry Andric     Expected<aarch32::EdgeKind_aarch32> Kind = getJITLinkEdgeKind(Type);
17106c3fb27SDimitry Andric     if (!Kind)
17206c3fb27SDimitry Andric       return Kind.takeError();
17306c3fb27SDimitry Andric 
17406c3fb27SDimitry Andric     auto FixupAddress = orc::ExecutorAddr(FixupSect.sh_addr) + Rel.r_offset;
17506c3fb27SDimitry Andric     Edge::OffsetT Offset = FixupAddress - BlockToFix.getAddress();
17606c3fb27SDimitry Andric 
17706c3fb27SDimitry Andric     Expected<int64_t> Addend =
178*5f757f3fSDimitry Andric         aarch32::readAddend(*Base::G, BlockToFix, Offset, *Kind, ArmCfg);
17906c3fb27SDimitry Andric     if (!Addend)
18006c3fb27SDimitry Andric       return Addend.takeError();
18106c3fb27SDimitry Andric 
182*5f757f3fSDimitry Andric     Edge E(*Kind, Offset, *GraphSymbol, *Addend);
18306c3fb27SDimitry Andric     LLVM_DEBUG({
18406c3fb27SDimitry Andric       dbgs() << "    ";
18506c3fb27SDimitry Andric       printEdge(dbgs(), BlockToFix, E, getELFAArch32EdgeKindName(*Kind));
18606c3fb27SDimitry Andric       dbgs() << "\n";
18706c3fb27SDimitry Andric     });
18806c3fb27SDimitry Andric 
18906c3fb27SDimitry Andric     BlockToFix.addEdge(std::move(E));
19006c3fb27SDimitry Andric     return Error::success();
19106c3fb27SDimitry Andric   }
19206c3fb27SDimitry Andric 
19306c3fb27SDimitry Andric   aarch32::ArmConfig ArmCfg;
19406c3fb27SDimitry Andric 
19506c3fb27SDimitry Andric protected:
19606c3fb27SDimitry Andric   TargetFlagsType makeTargetFlags(const typename ELFT::Sym &Sym) override {
19706c3fb27SDimitry Andric     if (Sym.getValue() & 0x01)
19806c3fb27SDimitry Andric       return aarch32::ThumbSymbol;
19906c3fb27SDimitry Andric     return TargetFlagsType{};
20006c3fb27SDimitry Andric   }
20106c3fb27SDimitry Andric 
20206c3fb27SDimitry Andric   orc::ExecutorAddrDiff getRawOffset(const typename ELFT::Sym &Sym,
20306c3fb27SDimitry Andric                                      TargetFlagsType Flags) override {
20406c3fb27SDimitry Andric     assert((makeTargetFlags(Sym) & Flags) == Flags);
20506c3fb27SDimitry Andric     static constexpr uint64_t ThumbBit = 0x01;
20606c3fb27SDimitry Andric     return Sym.getValue() & ~ThumbBit;
20706c3fb27SDimitry Andric   }
20806c3fb27SDimitry Andric 
20906c3fb27SDimitry Andric public:
21006c3fb27SDimitry Andric   ELFLinkGraphBuilder_aarch32(StringRef FileName,
21106c3fb27SDimitry Andric                               const llvm::object::ELFFile<ELFT> &Obj, Triple TT,
21206c3fb27SDimitry Andric                               SubtargetFeatures Features,
21306c3fb27SDimitry Andric                               aarch32::ArmConfig ArmCfg)
21406c3fb27SDimitry Andric       : ELFLinkGraphBuilder<ELFT>(Obj, std::move(TT), std::move(Features),
21506c3fb27SDimitry Andric                                   FileName, getELFAArch32EdgeKindName),
21606c3fb27SDimitry Andric         ArmCfg(std::move(ArmCfg)) {}
21706c3fb27SDimitry Andric };
21806c3fb27SDimitry Andric 
21906c3fb27SDimitry Andric template <aarch32::StubsFlavor Flavor>
22006c3fb27SDimitry Andric Error buildTables_ELF_aarch32(LinkGraph &G) {
22106c3fb27SDimitry Andric   LLVM_DEBUG(dbgs() << "Visiting edges in graph:\n");
22206c3fb27SDimitry Andric 
22306c3fb27SDimitry Andric   aarch32::StubsManager<Flavor> PLT;
22406c3fb27SDimitry Andric   visitExistingEdges(G, PLT);
22506c3fb27SDimitry Andric   return Error::success();
22606c3fb27SDimitry Andric }
22706c3fb27SDimitry Andric 
22806c3fb27SDimitry Andric Expected<std::unique_ptr<LinkGraph>>
22906c3fb27SDimitry Andric createLinkGraphFromELFObject_aarch32(MemoryBufferRef ObjectBuffer) {
23006c3fb27SDimitry Andric   LLVM_DEBUG({
23106c3fb27SDimitry Andric     dbgs() << "Building jitlink graph for new input "
23206c3fb27SDimitry Andric            << ObjectBuffer.getBufferIdentifier() << "...\n";
23306c3fb27SDimitry Andric   });
23406c3fb27SDimitry Andric 
23506c3fb27SDimitry Andric   auto ELFObj = ObjectFile::createELFObjectFile(ObjectBuffer);
23606c3fb27SDimitry Andric   if (!ELFObj)
23706c3fb27SDimitry Andric     return ELFObj.takeError();
23806c3fb27SDimitry Andric 
23906c3fb27SDimitry Andric   auto Features = (*ELFObj)->getFeatures();
24006c3fb27SDimitry Andric   if (!Features)
24106c3fb27SDimitry Andric     return Features.takeError();
24206c3fb27SDimitry Andric 
24306c3fb27SDimitry Andric   // Find out what exact AArch32 instruction set and features we target.
24406c3fb27SDimitry Andric   auto TT = (*ELFObj)->makeTriple();
24506c3fb27SDimitry Andric   ARM::ArchKind AK = ARM::parseArch(TT.getArchName());
24606c3fb27SDimitry Andric   if (AK == ARM::ArchKind::INVALID)
24706c3fb27SDimitry Andric     return make_error<JITLinkError>(
24806c3fb27SDimitry Andric         "Failed to build ELF link graph: Invalid ARM ArchKind");
24906c3fb27SDimitry Andric 
25006c3fb27SDimitry Andric   // Resolve our internal configuration for the target. If at some point the
25106c3fb27SDimitry Andric   // CPUArch alone becomes too unprecise, we can find more details in the
25206c3fb27SDimitry Andric   // Tag_CPU_arch_profile.
25306c3fb27SDimitry Andric   aarch32::ArmConfig ArmCfg;
25406c3fb27SDimitry Andric   using namespace ARMBuildAttrs;
25506c3fb27SDimitry Andric   auto Arch = static_cast<CPUArch>(ARM::getArchAttr(AK));
25606c3fb27SDimitry Andric   switch (Arch) {
25706c3fb27SDimitry Andric   case v7:
25806c3fb27SDimitry Andric   case v8_A:
25906c3fb27SDimitry Andric     ArmCfg = aarch32::getArmConfigForCPUArch(Arch);
26006c3fb27SDimitry Andric     assert(ArmCfg.Stubs != aarch32::Unsupported &&
26106c3fb27SDimitry Andric            "Provide a config for each supported CPU");
26206c3fb27SDimitry Andric     break;
26306c3fb27SDimitry Andric   default:
26406c3fb27SDimitry Andric     return make_error<JITLinkError>(
26506c3fb27SDimitry Andric         "Failed to build ELF link graph: Unsupported CPU arch " +
26606c3fb27SDimitry Andric         StringRef(aarch32::getCPUArchName(Arch)));
26706c3fb27SDimitry Andric   }
26806c3fb27SDimitry Andric 
26906c3fb27SDimitry Andric   // Populate the link-graph.
27006c3fb27SDimitry Andric   switch (TT.getArch()) {
27106c3fb27SDimitry Andric   case Triple::arm:
27206c3fb27SDimitry Andric   case Triple::thumb: {
27306c3fb27SDimitry Andric     auto &ELFFile = cast<ELFObjectFile<ELF32LE>>(**ELFObj).getELFFile();
274*5f757f3fSDimitry Andric     return ELFLinkGraphBuilder_aarch32<llvm::endianness::little>(
27506c3fb27SDimitry Andric                (*ELFObj)->getFileName(), ELFFile, TT, std::move(*Features),
27606c3fb27SDimitry Andric                ArmCfg)
27706c3fb27SDimitry Andric         .buildGraph();
27806c3fb27SDimitry Andric   }
27906c3fb27SDimitry Andric   case Triple::armeb:
28006c3fb27SDimitry Andric   case Triple::thumbeb: {
28106c3fb27SDimitry Andric     auto &ELFFile = cast<ELFObjectFile<ELF32BE>>(**ELFObj).getELFFile();
282*5f757f3fSDimitry Andric     return ELFLinkGraphBuilder_aarch32<llvm::endianness::big>(
28306c3fb27SDimitry Andric                (*ELFObj)->getFileName(), ELFFile, TT, std::move(*Features),
28406c3fb27SDimitry Andric                ArmCfg)
28506c3fb27SDimitry Andric         .buildGraph();
28606c3fb27SDimitry Andric   }
28706c3fb27SDimitry Andric   default:
28806c3fb27SDimitry Andric     return make_error<JITLinkError>(
28906c3fb27SDimitry Andric         "Failed to build ELF/aarch32 link graph: Invalid target triple " +
29006c3fb27SDimitry Andric         TT.getTriple());
29106c3fb27SDimitry Andric   }
29206c3fb27SDimitry Andric }
29306c3fb27SDimitry Andric 
29406c3fb27SDimitry Andric void link_ELF_aarch32(std::unique_ptr<LinkGraph> G,
29506c3fb27SDimitry Andric                       std::unique_ptr<JITLinkContext> Ctx) {
29606c3fb27SDimitry Andric   const Triple &TT = G->getTargetTriple();
29706c3fb27SDimitry Andric 
29806c3fb27SDimitry Andric   using namespace ARMBuildAttrs;
29906c3fb27SDimitry Andric   ARM::ArchKind AK = ARM::parseArch(TT.getArchName());
30006c3fb27SDimitry Andric   auto CPU = static_cast<CPUArch>(ARM::getArchAttr(AK));
30106c3fb27SDimitry Andric   aarch32::ArmConfig ArmCfg = aarch32::getArmConfigForCPUArch(CPU);
30206c3fb27SDimitry Andric 
30306c3fb27SDimitry Andric   PassConfiguration PassCfg;
30406c3fb27SDimitry Andric   if (Ctx->shouldAddDefaultTargetPasses(TT)) {
30506c3fb27SDimitry Andric     // Add a mark-live pass.
30606c3fb27SDimitry Andric     if (auto MarkLive = Ctx->getMarkLivePass(TT))
30706c3fb27SDimitry Andric       PassCfg.PrePrunePasses.push_back(std::move(MarkLive));
30806c3fb27SDimitry Andric     else
30906c3fb27SDimitry Andric       PassCfg.PrePrunePasses.push_back(markAllSymbolsLive);
31006c3fb27SDimitry Andric 
31106c3fb27SDimitry Andric     switch (ArmCfg.Stubs) {
31206c3fb27SDimitry Andric     case aarch32::Thumbv7:
31306c3fb27SDimitry Andric       PassCfg.PostPrunePasses.push_back(
31406c3fb27SDimitry Andric           buildTables_ELF_aarch32<aarch32::Thumbv7>);
31506c3fb27SDimitry Andric       break;
31606c3fb27SDimitry Andric     case aarch32::Unsupported:
31706c3fb27SDimitry Andric       llvm_unreachable("Check before building graph");
31806c3fb27SDimitry Andric     }
31906c3fb27SDimitry Andric   }
32006c3fb27SDimitry Andric 
32106c3fb27SDimitry Andric   if (auto Err = Ctx->modifyPassConfig(*G, PassCfg))
32206c3fb27SDimitry Andric     return Ctx->notifyFailed(std::move(Err));
32306c3fb27SDimitry Andric 
32406c3fb27SDimitry Andric   ELFJITLinker_aarch32::link(std::move(Ctx), std::move(G), std::move(PassCfg),
32506c3fb27SDimitry Andric                              std::move(ArmCfg));
32606c3fb27SDimitry Andric }
32706c3fb27SDimitry Andric 
32806c3fb27SDimitry Andric } // namespace jitlink
32906c3fb27SDimitry Andric } // namespace llvm
330