//===- KaleidoscopeJIT.h - A simple JIT for Kaleidoscope --------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // // Contains a simple JIT definition for use in the kaleidoscope tutorials. // //===----------------------------------------------------------------------===// #ifndef LLVM_EXECUTIONENGINE_ORC_KALEIDOSCOPEJIT_H #define LLVM_EXECUTIONENGINE_ORC_KALEIDOSCOPEJIT_H #include "llvm/ADT/StringRef.h" #include "llvm/ExecutionEngine/JITSymbol.h" #include "llvm/ExecutionEngine/Orc/CompileOnDemandLayer.h" #include "llvm/ExecutionEngine/Orc/CompileUtils.h" #include "llvm/ExecutionEngine/Orc/Core.h" #include "llvm/ExecutionEngine/Orc/ExecutionUtils.h" #include "llvm/ExecutionEngine/Orc/IRCompileLayer.h" #include "llvm/ExecutionEngine/Orc/IRTransformLayer.h" #include "llvm/ExecutionEngine/Orc/JITTargetMachineBuilder.h" #include "llvm/ExecutionEngine/Orc/RTDyldObjectLinkingLayer.h" #include "llvm/ExecutionEngine/Orc/TPCIndirectionUtils.h" #include "llvm/ExecutionEngine/Orc/TargetProcessControl.h" #include "llvm/ExecutionEngine/SectionMemoryManager.h" #include "llvm/IR/DataLayout.h" #include "llvm/IR/LLVMContext.h" #include "llvm/IR/LegacyPassManager.h" #include "llvm/Transforms/InstCombine/InstCombine.h" #include "llvm/Transforms/Scalar.h" #include "llvm/Transforms/Scalar/GVN.h" #include namespace llvm { namespace orc { class KaleidoscopeJIT { private: std::unique_ptr TPC; std::unique_ptr ES; std::unique_ptr TPCIU; DataLayout DL; MangleAndInterner Mangle; RTDyldObjectLinkingLayer ObjectLayer; IRCompileLayer CompileLayer; IRTransformLayer OptimizeLayer; CompileOnDemandLayer CODLayer; JITDylib &MainJD; static void handleLazyCallThroughError() { errs() << "LazyCallThrough error: Could not find function body"; exit(1); } public: KaleidoscopeJIT(std::unique_ptr TPC, std::unique_ptr ES, std::unique_ptr TPCIU, JITTargetMachineBuilder JTMB, DataLayout DL) : TPC(std::move(TPC)), ES(std::move(ES)), TPCIU(std::move(TPCIU)), DL(std::move(DL)), Mangle(*this->ES, this->DL), ObjectLayer(*this->ES, []() { return std::make_unique(); }), CompileLayer(*this->ES, ObjectLayer, std::make_unique(std::move(JTMB))), OptimizeLayer(*this->ES, CompileLayer, optimizeModule), CODLayer(*this->ES, OptimizeLayer, this->TPCIU->getLazyCallThroughManager(), [this] { return this->TPCIU->createIndirectStubsManager(); }), MainJD(this->ES->createBareJITDylib("
")) { MainJD.addGenerator( cantFail(DynamicLibrarySearchGenerator::GetForCurrentProcess( DL.getGlobalPrefix()))); } ~KaleidoscopeJIT() { if (auto Err = ES->endSession()) ES->reportError(std::move(Err)); if (auto Err = TPCIU->cleanup()) ES->reportError(std::move(Err)); } static Expected> Create() { auto SSP = std::make_shared(); auto TPC = SelfTargetProcessControl::Create(SSP); if (!TPC) return TPC.takeError(); auto ES = std::make_unique(std::move(SSP)); auto TPCIU = TPCIndirectionUtils::Create(**TPC); if (!TPCIU) return TPCIU.takeError(); (*TPCIU)->createLazyCallThroughManager( *ES, pointerToJITTargetAddress(&handleLazyCallThroughError)); if (auto Err = setUpInProcessLCTMReentryViaTPCIU(**TPCIU)) return std::move(Err); JITTargetMachineBuilder JTMB((*TPC)->getTargetTriple()); auto DL = JTMB.getDefaultDataLayoutForTarget(); if (!DL) return DL.takeError(); return std::make_unique(std::move(*TPC), std::move(ES), std::move(*TPCIU), std::move(JTMB), std::move(*DL)); } const DataLayout &getDataLayout() const { return DL; } JITDylib &getMainJITDylib() { return MainJD; } Error addModule(ThreadSafeModule TSM, ResourceTrackerSP RT = nullptr) { if (!RT) RT = MainJD.getDefaultResourceTracker(); return OptimizeLayer.add(RT, std::move(TSM)); } Expected lookup(StringRef Name) { return ES->lookup({&MainJD}, Mangle(Name.str())); } private: static Expected optimizeModule(ThreadSafeModule TSM, const MaterializationResponsibility &R) { TSM.withModuleDo([](Module &M) { // Create a function pass manager. auto FPM = std::make_unique(&M); // Add some optimizations. FPM->add(createInstructionCombiningPass()); FPM->add(createReassociatePass()); FPM->add(createGVNPass()); FPM->add(createCFGSimplificationPass()); FPM->doInitialization(); // Run the optimizations over all functions in the module being added to // the JIT. for (auto &F : M) FPM->run(F); }); return std::move(TSM); } }; } // end namespace orc } // end namespace llvm #endif // LLVM_EXECUTIONENGINE_ORC_KALEIDOSCOPEJIT_H