//===------ OrcTestCommon.h - Utilities for Orc Unit Tests ------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // Common utilities for the Orc unit tests. // //===----------------------------------------------------------------------===// #ifndef LLVM_UNITTESTS_EXECUTIONENGINE_ORC_ORCTESTCOMMON_H #define LLVM_UNITTESTS_EXECUTIONENGINE_ORC_ORCTESTCOMMON_H #include "llvm/ExecutionEngine/ExecutionEngine.h" #include "llvm/ExecutionEngine/JITSymbol.h" #include "llvm/IR/Function.h" #include "llvm/IR/IRBuilder.h" #include "llvm/IR/LLVMContext.h" #include "llvm/IR/Module.h" #include "llvm/IR/TypeBuilder.h" #include "llvm/Object/ObjectFile.h" #include "llvm/Support/TargetSelect.h" #include namespace llvm { class OrcNativeTarget { public: static void initialize() { if (!NativeTargetInitialized) { InitializeNativeTarget(); InitializeNativeTargetAsmParser(); InitializeNativeTargetAsmPrinter(); NativeTargetInitialized = true; } } private: static bool NativeTargetInitialized; }; // Base class for Orc tests that will execute code. class OrcExecutionTest { public: OrcExecutionTest() { // Initialize the native target if it hasn't been done already. OrcNativeTarget::initialize(); // Try to select a TargetMachine for the host. TM.reset(EngineBuilder().selectTarget()); if (TM) { // If we found a TargetMachine, check that it's one that Orc supports. const Triple& TT = TM->getTargetTriple(); if ((TT.getArch() != Triple::x86_64 && TT.getArch() != Triple::x86) || TT.isOSWindows()) TM = nullptr; } }; protected: LLVMContext Context; std::unique_ptr TM; }; class ModuleBuilder { public: ModuleBuilder(LLVMContext &Context, StringRef Triple, StringRef Name); template Function* createFunctionDecl(StringRef Name) { return Function::Create( TypeBuilder::get(M->getContext()), GlobalValue::ExternalLinkage, Name, M.get()); } Module* getModule() { return M.get(); } const Module* getModule() const { return M.get(); } std::unique_ptr takeModule() { return std::move(M); } private: std::unique_ptr M; }; // Dummy struct type. struct DummyStruct { int X[256]; }; // TypeBuilder specialization for DummyStruct. template class TypeBuilder { public: static StructType *get(LLVMContext &Context) { return StructType::get( TypeBuilder[256], XCompile>::get(Context)); } }; template class MockBaseLayer { public: using ModuleHandleT = HandleT; using AddModuleSignature = Expected(ModuleT M, std::shared_ptr R); using RemoveModuleSignature = Error(ModuleHandleT H); using FindSymbolSignature = JITSymbol(const std::string &Name, bool ExportedSymbolsOnly); using FindSymbolInSignature = JITSymbol(ModuleHandleT H, const std::string &Name, bool ExportedSymbolsONly); using EmitAndFinalizeSignature = Error(ModuleHandleT H); std::function addModuleImpl; std::function removeModuleImpl; std::function findSymbolImpl; std::function findSymbolInImpl; std::function emitAndFinalizeImpl; Expected addModule(ModuleT M, std::shared_ptr R) { assert(addModuleImpl && "addModule called, but no mock implementation was provided"); return addModuleImpl(std::move(M), std::move(R)); } Error removeModule(ModuleHandleT H) { assert(removeModuleImpl && "removeModule called, but no mock implementation was provided"); return removeModuleImpl(H); } JITSymbol findSymbol(const std::string &Name, bool ExportedSymbolsOnly) { assert(findSymbolImpl && "findSymbol called, but no mock implementation was provided"); return findSymbolImpl(Name, ExportedSymbolsOnly); } JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name, bool ExportedSymbolsOnly) { assert(findSymbolInImpl && "findSymbolIn called, but no mock implementation was provided"); return findSymbolInImpl(H, Name, ExportedSymbolsOnly); } Error emitAndFinaliez(ModuleHandleT H) { assert(emitAndFinalizeImpl && "emitAndFinalize called, but no mock implementation was provided"); return emitAndFinalizeImpl(H); } }; class ReturnNullJITSymbol { public: template JITSymbol operator()(Args...) const { return nullptr; } }; template class DoNothingAndReturn { public: DoNothingAndReturn(ReturnT Ret) : Ret(std::move(Ret)) {} template void operator()(Args...) const { return Ret; } private: ReturnT Ret; }; template <> class DoNothingAndReturn { public: template void operator()(Args...) const { } }; } // namespace llvm #endif