79#define DEBUG_TYPE "gvn-hoist"
81STATISTIC(NumHoisted,
"Number of instructions hoisted");
82STATISTIC(NumRemoved,
"Number of instructions removed");
83STATISTIC(NumLoadsHoisted,
"Number of loads hoisted");
84STATISTIC(NumLoadsRemoved,
"Number of loads removed");
85STATISTIC(NumStoresHoisted,
"Number of stores hoisted");
86STATISTIC(NumStoresRemoved,
"Number of stores removed");
87STATISTIC(NumCallsHoisted,
"Number of calls hoisted");
88STATISTIC(NumCallsRemoved,
"Number of calls removed");
92 cl::desc(
"Max number of instructions to hoist "
93 "(default unlimited = -1)"));
97 cl::desc(
"Max number of basic blocks on the path between "
98 "hoisting locations (default = 4, unlimited = -1)"));
102 cl::desc(
"Hoist instructions from the beginning of the BB up to the "
103 "maximum specified depth (default = 100, unlimited = -1)"));
107 cl::desc(
"Maximum length of dependent chains to hoist "
108 "(default = 10, unlimited = -1)"));
123using VNType = std::pair<unsigned, uintptr_t>;
182 if (Load->isSimple()) {
183 unsigned V = VN.
lookupOrAdd(Load->getPointerOperand());
186 VNtoLoads[{V, (uintptr_t)Load->getType()}].push_back(Load);
201 if (!Store->isSimple())
204 Value *
Ptr = Store->getPointerOperand();
205 Value *Val = Store->getValueOperand();
225 auto Entry = std::make_pair(V,
InvalidVN);
227 if (
Call->doesNotAccessMemory())
228 VNtoCallsScalars[Entry].push_back(
Call);
229 else if (
Call->onlyReadsMemory())
230 VNtoCallsLoads[Entry].push_back(
Call);
232 VNtoCallsStores[Entry].push_back(
Call);
247 : DT(DT), PDT(PDT), AA(AA), MD(MD), MSSA(MSSA),
249 MSSA->ensureOptimizedUses();
269 std::unique_ptr<MemorySSAUpdater> MSSAUpdater;
274 unsigned NumFuncArgs;
275 const bool HoistingGeps =
false;
277 enum InsKind {
Unknown, Scalar, Load, Store };
285 unsigned I1DFS = DFSNumber.
lookup(I1);
286 unsigned I2DFS = DFSNumber.
lookup(I2);
288 return I1DFS < I2DFS;
292 bool hasMemoryUse(
const Instruction *NewPt, MemoryDef *Def,
293 const BasicBlock *BB);
295 bool hasEHhelper(
const BasicBlock *BB,
const BasicBlock *SrcBB,
296 int &NBBsOnAllPaths);
305 bool hasEHOrLoadsOnPath(
const Instruction *NewPt, MemoryDef *Def,
306 int &NBBsOnAllPaths);
312 bool hasEHOnPath(
const BasicBlock *HoistPt,
const BasicBlock *SrcBB,
313 int &NBBsOnAllPaths);
317 bool safeToHoistLdSt(
const Instruction *NewPt,
const Instruction *OldPt,
318 MemoryUseOrDef *U, InsKind K,
int &NBBsOnAllPaths);
322 bool safeToHoistScalar(
const BasicBlock *HoistBB,
const BasicBlock *BB,
323 int &NBBsOnAllPaths) {
324 return !hasEHOnPath(HoistBB, BB, NBBsOnAllPaths);
341 bool valueAnticipable(
CHIArgs C, Instruction *TI)
const;
345 void checkSafety(
CHIArgs C, BasicBlock *BB, InsKind K,
346 SmallVectorImpl<CHIArg> &Safe);
348 using RenameStackType = DenseMap<VNType, SmallVector<Instruction *, 2>>;
351 void fillRenameStack(BasicBlock *BB,
InValuesType &ValueBBs,
352 RenameStackType &RenameStack);
355 RenameStackType &RenameStack);
362 auto Root = PDT->getNode(
nullptr);
371 RenameStackType RenameStack;
373 fillRenameStack(BB, ValueBBs, RenameStack);
376 fillChiArgs(BB, CHIBBs, RenameStack);
384 void findHoistableCandidates(
OutValuesType &CHIBBs, InsKind K,
392 std::vector<VNType> Ranks;
393 for (
const auto &Entry : Map) {
394 Ranks.push_back(
Entry.first);
416 for (
const auto &R : Ranks) {
421 SmallPtrSet<BasicBlock *, 2> VNBlocks;
422 for (
const auto &
I : V) {
433 IDFs.setDefiningBlocks(VNBlocks);
435 IDFs.calculate(IDFBlocks);
438 for (
unsigned i = 0; i <
V.size(); ++i) {
439 InValue[
V[i]->getParent()].push_back(std::make_pair(VN, V[i]));
443 CHIArg EmptyChi = {VN,
nullptr,
nullptr};
444 for (
auto *IDFBB : IDFBlocks) {
445 for (
unsigned i = 0; i <
V.size(); ++i) {
447 if (DT->properlyDominates(IDFBB, V[i]->getParent())) {
448 OutValue[IDFBB].push_back(EmptyChi);
450 << IDFBB->getName() <<
", for Insn: " << *V[i]);
458 insertCHI(InValue, OutValue);
460 findHoistableCandidates(OutValue, K, HPL);
467 bool allOperandsAvailable(
const Instruction *
I,
468 const BasicBlock *HoistPt)
const;
471 bool allGepOperandsAvailable(
const Instruction *
I,
472 const BasicBlock *HoistPt)
const;
475 void makeGepsAvailable(Instruction *Repl, BasicBlock *HoistPt,
477 Instruction *Gep)
const;
479 void updateAlignment(Instruction *
I, Instruction *Repl);
483 unsigned rauw(
const SmallVecInsn &Candidates, Instruction *Repl,
484 MemoryUseOrDef *NewMemAcc);
487 void raMPHIuw(MemoryUseOrDef *NewMemAcc);
490 unsigned removeAndReplace(
const SmallVecInsn &Candidates, Instruction *Repl,
491 BasicBlock *DestBB,
bool MoveAccess);
496 bool makeGepOperandsAvailable(Instruction *Repl, BasicBlock *HoistPt,
503 std::pair<unsigned, unsigned> hoistExpressions(Function &
F);
507 NumFuncArgs =
F.arg_size();
509 VN.setAliasAnalysis(AA);
515 DFSNumber[BB] = ++BBI;
517 for (
const auto &Inst : *BB)
518 DFSNumber[&Inst] = ++
I;
528 auto HoistStat = hoistExpressions(
F);
529 if (HoistStat.first + HoistStat.second == 0)
532 if (HoistStat.second > 0)
555 return 3 +
A->getArgNo();
559 auto Result = DFSNumber.lookup(V);
561 return 4 + NumFuncArgs + Result;
567 auto [It, Inserted] = BBSideEffects.
try_emplace(BB);
593 bool ReachedNewPt =
false;
595 for (
const MemoryAccess &MA : *Acc)
600 if (BB == OldBB && firstInBB(OldPt, Insn))
606 if (firstInBB(Insn, NewPt))
619 int &NBBsOnAllPaths) {
621 if (NBBsOnAllPaths == 0)
631 if ((BB != SrcBB) && HoistBarrier.count(BB))
638 int &NBBsOnAllPaths) {
641 assert(DT->dominates(NewBB, OldBB) &&
"invalid path");
642 assert(DT->dominates(
Def->getDefiningAccess()->getBlock(), NewBB) &&
643 "def does not dominate new hoisting point");
657 if (hasEHhelper(BB, OldBB, NBBsOnAllPaths))
661 if (hasMemoryUse(NewPt, Def, BB))
665 if (NBBsOnAllPaths != -1)
675 int &NBBsOnAllPaths) {
676 assert(DT->dominates(HoistPt, SrcBB) &&
"Invalid path");
691 if (hasEHhelper(BB, SrcBB, NBBsOnAllPaths))
695 if (NBBsOnAllPaths != -1)
704bool GVNHoist::safeToHoistLdSt(
const Instruction *NewPt,
706 GVNHoist::InsKind K,
int &NBBsOnAllPaths) {
716 MemoryAccess *
D =
U->getDefiningAccess();
718 if (DT->properlyDominates(NewBB, DBB))
722 if (NewBB == DBB && !MSSA->isLiveOnEntryDef(
D))
724 if (!firstInBB(UD->getMemoryInst(), NewPt))
729 if (K == InsKind::Store) {
732 }
else if (hasEHOnPath(NewBB, OldBB, NBBsOnAllPaths))
736 if (DT->properlyDominates(DBB, NewBB))
739 assert(MSSA->locallyDominates(
D, U));
771 if (K == InsKind::Scalar) {
772 if (safeToHoistScalar(BB, Insn->
getParent(), NumBBsOnAllPaths))
775 if (MemoryUseOrDef *UD = MSSA->getMemoryAccess(Insn))
776 if (safeToHoistLdSt(
T, Insn, UD, K, NumBBsOnAllPaths))
783 GVNHoist::RenameStackType &RenameStack) {
784 auto it1 = ValueBBs.find(BB);
785 if (it1 != ValueBBs.end()) {
788 <<
" for pushing instructions on stack";);
789 for (std::pair<VNType, Instruction *> &VI :
reverse(it1->second)) {
792 RenameStack[
VI.first].push_back(
VI.second);
798 GVNHoist::RenameStackType &RenameStack) {
801 auto P = CHIBBs.find(Pred);
802 if (
P == CHIBBs.end()) {
805 LLVM_DEBUG(
dbgs() <<
"\nLooking at CHIs in: " << Pred->getName(););
808 auto &VCHI =
P->second;
809 for (
auto It = VCHI.begin(),
E = VCHI.end(); It !=
E;) {
812 auto si = RenameStack.find(
C.VN);
816 if (si != RenameStack.end() && si->second.size() &&
817 DT->properlyDominates(Pred, si->second.back()->getParent())) {
819 C.I = si->second.pop_back_val();
821 <<
"\nCHI Inserted in BB: " <<
C.Dest->getName() << *
C.I
822 <<
", VN: " <<
C.VN.first <<
", " <<
C.VN.second);
825 It = std::find_if(It, VCHI.end(), [It](CHIArg &
A) { return A != *It; });
835 auto cmpVN = [](
const CHIArg &
A,
const CHIArg &
B) {
return A.VN <
B.VN; };
841 SmallVectorImpl<CHIArg> &CHIs =
A.second;
850 auto PrevIt = CHIs.
begin();
851 while (PrevIt != PHIIt) {
858 checkSafety(
make_range(PrevIt, PHIIt), BB, K, Safe);
870 PHIIt = std::find_if(PrevIt, CHIs.
end(),
871 [PrevIt](CHIArg &
A) { return A != *PrevIt; });
876bool GVNHoist::allOperandsAvailable(
const Instruction *
I,
878 for (
const Use &
Op :
I->operands())
880 if (!DT->dominates(Inst->getParent(), HoistPt))
886bool GVNHoist::allGepOperandsAvailable(
const Instruction *
I,
888 for (
const Use &
Op :
I->operands())
890 if (!DT->dominates(Inst->getParent(), HoistPt)) {
891 if (
const GetElementPtrInst *GepOp =
893 if (!allGepOperandsAvailable(GepOp, HoistPt))
908 assert(allGepOperandsAvailable(Gep, HoistPt) &&
"GEP operands not available");
914 if (DT->dominates(
Op->getParent(), HoistPt))
920 makeGepsAvailable(ClonedGep, HoistPt, InstructionsToHoist, GepOp);
932 for (
const Instruction *OtherInst : InstructionsToHoist) {
933 const GetElementPtrInst *OtherGep;
944 if (OtherGep != Gep) {
956 ReplacementLoad->setAlignment(
960 ReplacementStore->setAlignment(
964 ReplacementAlloca->setAlignment(std::max(ReplacementAlloca->getAlign(),
974 for (Instruction *
I : Candidates) {
977 updateAlignment(
I, Repl);
980 MemoryAccess *OldMA = MSSA->getMemoryAccess(
I);
982 MSSAUpdater->removeMemoryAccess(OldMA);
987 I->replaceAllUsesWith(Repl);
989 MD->removeInstruction(
I);
990 I->eraseFromParent();
997 SmallPtrSet<MemoryPhi *, 4> UsePhis;
998 for (User *U : NewMemAcc->
users())
1002 for (MemoryPhi *Phi : UsePhis) {
1003 auto In =
Phi->incoming_values();
1004 if (
llvm::all_of(In, [&](Use &U) {
return U == NewMemAcc; })) {
1005 Phi->replaceAllUsesWith(NewMemAcc);
1006 MSSAUpdater->removeMemoryAccess(Phi);
1011unsigned GVNHoist::removeAndReplace(
const SmallVecInsn &Candidates,
1014 MemoryUseOrDef *NewMemAcc = MSSA->getMemoryAccess(Repl);
1015 if (MoveAccess && NewMemAcc) {
1022 unsigned NR = rauw(Candidates, Repl, NewMemAcc);
1026 raMPHIuw(NewMemAcc);
1030bool GVNHoist::makeGepOperandsAvailable(
1034 GetElementPtrInst *Gep =
nullptr;
1045 if (!allGepOperandsAvailable(Val, HoistPt))
1047 }
else if (!DT->dominates(Val->
getParent(), HoistPt))
1053 if (!Gep || !allGepOperandsAvailable(Gep, HoistPt))
1056 makeGepsAvailable(Repl, HoistPt, InstructionsToHoist, Gep);
1059 makeGepsAvailable(Repl, HoistPt, InstructionsToHoist, Val);
1065 unsigned NI = 0, NL = 0, NS = 0,
NC = 0, NR = 0;
1072 for (Instruction *
I : InstructionsToHoist)
1073 if (
I->getParent() == DestBB)
1077 if (!Repl || firstInBB(
I, Repl))
1082 bool MoveAccess =
true;
1085 assert(allOperandsAvailable(Repl, DestBB) &&
1086 "instruction depends on operands that are not available");
1091 Repl = InstructionsToHoist.front();
1096 if (!allOperandsAvailable(Repl, DestBB)) {
1103 if (!makeGepOperandsAvailable(Repl, DestBB, InstructionsToHoist))
1109 MD->removeInstruction(Repl);
1112 DFSNumber[Repl] = DFSNumber[
Last]++;
1117 NR += removeAndReplace(InstructionsToHoist, Repl, DestBB, MoveAccess);
1130 MSSA->verifyMemorySSA();
1132 NumHoisted += NL + NS +
NC + NI;
1134 NumLoadsHoisted += NL;
1135 NumStoresHoisted += NS;
1136 NumCallsHoisted +=
NC;
1137 return {NI, NL +
NC + NS};
1140std::pair<unsigned, unsigned> GVNHoist::hoistExpressions(
Function &
F) {
1145 for (BasicBlock *BB :
depth_first(&
F.getEntryBlock())) {
1146 int InstructionNb = 0;
1147 for (Instruction &I1 : *BB) {
1151 HoistBarrier.insert(BB);
1160 if (
I1.isTerminator())
1164 LI.insert(Load, VN);
1166 SI.insert(Store, VN);
1169 if (Intr->getIntrinsicID() == Intrinsic::assume ||
1170 Intr->getIntrinsicID() == Intrinsic::sideeffect)
1179 CI.insert(
Call, VN);
1190 computeInsertionPoints(
II.getVNTable(), HPL, InsKind::Scalar);
1191 computeInsertionPoints(LI.getVNTable(), HPL, InsKind::Load);
1192 computeInsertionPoints(
SI.getVNTable(), HPL, InsKind::Store);
1193 computeInsertionPoints(CI.getScalarVNTable(), HPL, InsKind::Scalar);
1194 computeInsertionPoints(CI.getLoadVNTable(), HPL, InsKind::Load);
1195 computeInsertionPoints(CI.getStoreVNTable(), HPL, InsKind::Store);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< ErlangGC > A("erlang", "erlang-compatible garbage collector")
static GCRegistry::Add< StatepointGC > D("statepoint-example", "an example strategy for statepoint")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static GCRegistry::Add< OcamlGC > B("ocaml", "ocaml 3.10-compatible GC")
This file contains the declarations for the subclasses of Constant, which represent the different fla...
This file defines the DenseMap class.
This file defines the DenseSet and SmallDenseSet classes.
static cl::opt< int > MaxHoistedThreshold("gvn-max-hoisted", cl::Hidden, cl::init(-1), cl::desc("Max number of instructions to hoist " "(default unlimited = -1)"))
static cl::opt< int > MaxChainLength("gvn-hoist-max-chain-length", cl::Hidden, cl::init(10), cl::desc("Maximum length of dependent chains to hoist " "(default = 10, unlimited = -1)"))
static cl::opt< int > MaxDepthInBB("gvn-hoist-max-depth", cl::Hidden, cl::init(100), cl::desc("Hoist instructions from the beginning of the BB up to the " "maximum specified depth (default = 100, unlimited = -1)"))
static cl::opt< int > MaxNumberOfBBSInPath("gvn-hoist-max-bbs", cl::Hidden, cl::init(4), cl::desc("Max number of basic blocks on the path between " "hoisting locations (default = 4, unlimited = -1)"))
This file provides the interface for LLVM's Global Value Numbering pass which eliminates fully redund...
This is the interface for a simple mod/ref and alias analysis over globals.
This file provides various utilities for inspecting and working with the control flow graph in LLVM I...
This header defines various interfaces for pass management in LLVM.
This defines the Use class.
This file exposes an interface to building/using memory SSA to walk memory instructions using a use/d...
uint64_t IntrinsicInst * II
static void r2(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, int I, uint32_t *Buf)
static void r1(uint32_t &A, uint32_t &B, uint32_t &C, uint32_t &D, uint32_t &E, int I, uint32_t *Buf)
This file defines the SmallPtrSet class.
This file defines the SmallVector class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
A manager for alias analyses.
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
LLVM Basic Block Representation.
bool hasAddressTaken() const
Returns true if there are any uses of this basic block other than direct branches,...
bool isEHPad() const
Return true if this basic block is an exception handling block.
const Instruction * getTerminator() const LLVM_READONLY
Returns the terminator instruction if the block is well formed or null if the block is not well forme...
bool isConvergent() const
Determine if the invoke is convergent.
void insert(CallInst *Call, GVNPass::ValueTable &VN)
const VNtoInsns & getLoadVNTable() const
const VNtoInsns & getScalarVNTable() const
const VNtoInsns & getStoreVNTable() const
This class represents a function call, abstracting a target machine's calling convention.
ValueT lookup(const_arg_type_t< KeyT > Val) const
lookup - Return the entry for the specified key, or a default constructed value if no such entry exis...
std::pair< iterator, bool > try_emplace(KeyT &&Key, Ts &&...Args)
Implements a dense probed hash-table based set.
Analysis pass which computes a DominatorTree.
Concrete subclass of DominatorTreeBase that is used to compute a normal dominator tree.
GVNHoist(DominatorTree *DT, PostDominatorTree *PDT, AliasAnalysis *AA, MemoryDependenceResults *MD, MemorySSA *MSSA)
unsigned int rank(const Value *V) const
This class holds the mapping between values and value numbers.
LLVM_ABI uint32_t lookupOrAdd(MemoryAccess *MA)
const VNtoInsns & getVNTable() const
void insert(Instruction *I, GVNPass::ValueTable &VN)
LLVM_ABI bool mayThrow(bool IncludePhaseOneUnwind=false) const LLVM_READONLY
Return true if this instruction may throw an exception.
LLVM_ABI Instruction * clone() const
Create a copy of 'this' instruction that is identical in all ways except the following:
LLVM_ABI unsigned getNumSuccessors() const LLVM_READONLY
Return the number of successors that this instruction has.
LLVM_ABI void dropLocation()
Drop the instruction's debug location.
const DebugLoc & getDebugLoc() const
Return the debug location for this node as a DebugLoc.
LLVM_ABI void andIRFlags(const Value *V)
Logical 'and' of any supported wrapping, exact, and fast-math flags of V and this instruction.
LLVM_ABI void moveBefore(InstListType::iterator InsertPos)
Unlink this instruction from its current basic block and insert it into the basic block that MovePos ...
LLVM_ABI void insertBefore(InstListType::iterator InsertPos)
Insert an unlinked instruction into a basic block immediately before the specified position.
LLVM_ABI bool mayHaveSideEffects() const LLVM_READONLY
Return true if the instruction may have side effects.
LLVM_ABI void dropUnknownNonDebugMetadata(ArrayRef< unsigned > KnownIDs={})
Drop all unknown metadata except for debug locations.
LLVM_ABI void applyMergedLocation(DebugLoc LocA, DebugLoc LocB)
Merge 2 debug locations and apply it to the Instruction.
const VNtoInsns & getVNTable() const
void insert(LoadInst *Load, GVNPass::ValueTable &VN)
An instruction for reading from memory.
Represents a read-write access to memory, whether it is a must-alias, or a may-alias.
An analysis that produces MemoryDependenceResults for a function.
Provides a lazy, caching interface for making common memory aliasing information queries,...
An analysis that produces MemorySSA for a function.
static LLVM_ABI bool defClobbersUseOrDef(MemoryDef *MD, const MemoryUseOrDef *MU, AliasAnalysis &AA)
Encapsulates MemorySSA, including all data associated with memory accesses.
iplist< MemoryAccess, ilist_tag< MSSAHelpers::AllAccessTag > > AccessList
Class that has the common methods + fields of memory uses/defs.
Analysis pass which computes a PostDominatorTree.
PostDominatorTree Class - Concrete subclass of DominatorTree that is used to compute the post-dominat...
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalyses & preserve()
Mark an analysis as preserved.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
typename SuperClass::iterator iterator
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
void insert(StoreInst *Store, GVNPass::ValueTable &VN)
const VNtoInsns & getVNTable() const
An instruction for storing to memory.
LLVM_ABI bool replaceUsesOfWith(Value *From, Value *To)
Replace uses of one Value with another.
Value * getOperand(unsigned i) const
unsigned getNumOperands() const
LLVM Value Representation.
LLVM_ABI void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
iterator_range< user_iterator > users()
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
const ParentTy * getParent() const
self_iterator getIterator()
A range adaptor for a pair of iterators.
This provides a very simple, boring adaptor for a begin and end iterator into a range type.
Abstract Attribute helper functions.
@ C
The default llvm calling convention, compatible with C.
@ BasicBlock
Various leaf nodes.
initializer< Ty > init(const Ty &Val)
NodeAddr< DefNode * > Def
NodeAddr< PhiNode * > Phi
NodeAddr< NodeBase * > Node
friend class Instruction
Iterator for Instructions in a `BasicBlock.
This is an optimization pass for GlobalISel generic memory operations.
DenseMap< BasicBlock *, SmallVector< std::pair< VNType, Instruction * >, 2 > > InValuesType
void stable_sort(R &&Range)
SmallVector< HoistingPointInfo, 4 > HoistingPointList
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
auto size(R &&Range, std::enable_if_t< std::is_base_of< std::random_access_iterator_tag, typename std::iterator_traits< decltype(Range.begin())>::iterator_category >::value, void > *=nullptr)
Get the size of a range.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
auto successors(const MachineBasicBlock *BB)
iterator_range< T > make_range(T x, T y)
Convenience function for iterating over sub-ranges.
SmallVectorImpl< Instruction * > SmallVecImplInsn
SmallVector< Instruction *, 4 > SmallVecInsn
const Value * getPointerOperand(const Value *V)
A helper function that returns the pointer operand of a load, store or GEP instruction.
SmallVectorImpl< CHIArg >::iterator CHIIt
DenseMap< VNType, SmallVector< Instruction *, 4 > > VNtoInsns
auto reverse(ContainerTy &&C)
std::pair< unsigned, uintptr_t > VNType
IDFCalculator< true > ReverseIDFCalculator
void sort(IteratorTy Start, IteratorTy End)
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
std::pair< BasicBlock *, SmallVecInsn > HoistingPointInfo
idf_iterator< T > idf_end(const T &G)
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
bool isa(const From &Val)
isa<X> - Return true if the parameter to the template is an instance of one of the template type argu...
LLVM_ABI void combineMetadataForCSE(Instruction *K, const Instruction *J, bool DoesKMove)
Combine the metadata of two instructions so that K can replace J.
LLVM_ABI bool VerifyMemorySSA
Enables verification of MemorySSA.
DWARFExpression::Operation Op
idf_iterator< T > idf_begin(const T &G)
DenseMap< BasicBlock *, SmallVector< CHIArg, 2 > > OutValuesType
LLVM_ABI bool isGuaranteedToTransferExecutionToSuccessor(const Instruction *I)
Return true if this function can prove that the instruction I will always transfer execution to one o...
decltype(auto) cast(const From &Val)
cast<X> - Return the argument parameter cast to the specified type.
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
iterator_range< CHIIt > CHIArgs
auto predecessors(const MachineBasicBlock *BB)
bool is_contained(R &&Range, const E &Element)
Returns true if Element is found in Range.
iterator_range< df_iterator< T > > depth_first(const T &G)
AnalysisManager< Function > FunctionAnalysisManager
Convenience typedef for the Function analysis manager.
AAResults AliasAnalysis
Temporary typedef for legacy code that uses a generic AliasAnalysis pointer or reference.
DenseMap< const BasicBlock *, bool > BBSideEffectsSet
Implement std::hash so that hash_code can be used in STL containers.
bool operator!=(const CHIArg &A) const
bool operator==(const CHIArg &A) const
LLVM_ABI PreservedAnalyses run(Function &F, FunctionAnalysisManager &AM)
Run the pass over the function.