2015-04-20 22:41:45 +02:00
|
|
|
//===------ 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
|
|
|
|
|
2017-06-06 13:06:56 +02:00
|
|
|
#include "llvm/ExecutionEngine/ExecutionEngine.h"
|
|
|
|
#include "llvm/ExecutionEngine/JITSymbol.h"
|
2018-03-28 05:41:45 +02:00
|
|
|
#include "llvm/ExecutionEngine/Orc/IndirectionUtils.h"
|
2015-04-20 22:41:45 +02:00
|
|
|
#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"
|
2016-01-09 20:50:40 +01:00
|
|
|
#include "llvm/Object/ObjectFile.h"
|
2018-03-28 05:41:45 +02:00
|
|
|
#include "llvm/Support/TargetRegistry.h"
|
2018-07-20 20:31:50 +02:00
|
|
|
#include "llvm/Support/TargetSelect.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
2015-04-20 22:41:45 +02:00
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2018-07-20 20:31:50 +02:00
|
|
|
namespace orc {
|
|
|
|
// CoreAPIsStandardTest that saves a bunch of boilerplate by providing the
|
|
|
|
// following:
|
|
|
|
//
|
|
|
|
// (1) ES -- An ExecutionSession
|
|
|
|
// (2) Foo, Bar, Baz, Qux -- SymbolStringPtrs for strings "foo", "bar", "baz",
|
|
|
|
// and "qux" respectively.
|
|
|
|
// (3) FooAddr, BarAddr, BazAddr, QuxAddr -- Dummy addresses. Guaranteed
|
|
|
|
// distinct and non-null.
|
|
|
|
// (4) FooSym, BarSym, BazSym, QuxSym -- JITEvaluatedSymbols with FooAddr,
|
|
|
|
// BarAddr, BazAddr, and QuxAddr respectively. All with default strong,
|
|
|
|
// linkage and non-hidden visibility.
|
2018-08-17 23:18:18 +02:00
|
|
|
// (5) V -- A JITDylib associated with ES.
|
2018-07-20 20:31:50 +02:00
|
|
|
class CoreAPIsBasedStandardTest : public testing::Test {
|
|
|
|
public:
|
|
|
|
protected:
|
|
|
|
ExecutionSession ES;
|
2018-08-17 23:18:18 +02:00
|
|
|
JITDylib &JD = ES.createJITDylib("JD");
|
2018-07-20 20:31:50 +02:00
|
|
|
SymbolStringPtr Foo = ES.getSymbolStringPool().intern("foo");
|
|
|
|
SymbolStringPtr Bar = ES.getSymbolStringPool().intern("bar");
|
|
|
|
SymbolStringPtr Baz = ES.getSymbolStringPool().intern("baz");
|
|
|
|
SymbolStringPtr Qux = ES.getSymbolStringPool().intern("qux");
|
|
|
|
static const JITTargetAddress FooAddr = 1U;
|
|
|
|
static const JITTargetAddress BarAddr = 2U;
|
|
|
|
static const JITTargetAddress BazAddr = 3U;
|
|
|
|
static const JITTargetAddress QuxAddr = 4U;
|
|
|
|
JITEvaluatedSymbol FooSym =
|
|
|
|
JITEvaluatedSymbol(FooAddr, JITSymbolFlags::Exported);
|
|
|
|
JITEvaluatedSymbol BarSym =
|
|
|
|
JITEvaluatedSymbol(BarAddr, JITSymbolFlags::Exported);
|
|
|
|
JITEvaluatedSymbol BazSym =
|
|
|
|
JITEvaluatedSymbol(BazAddr, JITSymbolFlags::Exported);
|
|
|
|
JITEvaluatedSymbol QuxSym =
|
|
|
|
JITEvaluatedSymbol(QuxAddr, JITSymbolFlags::Exported);
|
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace orc
|
|
|
|
|
2017-04-04 19:03:49 +02:00
|
|
|
class OrcNativeTarget {
|
2015-10-28 03:40:04 +01:00
|
|
|
public:
|
2017-04-04 19:03:49 +02:00
|
|
|
static void initialize() {
|
2015-10-28 03:40:04 +01:00
|
|
|
if (!NativeTargetInitialized) {
|
|
|
|
InitializeNativeTarget();
|
|
|
|
InitializeNativeTargetAsmParser();
|
|
|
|
InitializeNativeTargetAsmPrinter();
|
|
|
|
NativeTargetInitialized = true;
|
|
|
|
}
|
2017-04-04 19:03:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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();
|
2015-10-28 03:40:04 +01:00
|
|
|
|
2015-10-30 04:20:21 +01:00
|
|
|
// Try to select a TargetMachine for the host.
|
|
|
|
TM.reset(EngineBuilder().selectTarget());
|
2015-10-28 03:40:04 +01:00
|
|
|
|
2015-10-30 04:20:21 +01:00
|
|
|
if (TM) {
|
|
|
|
// If we found a TargetMachine, check that it's one that Orc supports.
|
|
|
|
const Triple& TT = TM->getTargetTriple();
|
2016-02-10 02:02:33 +01:00
|
|
|
|
2018-03-28 16:47:11 +02:00
|
|
|
// Bail out for windows platforms. We do not support these yet.
|
2018-03-28 17:58:14 +02:00
|
|
|
if ((TT.getArch() != Triple::x86_64 && TT.getArch() != Triple::x86) ||
|
|
|
|
TT.isOSWindows())
|
2018-03-28 16:47:11 +02:00
|
|
|
return;
|
|
|
|
|
2018-03-28 05:41:45 +02:00
|
|
|
// Target can JIT?
|
|
|
|
SupportsJIT = TM->getTarget().hasJIT();
|
|
|
|
// Use ability to create callback manager to detect whether Orc
|
|
|
|
// has indirection support on this platform. This way the test
|
|
|
|
// and Orc code do not get out of sync.
|
2018-05-30 03:57:45 +02:00
|
|
|
SupportsIndirection = !!orc::createLocalCompileCallbackManager(TT, ES, 0);
|
2015-10-30 04:20:21 +01:00
|
|
|
}
|
|
|
|
};
|
2015-10-28 03:40:04 +01:00
|
|
|
|
2015-10-30 04:20:21 +01:00
|
|
|
protected:
|
2018-05-30 03:57:45 +02:00
|
|
|
orc::ExecutionSession ES;
|
2016-04-14 23:59:01 +02:00
|
|
|
LLVMContext Context;
|
2015-10-30 04:20:21 +01:00
|
|
|
std::unique_ptr<TargetMachine> TM;
|
2018-03-28 05:41:45 +02:00
|
|
|
bool SupportsJIT = false;
|
|
|
|
bool SupportsIndirection = false;
|
2015-10-28 03:40:04 +01:00
|
|
|
};
|
|
|
|
|
2015-10-27 18:45:48 +01:00
|
|
|
class ModuleBuilder {
|
|
|
|
public:
|
|
|
|
ModuleBuilder(LLVMContext &Context, StringRef Triple,
|
|
|
|
StringRef Name);
|
2015-04-20 22:41:45 +02:00
|
|
|
|
2015-10-27 18:45:48 +01:00
|
|
|
template <typename FuncType>
|
2015-10-28 03:40:04 +01:00
|
|
|
Function* createFunctionDecl(StringRef Name) {
|
2015-10-27 18:45:48 +01:00
|
|
|
return Function::Create(
|
|
|
|
TypeBuilder<FuncType, false>::get(M->getContext()),
|
2015-10-28 03:40:04 +01:00
|
|
|
GlobalValue::ExternalLinkage, Name, M.get());
|
2015-10-27 18:45:48 +01:00
|
|
|
}
|
2015-04-20 22:41:45 +02:00
|
|
|
|
2015-10-27 18:45:48 +01:00
|
|
|
Module* getModule() { return M.get(); }
|
|
|
|
const Module* getModule() const { return M.get(); }
|
|
|
|
std::unique_ptr<Module> takeModule() { return std::move(M); }
|
2015-04-20 22:41:45 +02:00
|
|
|
|
2015-10-27 18:45:48 +01:00
|
|
|
private:
|
|
|
|
std::unique_ptr<Module> M;
|
|
|
|
};
|
2015-04-20 22:41:45 +02:00
|
|
|
|
2015-10-27 18:45:48 +01:00
|
|
|
// Dummy struct type.
|
|
|
|
struct DummyStruct {
|
|
|
|
int X[256];
|
|
|
|
};
|
2015-04-20 22:41:45 +02:00
|
|
|
|
2015-10-27 18:45:48 +01:00
|
|
|
// TypeBuilder specialization for DummyStruct.
|
|
|
|
template <bool XCompile>
|
|
|
|
class TypeBuilder<DummyStruct, XCompile> {
|
|
|
|
public:
|
|
|
|
static StructType *get(LLVMContext &Context) {
|
|
|
|
return StructType::get(
|
2017-05-09 21:31:13 +02:00
|
|
|
TypeBuilder<types::i<32>[256], XCompile>::get(Context));
|
2015-10-27 18:45:48 +01:00
|
|
|
}
|
|
|
|
};
|
2015-04-20 22:41:45 +02:00
|
|
|
|
2017-09-28 04:17:35 +02:00
|
|
|
template <typename HandleT, typename ModuleT>
|
2015-10-20 06:35:02 +02:00
|
|
|
class MockBaseLayer {
|
|
|
|
public:
|
|
|
|
|
2017-09-28 04:17:35 +02:00
|
|
|
using ModuleHandleT = HandleT;
|
|
|
|
|
|
|
|
using AddModuleSignature =
|
|
|
|
Expected<ModuleHandleT>(ModuleT M,
|
|
|
|
std::shared_ptr<JITSymbolResolver> 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<AddModuleSignature> addModuleImpl;
|
|
|
|
std::function<RemoveModuleSignature> removeModuleImpl;
|
|
|
|
std::function<FindSymbolSignature> findSymbolImpl;
|
|
|
|
std::function<FindSymbolInSignature> findSymbolInImpl;
|
|
|
|
std::function<EmitAndFinalizeSignature> emitAndFinalizeImpl;
|
|
|
|
|
|
|
|
Expected<ModuleHandleT> addModule(ModuleT M,
|
|
|
|
std::shared_ptr<JITSymbolResolver> R) {
|
|
|
|
assert(addModuleImpl &&
|
|
|
|
"addModule called, but no mock implementation was provided");
|
|
|
|
return addModuleImpl(std::move(M), std::move(R));
|
2015-10-20 06:35:02 +02:00
|
|
|
}
|
|
|
|
|
2017-07-07 04:59:13 +02:00
|
|
|
Error removeModule(ModuleHandleT H) {
|
2017-09-28 04:17:35 +02:00
|
|
|
assert(removeModuleImpl &&
|
|
|
|
"removeModule called, but no mock implementation was provided");
|
|
|
|
return removeModuleImpl(H);
|
2015-10-20 06:35:02 +02:00
|
|
|
}
|
|
|
|
|
2016-08-01 22:49:11 +02:00
|
|
|
JITSymbol findSymbol(const std::string &Name, bool ExportedSymbolsOnly) {
|
2017-09-28 04:17:35 +02:00
|
|
|
assert(findSymbolImpl &&
|
|
|
|
"findSymbol called, but no mock implementation was provided");
|
|
|
|
return findSymbolImpl(Name, ExportedSymbolsOnly);
|
2015-10-20 06:35:02 +02:00
|
|
|
}
|
|
|
|
|
2017-06-24 01:25:28 +02:00
|
|
|
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
2015-10-20 06:35:02 +02:00
|
|
|
bool ExportedSymbolsOnly) {
|
2017-09-28 04:17:35 +02:00
|
|
|
assert(findSymbolInImpl &&
|
|
|
|
"findSymbolIn called, but no mock implementation was provided");
|
|
|
|
return findSymbolInImpl(H, Name, ExportedSymbolsOnly);
|
2015-10-20 06:35:02 +02:00
|
|
|
}
|
|
|
|
|
2017-09-28 04:17:35 +02:00
|
|
|
Error emitAndFinaliez(ModuleHandleT H) {
|
|
|
|
assert(emitAndFinalizeImpl &&
|
|
|
|
"emitAndFinalize called, but no mock implementation was provided");
|
|
|
|
return emitAndFinalizeImpl(H);
|
|
|
|
}
|
2015-10-20 06:35:02 +02:00
|
|
|
};
|
|
|
|
|
2017-07-07 04:59:13 +02:00
|
|
|
class ReturnNullJITSymbol {
|
|
|
|
public:
|
|
|
|
template <typename... Args>
|
|
|
|
JITSymbol operator()(Args...) const {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-10-20 06:35:02 +02:00
|
|
|
template <typename ReturnT>
|
|
|
|
class DoNothingAndReturn {
|
|
|
|
public:
|
2017-07-07 04:59:13 +02:00
|
|
|
DoNothingAndReturn(ReturnT Ret) : Ret(std::move(Ret)) {}
|
2015-10-20 06:35:02 +02:00
|
|
|
|
|
|
|
template <typename... Args>
|
2017-07-07 04:59:13 +02:00
|
|
|
void operator()(Args...) const { return Ret; }
|
2015-10-20 06:35:02 +02:00
|
|
|
private:
|
2017-07-07 04:59:13 +02:00
|
|
|
ReturnT Ret;
|
2015-10-20 06:35:02 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
class DoNothingAndReturn<void> {
|
|
|
|
public:
|
|
|
|
template <typename... Args>
|
|
|
|
void operator()(Args...) const { }
|
|
|
|
};
|
2015-04-20 22:41:45 +02:00
|
|
|
|
|
|
|
} // namespace llvm
|
|
|
|
|
|
|
|
#endif
|