#include "llvm/ExecutionEngine/Orc/Speculation.h"
#include "llvm/IR/BasicBlock.h"
#include "llvm/IR/Function.h"
#include "llvm/IR/IRBuilder.h"
#include "llvm/IR/Instruction.h"
#include "llvm/IR/Instructions.h"
#include "llvm/IR/LLVMContext.h"
#include "llvm/IR/Module.h"
#include "llvm/IR/Type.h"
#include "llvm/IR/Verifier.h"
namespace llvm {
namespace orc {
void ImplSymbolMap::trackImpls(SymbolAliasMap ImplMaps, JITDylib *SrcJD) {
assert(SrcJD && "Tracking on Null Source .impl dylib");
std::lock_guard<std::mutex> Lockit(ConcurrentAccess);
for (auto &I : ImplMaps) {
auto It = Maps.insert({I.first, {I.second.Aliasee, SrcJD}});
assert(It.second && "ImplSymbols are already tracked for this Symbol?");
(void)(It);
}
}
void Speculator::speculateForEntryPoint(Speculator *Ptr, uint64_t StubId) {
assert(Ptr && " Null Address Received in orc_speculate_for ");
Ptr->speculateFor(StubId);
}
Error Speculator::addSpeculationRuntime(JITDylib &JD,
MangleAndInterner &Mangle) {
JITEvaluatedSymbol ThisPtr(pointerToJITTargetAddress(this),
JITSymbolFlags::Exported);
JITEvaluatedSymbol SpeculateForEntryPtr(
pointerToJITTargetAddress(&speculateForEntryPoint),
JITSymbolFlags::Exported);
return JD.define(absoluteSymbols({
{Mangle("__orc_speculator"), ThisPtr}, {Mangle("__orc_speculate_for"), SpeculateForEntryPtr} }));
}
void IRSpeculationLayer::emit(std::unique_ptr<MaterializationResponsibility> R,
ThreadSafeModule TSM) {
assert(TSM && "Speculation Layer received Null Module ?");
assert(TSM.getContext().getContext() != nullptr &&
"Module with null LLVMContext?");
TSM.withModuleDo([this, &R](Module &M) {
auto &MContext = M.getContext();
auto SpeculatorVTy = StructType::create(MContext, "Class.Speculator");
auto RuntimeCallTy = FunctionType::get(
Type::getVoidTy(MContext),
{SpeculatorVTy->getPointerTo(), Type::getInt64Ty(MContext)}, false);
auto RuntimeCall =
Function::Create(RuntimeCallTy, Function::LinkageTypes::ExternalLinkage,
"__orc_speculate_for", &M);
auto SpeclAddr = new GlobalVariable(
M, SpeculatorVTy, false, GlobalValue::LinkageTypes::ExternalLinkage,
nullptr, "__orc_speculator");
IRBuilder<> Mutator(MContext);
for (auto &Fn : M.getFunctionList()) {
if (!Fn.isDeclaration()) {
auto IRNames = QueryAnalysis(Fn);
if (IRNames) {
auto LoadValueTy = Type::getInt8Ty(MContext);
auto SpeculatorGuard = new GlobalVariable(
M, LoadValueTy, false, GlobalValue::LinkageTypes::InternalLinkage,
ConstantInt::get(LoadValueTy, 0),
"__orc_speculate.guard.for." + Fn.getName());
SpeculatorGuard->setAlignment(Align(1));
SpeculatorGuard->setUnnamedAddr(GlobalValue::UnnamedAddr::Local);
BasicBlock &ProgramEntry = Fn.getEntryBlock();
BasicBlock *SpeculateBlock = BasicBlock::Create(
MContext, "__orc_speculate.block", &Fn, &ProgramEntry);
BasicBlock *SpeculateDecisionBlock = BasicBlock::Create(
MContext, "__orc_speculate.decision.block", &Fn, SpeculateBlock);
assert(SpeculateDecisionBlock == &Fn.getEntryBlock() &&
"SpeculateDecisionBlock not updated?");
Mutator.SetInsertPoint(SpeculateDecisionBlock);
auto LoadGuard =
Mutator.CreateLoad(LoadValueTy, SpeculatorGuard, "guard.value");
auto CanSpeculate =
Mutator.CreateICmpEQ(LoadGuard, ConstantInt::get(LoadValueTy, 0),
"compare.to.speculate");
Mutator.CreateCondBr(CanSpeculate, SpeculateBlock, &ProgramEntry);
Mutator.SetInsertPoint(SpeculateBlock);
auto ImplAddrToUint =
Mutator.CreatePtrToInt(&Fn, Type::getInt64Ty(MContext));
Mutator.CreateCall(RuntimeCallTy, RuntimeCall,
{SpeclAddr, ImplAddrToUint});
Mutator.CreateStore(ConstantInt::get(LoadValueTy, 1),
SpeculatorGuard);
Mutator.CreateBr(&ProgramEntry);
assert(Mutator.GetInsertBlock()->getParent() == &Fn &&
"IR builder association mismatch?");
S.registerSymbols(internToJITSymbols(*IRNames),
&R->getTargetJITDylib());
}
}
}
});
assert(!TSM.withModuleDo([](const Module &M) { return verifyModule(M); }) &&
"Speculation Instrumentation breaks IR?");
NextLayer.emit(std::move(R), std::move(TSM));
}
} }