2015-04-21 04:41:45 +08: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 19:06:56 +08:00
|
|
|
#include "llvm/ExecutionEngine/ExecutionEngine.h"
|
|
|
|
#include "llvm/ExecutionEngine/JITSymbol.h"
|
2015-04-21 04:41:45 +08: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-10 03:50:40 +08:00
|
|
|
#include "llvm/Object/ObjectFile.h"
|
2015-10-28 10:40:04 +08:00
|
|
|
#include "llvm/Support/TargetSelect.h"
|
2015-04-21 04:41:45 +08:00
|
|
|
#include <memory>
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2017-04-05 01:03:49 +08:00
|
|
|
class OrcNativeTarget {
|
2015-10-28 10:40:04 +08:00
|
|
|
public:
|
2017-04-05 01:03:49 +08:00
|
|
|
static void initialize() {
|
2015-10-28 10:40:04 +08:00
|
|
|
if (!NativeTargetInitialized) {
|
|
|
|
InitializeNativeTarget();
|
|
|
|
InitializeNativeTargetAsmParser();
|
|
|
|
InitializeNativeTargetAsmPrinter();
|
|
|
|
NativeTargetInitialized = true;
|
|
|
|
}
|
2017-04-05 01:03:49 +08: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 10:40:04 +08:00
|
|
|
|
2015-10-30 11:20:21 +08:00
|
|
|
// Try to select a TargetMachine for the host.
|
|
|
|
TM.reset(EngineBuilder().selectTarget());
|
2015-10-28 10:40:04 +08:00
|
|
|
|
2015-10-30 11:20:21 +08:00
|
|
|
if (TM) {
|
|
|
|
// If we found a TargetMachine, check that it's one that Orc supports.
|
|
|
|
const Triple& TT = TM->getTargetTriple();
|
2016-02-10 09:02:33 +08:00
|
|
|
|
|
|
|
if ((TT.getArch() != Triple::x86_64 && TT.getArch() != Triple::x86) ||
|
|
|
|
TT.isOSWindows())
|
2015-10-30 11:20:21 +08:00
|
|
|
TM = nullptr;
|
|
|
|
}
|
|
|
|
};
|
2015-10-28 10:40:04 +08:00
|
|
|
|
2015-10-30 11:20:21 +08:00
|
|
|
protected:
|
2016-04-15 05:59:01 +08:00
|
|
|
LLVMContext Context;
|
2015-10-30 11:20:21 +08:00
|
|
|
std::unique_ptr<TargetMachine> TM;
|
2015-10-28 10:40:04 +08:00
|
|
|
};
|
|
|
|
|
2015-10-28 01:45:48 +08:00
|
|
|
class ModuleBuilder {
|
|
|
|
public:
|
|
|
|
ModuleBuilder(LLVMContext &Context, StringRef Triple,
|
|
|
|
StringRef Name);
|
2015-04-21 04:41:45 +08:00
|
|
|
|
2015-10-28 01:45:48 +08:00
|
|
|
template <typename FuncType>
|
2015-10-28 10:40:04 +08:00
|
|
|
Function* createFunctionDecl(StringRef Name) {
|
2015-10-28 01:45:48 +08:00
|
|
|
return Function::Create(
|
|
|
|
TypeBuilder<FuncType, false>::get(M->getContext()),
|
2015-10-28 10:40:04 +08:00
|
|
|
GlobalValue::ExternalLinkage, Name, M.get());
|
2015-10-28 01:45:48 +08:00
|
|
|
}
|
2015-04-21 04:41:45 +08:00
|
|
|
|
2015-10-28 01:45:48 +08:00
|
|
|
Module* getModule() { return M.get(); }
|
|
|
|
const Module* getModule() const { return M.get(); }
|
|
|
|
std::unique_ptr<Module> takeModule() { return std::move(M); }
|
2015-04-21 04:41:45 +08:00
|
|
|
|
2015-10-28 01:45:48 +08:00
|
|
|
private:
|
|
|
|
std::unique_ptr<Module> M;
|
|
|
|
};
|
2015-04-21 04:41:45 +08:00
|
|
|
|
2015-10-28 01:45:48 +08:00
|
|
|
// Dummy struct type.
|
|
|
|
struct DummyStruct {
|
|
|
|
int X[256];
|
|
|
|
};
|
2015-04-21 04:41:45 +08:00
|
|
|
|
2015-10-28 01:45:48 +08:00
|
|
|
// TypeBuilder specialization for DummyStruct.
|
|
|
|
template <bool XCompile>
|
|
|
|
class TypeBuilder<DummyStruct, XCompile> {
|
|
|
|
public:
|
|
|
|
static StructType *get(LLVMContext &Context) {
|
|
|
|
return StructType::get(
|
2017-05-10 03:31:13 +08:00
|
|
|
TypeBuilder<types::i<32>[256], XCompile>::get(Context));
|
2015-10-28 01:45:48 +08:00
|
|
|
}
|
|
|
|
};
|
2015-04-21 04:41:45 +08:00
|
|
|
|
2017-09-28 10:17:35 +08:00
|
|
|
template <typename HandleT, typename ModuleT>
|
2015-10-20 12:35:02 +08:00
|
|
|
class MockBaseLayer {
|
|
|
|
public:
|
|
|
|
|
2017-09-28 10:17:35 +08: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 12:35:02 +08:00
|
|
|
}
|
|
|
|
|
2017-07-07 10:59:13 +08:00
|
|
|
Error removeModule(ModuleHandleT H) {
|
2017-09-28 10:17:35 +08:00
|
|
|
assert(removeModuleImpl &&
|
|
|
|
"removeModule called, but no mock implementation was provided");
|
|
|
|
return removeModuleImpl(H);
|
2015-10-20 12:35:02 +08:00
|
|
|
}
|
|
|
|
|
2016-08-02 04:49:11 +08:00
|
|
|
JITSymbol findSymbol(const std::string &Name, bool ExportedSymbolsOnly) {
|
2017-09-28 10:17:35 +08:00
|
|
|
assert(findSymbolImpl &&
|
|
|
|
"findSymbol called, but no mock implementation was provided");
|
|
|
|
return findSymbolImpl(Name, ExportedSymbolsOnly);
|
2015-10-20 12:35:02 +08:00
|
|
|
}
|
|
|
|
|
2017-06-24 07:25:28 +08:00
|
|
|
JITSymbol findSymbolIn(ModuleHandleT H, const std::string &Name,
|
2015-10-20 12:35:02 +08:00
|
|
|
bool ExportedSymbolsOnly) {
|
2017-09-28 10:17:35 +08:00
|
|
|
assert(findSymbolInImpl &&
|
|
|
|
"findSymbolIn called, but no mock implementation was provided");
|
|
|
|
return findSymbolInImpl(H, Name, ExportedSymbolsOnly);
|
2015-10-20 12:35:02 +08:00
|
|
|
}
|
|
|
|
|
2017-09-28 10:17:35 +08:00
|
|
|
Error emitAndFinaliez(ModuleHandleT H) {
|
|
|
|
assert(emitAndFinalizeImpl &&
|
|
|
|
"emitAndFinalize called, but no mock implementation was provided");
|
|
|
|
return emitAndFinalizeImpl(H);
|
|
|
|
}
|
2015-10-20 12:35:02 +08:00
|
|
|
};
|
|
|
|
|
2017-07-07 10:59:13 +08:00
|
|
|
class ReturnNullJITSymbol {
|
|
|
|
public:
|
|
|
|
template <typename... Args>
|
|
|
|
JITSymbol operator()(Args...) const {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-10-20 12:35:02 +08:00
|
|
|
template <typename ReturnT>
|
|
|
|
class DoNothingAndReturn {
|
|
|
|
public:
|
2017-07-07 10:59:13 +08:00
|
|
|
DoNothingAndReturn(ReturnT Ret) : Ret(std::move(Ret)) {}
|
2015-10-20 12:35:02 +08:00
|
|
|
|
|
|
|
template <typename... Args>
|
2017-07-07 10:59:13 +08:00
|
|
|
void operator()(Args...) const { return Ret; }
|
2015-10-20 12:35:02 +08:00
|
|
|
private:
|
2017-07-07 10:59:13 +08:00
|
|
|
ReturnT Ret;
|
2015-10-20 12:35:02 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
class DoNothingAndReturn<void> {
|
|
|
|
public:
|
|
|
|
template <typename... Args>
|
|
|
|
void operator()(Args...) const { }
|
|
|
|
};
|
2015-04-21 04:41:45 +08:00
|
|
|
|
|
|
|
} // namespace llvm
|
|
|
|
|
|
|
|
#endif
|