annotate llvm/lib/ExecutionEngine/Orc/Speculation.cpp @ 181:df311c476dd5

CreateIdentifierInfo in ParseCbC (not yet worked)
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Sun, 31 May 2020 12:30:11 +0900
parents 1d019706d866
children 2e18cbf3894f
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===---------- speculation.cpp - Utilities for Speculation ----------===//
anatofuz
parents:
diff changeset
2 //
anatofuz
parents:
diff changeset
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
anatofuz
parents:
diff changeset
4 // See https://llvm.org/LICENSE.txt for license information.
anatofuz
parents:
diff changeset
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
anatofuz
parents:
diff changeset
6 //
anatofuz
parents:
diff changeset
7 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
8
anatofuz
parents:
diff changeset
9 #include "llvm/ExecutionEngine/Orc/Speculation.h"
anatofuz
parents:
diff changeset
10 #include "llvm/IR/BasicBlock.h"
anatofuz
parents:
diff changeset
11 #include "llvm/IR/Function.h"
anatofuz
parents:
diff changeset
12 #include "llvm/IR/IRBuilder.h"
anatofuz
parents:
diff changeset
13 #include "llvm/IR/Instruction.h"
anatofuz
parents:
diff changeset
14 #include "llvm/IR/Instructions.h"
anatofuz
parents:
diff changeset
15 #include "llvm/IR/LLVMContext.h"
anatofuz
parents:
diff changeset
16 #include "llvm/IR/Module.h"
anatofuz
parents:
diff changeset
17 #include "llvm/IR/Type.h"
anatofuz
parents:
diff changeset
18 #include "llvm/IR/Verifier.h"
anatofuz
parents:
diff changeset
19 #include "llvm/Support/Debug.h"
anatofuz
parents:
diff changeset
20
anatofuz
parents:
diff changeset
21 #include <vector>
anatofuz
parents:
diff changeset
22
anatofuz
parents:
diff changeset
23 namespace llvm {
anatofuz
parents:
diff changeset
24
anatofuz
parents:
diff changeset
25 namespace orc {
anatofuz
parents:
diff changeset
26
anatofuz
parents:
diff changeset
27 // ImplSymbolMap methods
anatofuz
parents:
diff changeset
28 void ImplSymbolMap::trackImpls(SymbolAliasMap ImplMaps, JITDylib *SrcJD) {
anatofuz
parents:
diff changeset
29 assert(SrcJD && "Tracking on Null Source .impl dylib");
anatofuz
parents:
diff changeset
30 std::lock_guard<std::mutex> Lockit(ConcurrentAccess);
anatofuz
parents:
diff changeset
31 for (auto &I : ImplMaps) {
anatofuz
parents:
diff changeset
32 auto It = Maps.insert({I.first, {I.second.Aliasee, SrcJD}});
anatofuz
parents:
diff changeset
33 // check rationale when independent dylibs have same symbol name?
anatofuz
parents:
diff changeset
34 assert(It.second && "ImplSymbols are already tracked for this Symbol?");
anatofuz
parents:
diff changeset
35 (void)(It);
anatofuz
parents:
diff changeset
36 }
anatofuz
parents:
diff changeset
37 }
anatofuz
parents:
diff changeset
38
anatofuz
parents:
diff changeset
39 // Trigger Speculative Compiles.
anatofuz
parents:
diff changeset
40 void Speculator::speculateForEntryPoint(Speculator *Ptr, uint64_t StubId) {
anatofuz
parents:
diff changeset
41 assert(Ptr && " Null Address Received in orc_speculate_for ");
anatofuz
parents:
diff changeset
42 Ptr->speculateFor(StubId);
anatofuz
parents:
diff changeset
43 }
anatofuz
parents:
diff changeset
44
anatofuz
parents:
diff changeset
45 Error Speculator::addSpeculationRuntime(JITDylib &JD,
anatofuz
parents:
diff changeset
46 MangleAndInterner &Mangle) {
anatofuz
parents:
diff changeset
47 JITEvaluatedSymbol ThisPtr(pointerToJITTargetAddress(this),
anatofuz
parents:
diff changeset
48 JITSymbolFlags::Exported);
anatofuz
parents:
diff changeset
49 JITEvaluatedSymbol SpeculateForEntryPtr(
anatofuz
parents:
diff changeset
50 pointerToJITTargetAddress(&speculateForEntryPoint),
anatofuz
parents:
diff changeset
51 JITSymbolFlags::Exported);
anatofuz
parents:
diff changeset
52 return JD.define(absoluteSymbols({
anatofuz
parents:
diff changeset
53 {Mangle("__orc_speculator"), ThisPtr}, // Data Symbol
anatofuz
parents:
diff changeset
54 {Mangle("__orc_speculate_for"), SpeculateForEntryPtr} // Callable Symbol
anatofuz
parents:
diff changeset
55 }));
anatofuz
parents:
diff changeset
56 }
anatofuz
parents:
diff changeset
57
anatofuz
parents:
diff changeset
58 // If two modules, share the same LLVMContext, different threads must
anatofuz
parents:
diff changeset
59 // not access them concurrently without locking the associated LLVMContext
anatofuz
parents:
diff changeset
60 // this implementation follows this contract.
anatofuz
parents:
diff changeset
61 void IRSpeculationLayer::emit(MaterializationResponsibility R,
anatofuz
parents:
diff changeset
62 ThreadSafeModule TSM) {
anatofuz
parents:
diff changeset
63
anatofuz
parents:
diff changeset
64 assert(TSM && "Speculation Layer received Null Module ?");
anatofuz
parents:
diff changeset
65 assert(TSM.getContext().getContext() != nullptr &&
anatofuz
parents:
diff changeset
66 "Module with null LLVMContext?");
anatofuz
parents:
diff changeset
67
anatofuz
parents:
diff changeset
68 // Instrumentation of runtime calls, lock the Module
anatofuz
parents:
diff changeset
69 TSM.withModuleDo([this, &R](Module &M) {
anatofuz
parents:
diff changeset
70 auto &MContext = M.getContext();
anatofuz
parents:
diff changeset
71 auto SpeculatorVTy = StructType::create(MContext, "Class.Speculator");
anatofuz
parents:
diff changeset
72 auto RuntimeCallTy = FunctionType::get(
anatofuz
parents:
diff changeset
73 Type::getVoidTy(MContext),
anatofuz
parents:
diff changeset
74 {SpeculatorVTy->getPointerTo(), Type::getInt64Ty(MContext)}, false);
anatofuz
parents:
diff changeset
75 auto RuntimeCall =
anatofuz
parents:
diff changeset
76 Function::Create(RuntimeCallTy, Function::LinkageTypes::ExternalLinkage,
anatofuz
parents:
diff changeset
77 "__orc_speculate_for", &M);
anatofuz
parents:
diff changeset
78 auto SpeclAddr = new GlobalVariable(
anatofuz
parents:
diff changeset
79 M, SpeculatorVTy, false, GlobalValue::LinkageTypes::ExternalLinkage,
anatofuz
parents:
diff changeset
80 nullptr, "__orc_speculator");
anatofuz
parents:
diff changeset
81
anatofuz
parents:
diff changeset
82 IRBuilder<> Mutator(MContext);
anatofuz
parents:
diff changeset
83
anatofuz
parents:
diff changeset
84 // QueryAnalysis allowed to transform the IR source, one such example is
anatofuz
parents:
diff changeset
85 // Simplify CFG helps the static branch prediction heuristics!
anatofuz
parents:
diff changeset
86 for (auto &Fn : M.getFunctionList()) {
anatofuz
parents:
diff changeset
87 if (!Fn.isDeclaration()) {
anatofuz
parents:
diff changeset
88
anatofuz
parents:
diff changeset
89 auto IRNames = QueryAnalysis(Fn);
anatofuz
parents:
diff changeset
90 // Instrument and register if Query has result
anatofuz
parents:
diff changeset
91 if (IRNames.hasValue()) {
anatofuz
parents:
diff changeset
92
anatofuz
parents:
diff changeset
93 // Emit globals for each function.
anatofuz
parents:
diff changeset
94 auto LoadValueTy = Type::getInt8Ty(MContext);
anatofuz
parents:
diff changeset
95 auto SpeculatorGuard = new GlobalVariable(
anatofuz
parents:
diff changeset
96 M, LoadValueTy, false, GlobalValue::LinkageTypes::InternalLinkage,
anatofuz
parents:
diff changeset
97 ConstantInt::get(LoadValueTy, 0),
anatofuz
parents:
diff changeset
98 "__orc_speculate.guard.for." + Fn.getName());
anatofuz
parents:
diff changeset
99 SpeculatorGuard->setAlignment(Align(1));
anatofuz
parents:
diff changeset
100 SpeculatorGuard->setUnnamedAddr(GlobalValue::UnnamedAddr::Local);
anatofuz
parents:
diff changeset
101
anatofuz
parents:
diff changeset
102 BasicBlock &ProgramEntry = Fn.getEntryBlock();
anatofuz
parents:
diff changeset
103 // Create BasicBlocks before the program's entry basicblock
anatofuz
parents:
diff changeset
104 BasicBlock *SpeculateBlock = BasicBlock::Create(
anatofuz
parents:
diff changeset
105 MContext, "__orc_speculate.block", &Fn, &ProgramEntry);
anatofuz
parents:
diff changeset
106 BasicBlock *SpeculateDecisionBlock = BasicBlock::Create(
anatofuz
parents:
diff changeset
107 MContext, "__orc_speculate.decision.block", &Fn, SpeculateBlock);
anatofuz
parents:
diff changeset
108
anatofuz
parents:
diff changeset
109 assert(SpeculateDecisionBlock == &Fn.getEntryBlock() &&
anatofuz
parents:
diff changeset
110 "SpeculateDecisionBlock not updated?");
anatofuz
parents:
diff changeset
111 Mutator.SetInsertPoint(SpeculateDecisionBlock);
anatofuz
parents:
diff changeset
112
anatofuz
parents:
diff changeset
113 auto LoadGuard =
anatofuz
parents:
diff changeset
114 Mutator.CreateLoad(LoadValueTy, SpeculatorGuard, "guard.value");
anatofuz
parents:
diff changeset
115 // if just loaded value equal to 0,return true.
anatofuz
parents:
diff changeset
116 auto CanSpeculate =
anatofuz
parents:
diff changeset
117 Mutator.CreateICmpEQ(LoadGuard, ConstantInt::get(LoadValueTy, 0),
anatofuz
parents:
diff changeset
118 "compare.to.speculate");
anatofuz
parents:
diff changeset
119 Mutator.CreateCondBr(CanSpeculate, SpeculateBlock, &ProgramEntry);
anatofuz
parents:
diff changeset
120
anatofuz
parents:
diff changeset
121 Mutator.SetInsertPoint(SpeculateBlock);
anatofuz
parents:
diff changeset
122 auto ImplAddrToUint =
anatofuz
parents:
diff changeset
123 Mutator.CreatePtrToInt(&Fn, Type::getInt64Ty(MContext));
anatofuz
parents:
diff changeset
124 Mutator.CreateCall(RuntimeCallTy, RuntimeCall,
anatofuz
parents:
diff changeset
125 {SpeclAddr, ImplAddrToUint});
anatofuz
parents:
diff changeset
126 Mutator.CreateStore(ConstantInt::get(LoadValueTy, 1),
anatofuz
parents:
diff changeset
127 SpeculatorGuard);
anatofuz
parents:
diff changeset
128 Mutator.CreateBr(&ProgramEntry);
anatofuz
parents:
diff changeset
129
anatofuz
parents:
diff changeset
130 assert(Mutator.GetInsertBlock()->getParent() == &Fn &&
anatofuz
parents:
diff changeset
131 "IR builder association mismatch?");
anatofuz
parents:
diff changeset
132 S.registerSymbols(internToJITSymbols(IRNames.getValue()),
anatofuz
parents:
diff changeset
133 &R.getTargetJITDylib());
anatofuz
parents:
diff changeset
134 }
anatofuz
parents:
diff changeset
135 }
anatofuz
parents:
diff changeset
136 }
anatofuz
parents:
diff changeset
137 });
anatofuz
parents:
diff changeset
138
anatofuz
parents:
diff changeset
139 assert(!TSM.withModuleDo([](const Module &M) { return verifyModule(M); }) &&
anatofuz
parents:
diff changeset
140 "Speculation Instrumentation breaks IR?");
anatofuz
parents:
diff changeset
141
anatofuz
parents:
diff changeset
142 NextLayer.emit(std::move(R), std::move(TSM));
anatofuz
parents:
diff changeset
143 }
anatofuz
parents:
diff changeset
144
anatofuz
parents:
diff changeset
145 } // namespace orc
anatofuz
parents:
diff changeset
146 } // namespace llvm