94#define DEBUG_TYPE "licm"
96STATISTIC(NumCreatedBlocks,
"Number of blocks created");
97STATISTIC(NumClonedBranches,
"Number of branches cloned");
98STATISTIC(NumSunk,
"Number of instructions sunk out of loop");
99STATISTIC(NumHoisted,
"Number of instructions hoisted out of loop");
100STATISTIC(NumMovedLoads,
"Number of load insts hoisted or sunk");
101STATISTIC(NumMovedCalls,
"Number of call insts hoisted or sunk");
102STATISTIC(NumPromotionCandidates,
"Number of promotion candidates");
103STATISTIC(NumLoadPromoted,
"Number of load-only promotions");
104STATISTIC(NumLoadStorePromoted,
"Number of load and store promotions");
106 "Number of min/max expressions hoisted out of the loop");
108 "Number of geps reassociated and hoisted out of the loop");
109STATISTIC(NumAddSubHoisted,
"Number of add/subtract expressions reassociated "
110 "and hoisted out of the loop");
111STATISTIC(NumFPAssociationsHoisted,
"Number of invariant FP expressions "
112 "reassociated and hoisted out of the loop");
114 "Number of invariant int expressions "
115 "reassociated and hoisted out of the loop");
116STATISTIC(NumBOAssociationsHoisted,
"Number of invariant BinaryOp expressions "
117 "reassociated and hoisted out of the loop");
122 cl::desc(
"Disable memory promotion in LICM pass"));
126 cl::desc(
"Enable control flow (and PHI) hoisting in LICM"));
130 cl::desc(
"Force thread model single in LICM pass"));
134 cl::desc(
"Max num uses visited for identifying load "
135 "invariance in loop using invariant start (default = 8)"));
140 "Set upper limit for the number of transformations performed "
141 "during a single round of hoisting the reassociated expressions."));
146 "Set upper limit for the number of transformations performed "
147 "during a single round of hoisting the reassociated expressions."));
159 cl::desc(
"Enable imprecision in LICM in pathological cases, in exchange "
160 "for faster compile. Caps the MemorySSA clobbering calls."));
167 cl::desc(
"[LICM & MemorySSA] When MSSA in LICM is disabled, this has no "
168 "effect. When MSSA in LICM is enabled, then this is the maximum "
169 "number of accesses allowed to be present in a loop in order to "
170 "enable memory promotion."));
176 bool &FoldableInLoop,
bool LoopNestMode);
192 bool InvariantGroup);
214 std::pair<SmallSetVector<Value *, 8>,
bool>;
219struct LoopInvariantCodeMotion {
225 LoopInvariantCodeMotion(
unsigned LicmMssaOptCap,
226 unsigned LicmMssaNoAccForPromotionCap,
227 bool LicmAllowSpeculation)
228 : LicmMssaOptCap(LicmMssaOptCap),
229 LicmMssaNoAccForPromotionCap(LicmMssaNoAccForPromotionCap),
230 LicmAllowSpeculation(LicmAllowSpeculation) {}
233 unsigned LicmMssaOptCap;
234 unsigned LicmMssaNoAccForPromotionCap;
235 bool LicmAllowSpeculation;
238struct LegacyLICMPass :
public LoopPass {
243 bool LicmAllowSpeculation =
true)
244 :
LoopPass(
ID), LICM(LicmMssaOptCap, LicmMssaNoAccForPromotionCap,
245 LicmAllowSpeculation) {
254 <<
L->getHeader()->getNameOrAsOperand() <<
"\n");
258 auto *SE = getAnalysisIfAvailable<ScalarEvolutionWrapperPass>();
259 MemorySSA *MSSA = &getAnalysis<MemorySSAWrapperPass>().getMSSA();
264 return LICM.runOnLoop(
265 L, &getAnalysis<AAResultsWrapperPass>().getAAResults(),
266 &getAnalysis<LoopInfoWrapperPass>().getLoopInfo(),
267 &getAnalysis<DominatorTreeWrapperPass>().getDomTree(),
268 &getAnalysis<AssumptionCacheTracker>().getAssumptionCache(*
F),
269 &getAnalysis<TargetLibraryInfoWrapperPass>().getTLI(*
F),
270 &getAnalysis<TargetTransformInfoWrapperPass>().getTTI(*
F),
271 SE ? &SE->getSE() :
nullptr, MSSA, &ORE);
292 LoopInvariantCodeMotion LICM;
309 if (!LICM.runOnLoop(&L, &AR.
AA, &AR.
LI, &AR.
DT, &AR.
AC, &AR.
TLI, &AR.
TTI,
322 OS, MapClassName2PassName);
345 bool Changed = LICM.runOnLoop(&OutermostLoop, &AR.
AA, &AR.
LI, &AR.
DT, &AR.
AC,
363 OS, MapClassName2PassName);
370char LegacyLICMPass::ID = 0;
389 unsigned LicmMssaOptCap,
unsigned LicmMssaNoAccForPromotionCap,
bool IsSink,
391 : LicmMssaOptCap(LicmMssaOptCap),
392 LicmMssaNoAccForPromotionCap(LicmMssaNoAccForPromotionCap),
394 unsigned AccessCapCount = 0;
395 for (
auto *BB : L.getBlocks())
397 for (
const auto &MA : *Accesses) {
417 bool Changed =
false;
419 assert(L->isLCSSAForm(*DT) &&
"Loop is not in LCSSA form.");
437 return llvm::any_of(*BB, [](Instruction &I) {
438 IntrinsicInst *II = dyn_cast<IntrinsicInst>(&I);
439 return II && II->getIntrinsicID() == Intrinsic::coro_suspend;
463 if (
L->hasDedicatedExits())
467 TLI,
TTI, L, MSSAU, &SafetyInfo, Flags, ORE)
469 MSSAU, &SafetyInfo,
Flags, ORE);
470 Flags.setIsSink(
false);
473 MSSAU, SE, &SafetyInfo, Flags, ORE, LoopNestMode,
474 LicmAllowSpeculation);
484 !
Flags.tooManyMemoryAccesses() && !HasCoroSuspendInst) {
487 L->getUniqueExitBlocks(ExitBlocks);
491 return isa<CatchSwitchInst>(
Exit->getTerminator());
494 if (!HasCatchSwitch) {
500 InsertPts.
push_back(ExitBlock->getFirstInsertionPt());
508 bool Promoted =
false;
511 LocalPromoted =
false;
512 for (
auto [PointerMustAliases, HasReadsOutsideSet] :
515 PointerMustAliases, ExitBlocks, InsertPts, MSSAInsertPts,
PIC, LI,
516 DT, AC, TLI,
TTI, L, MSSAU, &SafetyInfo, ORE,
517 LicmAllowSpeculation, HasReadsOutsideSet);
519 Promoted |= LocalPromoted;
520 }
while (LocalPromoted);
538 assert(
L->isLCSSAForm(*DT) &&
"Loop not left in LCSSA form after LICM!");
539 assert((
L->isOutermost() ||
L->getParentLoop()->isLCSSAForm(*DT)) &&
540 "Parent loop not left in LCSSA form after LICM!");
563 assert(
N !=
nullptr && AA !=
nullptr && LI !=
nullptr && DT !=
nullptr &&
564 CurLoop !=
nullptr && SafetyInfo !=
nullptr &&
565 "Unexpected input to sinkRegion.");
572 bool Changed =
false;
600 bool FoldableInLoop =
false;
601 bool LoopNestMode = OutermostLoop !=
nullptr;
602 if (!
I.mayHaveSideEffects() &&
604 SafetyInfo,
TTI, FoldableInLoop,
607 if (
sink(
I, LI, DT, CurLoop, SafetyInfo, MSSAU, ORE)) {
608 if (!FoldableInLoop) {
631 bool Changed =
false;
635 while (!Worklist.
empty()) {
638 MSSAU, SafetyInfo, Flags, ORE, CurLoop);
651class ControlFlowHoister {
670 : LI(LI), DT(DT), CurLoop(CurLoop), MSSAU(MSSAU) {}
672 void registerPossiblyHoistableBranch(
BranchInst *BI) {
684 TrueDest == FalseDest)
696 if (TrueDestSucc.
count(FalseDest)) {
697 CommonSucc = FalseDest;
698 }
else if (FalseDestSucc.
count(TrueDest)) {
699 CommonSucc = TrueDest;
703 if (TrueDestSucc.
size() == 1)
704 CommonSucc = *TrueDestSucc.
begin();
708 else if (!TrueDestSucc.
empty()) {
712 assert(It !=
F->end() &&
"Could not find successor in function");
724 if (CommonSucc && DT->
dominates(BI, CommonSucc))
725 HoistableBranches[BI] = CommonSucc;
728 bool canHoistPHI(
PHINode *PN) {
737 PredecessorBlocks.
insert(PredBB);
745 for (
auto &Pair : HoistableBranches) {
746 if (Pair.second == BB) {
749 if (Pair.first->getSuccessor(0) == BB) {
750 PredecessorBlocks.
erase(Pair.first->getParent());
751 PredecessorBlocks.
erase(Pair.first->getSuccessor(1));
752 }
else if (Pair.first->getSuccessor(1) == BB) {
753 PredecessorBlocks.
erase(Pair.first->getParent());
754 PredecessorBlocks.
erase(Pair.first->getSuccessor(0));
756 PredecessorBlocks.
erase(Pair.first->getSuccessor(0));
757 PredecessorBlocks.
erase(Pair.first->getSuccessor(1));
763 return PredecessorBlocks.
empty();
770 if (HoistDestinationMap.
count(BB))
771 return HoistDestinationMap[BB];
774 auto HasBBAsSuccessor =
776 return BB != Pair.second && (Pair.first->getSuccessor(0) == BB ||
777 Pair.first->getSuccessor(1) == BB);
779 auto It =
llvm::find_if(HoistableBranches, HasBBAsSuccessor);
783 if (It == HoistableBranches.end()) {
786 <<
" as hoist destination for "
788 HoistDestinationMap[BB] = InitialPreheader;
789 return InitialPreheader;
792 assert(std::find_if(++It, HoistableBranches.end(), HasBBAsSuccessor) ==
793 HoistableBranches.end() &&
794 "BB is expected to be the target of at most one branch");
799 BasicBlock *CommonSucc = HoistableBranches[BI];
803 auto CreateHoistedBlock = [&](
BasicBlock *Orig) {
804 if (HoistDestinationMap.
count(Orig))
805 return HoistDestinationMap[Orig];
808 HoistDestinationMap[Orig] =
New;
814 <<
" as hoist destination for " << Orig->getName()
818 BasicBlock *HoistTrueDest = CreateHoistedBlock(TrueDest);
819 BasicBlock *HoistFalseDest = CreateHoistedBlock(FalseDest);
820 BasicBlock *HoistCommonSucc = CreateHoistedBlock(CommonSucc);
827 assert(TargetSucc &&
"Expected hoist target to have a single successor");
842 if (HoistTarget == InitialPreheader) {
853 for (
auto &Pair : HoistDestinationMap)
854 if (Pair.second == InitialPreheader && Pair.first != BI->
getParent())
855 Pair.second = HoistCommonSucc;
865 "Hoisting blocks should not have destroyed preheader");
866 return HoistDestinationMap[BB];
883 bool AllowSpeculation) {
885 assert(
N !=
nullptr && AA !=
nullptr && LI !=
nullptr && DT !=
nullptr &&
886 CurLoop !=
nullptr && SafetyInfo !=
nullptr &&
887 "Unexpected input to hoistRegion.");
889 ControlFlowHoister CFH(LI, DT, CurLoop, MSSAU);
900 bool Changed =
false;
905 if (!LoopNestMode &&
inSubLoop(BB, CurLoop, LI))
919 I, DT, TLI, CurLoop, SafetyInfo, ORE,
921 hoist(
I, DT, CurLoop, CFH.getOrCreateHoistedBlock(BB), SafetyInfo,
930 if (
I.getOpcode() == Instruction::FDiv &&
I.hasAllowReciprocal() &&
932 auto Divisor =
I.getOperand(1);
933 auto One = llvm::ConstantFP::get(Divisor->getType(), 1.0);
934 auto ReciprocalDivisor = BinaryOperator::CreateFDiv(One, Divisor);
935 ReciprocalDivisor->setFastMathFlags(
I.getFastMathFlags());
937 ReciprocalDivisor->insertBefore(&
I);
938 ReciprocalDivisor->setDebugLoc(
I.getDebugLoc());
941 BinaryOperator::CreateFMul(
I.getOperand(0), ReciprocalDivisor);
942 Product->setFastMathFlags(
I.getFastMathFlags());
944 Product->insertAfter(&
I);
945 Product->setDebugLoc(
I.getDebugLoc());
946 I.replaceAllUsesWith(Product);
949 hoist(*ReciprocalDivisor, DT, CurLoop, CFH.getOrCreateHoistedBlock(BB),
950 SafetyInfo, MSSAU, SE, ORE);
951 HoistedInstructions.
push_back(ReciprocalDivisor);
957 using namespace PatternMatch;
958 return I.use_empty() &&
959 match(&
I, m_Intrinsic<Intrinsic::invariant_start>());
961 auto MustExecuteWithoutWritesBefore = [&](
Instruction &
I) {
965 if ((IsInvariantStart(
I) ||
isGuard(&
I)) &&
967 MustExecuteWithoutWritesBefore(
I)) {
968 hoist(
I, DT, CurLoop, CFH.getOrCreateHoistedBlock(BB), SafetyInfo,
975 if (
PHINode *PN = dyn_cast<PHINode>(&
I)) {
976 if (CFH.canHoistPHI(PN)) {
982 hoist(*PN, DT, CurLoop, CFH.getOrCreateHoistedBlock(BB), SafetyInfo,
1000 CFH.registerPossiblyHoistableBranch(BI);
1015 [&](
Use &U) { return DT->dominates(I, U); })) {
1021 "New hoist point expected to dominate old hoist point");
1025 << HoistPoint->
getParent()->getNameOrAsOperand()
1026 <<
": " << *
I <<
"\n");
1039#ifdef EXPENSIVE_CHECKS
1041 assert(DT->
verify(DominatorTree::VerificationLevel::Fast) &&
1042 "Dominator tree verification failed");
1074 if (isa<Constant>(
Addr))
1077 unsigned UsesVisited = 0;
1080 for (
auto *U :
Addr->users()) {
1087 if (!
II ||
II->getIntrinsicID() != Intrinsic::invariant_start ||
1090 ConstantInt *InvariantSize = cast<ConstantInt>(
II->getArgOperand(0));
1114 return (isa<LoadInst>(
I) || isa<StoreInst>(
I) || isa<CallInst>(
I) ||
1115 isa<FenceInst>(
I) || isa<CastInst>(
I) || isa<UnaryOperator>(
I) ||
1116 isa<BinaryOperator>(
I) || isa<SelectInst>(
I) ||
1117 isa<GetElementPtrInst>(
I) || isa<CmpInst>(
I) ||
1118 isa<InsertElementInst>(
I) || isa<ExtractElementInst>(
I) ||
1119 isa<ShuffleVectorInst>(
I) || isa<ExtractValueInst>(
I) ||
1120 isa<InsertValueInst>(
I) || isa<FreezeInst>(
I));
1124 for (
auto *BB :
L->getBlocks())
1133 for (
auto *BB :
L->getBlocks())
1136 for (
const auto &Acc : *Accs) {
1137 if (isa<MemoryPhi>(&Acc))
1139 const auto *MUD = cast<MemoryUseOrDef>(&Acc);
1140 if (MUD->getMemoryInst() !=
I || NotAPhi++ == 1)
1153 if (Flags.tooManyClobberingCalls())
1158 Flags.incrementClobberingCalls();
1164 bool TargetExecutesOncePerLoop,
1168 if (!isHoistableAndSinkableInst(
I))
1173 if (
LoadInst *LI = dyn_cast<LoadInst>(&
I)) {
1174 if (!LI->isUnordered())
1181 if (LI->hasMetadata(LLVMContext::MD_invariant_load))
1184 if (LI->isAtomic() && !TargetExecutesOncePerLoop)
1193 bool InvariantGroup = LI->hasMetadata(LLVMContext::MD_invariant_group);
1196 MSSA, MU, CurLoop,
I, Flags, InvariantGroup);
1202 DEBUG_TYPE,
"LoadWithLoopInvariantAddressInvalidated", LI)
1203 <<
"failed to move load with loop-invariant address "
1204 "because the loop may invalidate its value";
1208 }
else if (
CallInst *CI = dyn_cast<CallInst>(&
I)) {
1210 if (isa<DbgInfoIntrinsic>(
I))
1221 if (CI->isConvergent())
1229 if (CI->getFunction()->isPresplitCoroutine())
1232 using namespace PatternMatch;
1233 if (
match(CI, m_Intrinsic<Intrinsic::assume>()))
1249 if (
Op->getType()->isPointerTy() &&
1259 if (isReadOnly(MSSAU, CurLoop))
1267 }
else if (
auto *FI = dyn_cast<FenceInst>(&
I)) {
1270 return isOnlyMemoryAccess(FI, CurLoop, MSSAU);
1271 }
else if (
auto *SI = dyn_cast<StoreInst>(&
I)) {
1272 if (!SI->isUnordered())
1280 if (isOnlyMemoryAccess(SI, CurLoop, MSSAU))
1284 if (Flags.tooManyMemoryAccesses())
1292 CurLoop->
contains(Source->getBlock()))
1302 for (
const auto &MA : *Accesses)
1303 if (
const auto *MU = dyn_cast<MemoryUse>(&MA)) {
1313 if (!Flags.getIsSink() && !MSSA->
dominates(SIMD, MU))
1315 }
else if (
const auto *MD = dyn_cast<MemoryDef>(&MA)) {
1316 if (
auto *LI = dyn_cast<LoadInst>(MD->getMemoryInst())) {
1318 assert(!LI->isUnordered() &&
"Expected unordered load");
1322 if (
auto *CI = dyn_cast<CallInst>(MD->getMemoryInst())) {
1335 assert(!
I.mayReadOrWriteMemory() &&
"unhandled aliasing");
1358 if (
auto *
GEP = dyn_cast<GetElementPtrInst>(&
I)) {
1367 for (
const User *U :
GEP->users()) {
1371 (!isa<StoreInst>(UI) && !isa<LoadInst>(UI))))
1389 bool &FoldableInLoop,
bool LoopNestMode) {
1392 for (
const User *U :
I.users()) {
1394 if (
const PHINode *PN = dyn_cast<PHINode>(UI)) {
1402 if (isa<CallInst>(
I))
1403 if (!BlockColors.empty() &&
1404 BlockColors.find(
const_cast<BasicBlock *
>(BB))->second.size() != 1)
1408 while (isa<PHINode>(UI) && UI->
hasOneUser() &&
1412 UI = cast<Instruction>(UI->
user_back());
1419 FoldableInLoop =
true;
1432 if (
auto *CI = dyn_cast<CallInst>(&
I)) {
1439 for (
unsigned BundleIdx = 0, BundleEnd = CI->getNumOperandBundles();
1440 BundleIdx != BundleEnd; ++BundleIdx) {
1448 if (!BlockColors.empty()) {
1449 const ColorVector &CV = BlockColors.find(&ExitBlock)->second;
1450 assert(CV.
size() == 1 &&
"non-unique color for exit block!");
1458 New->copyMetadata(*CI);
1464 if (!
I.getName().empty())
1465 New->setName(
I.getName() +
".le");
1472 if (
auto *MemDef = dyn_cast<MemoryDef>(NewMemAcc))
1475 auto *MemUse = cast<MemoryUse>(NewMemAcc);
1488 for (
Use &
Op : New->operands())
1490 auto *OInst = cast<Instruction>(
Op.get());
1493 OInst->getName() +
".lcssa");
1506 I.eraseFromParent();
1515 I.moveBefore(*Dest->getParent(), Dest);
1530 "Expect only trivially replaceable PHI");
1533 auto It = SunkCopies.
find(ExitBlock);
1534 if (It != SunkCopies.
end())
1538 *
I, *ExitBlock, *TPN, LI, SafetyInfo, MSSAU);
1553 if (isa<IndirectBrInst>(BBPred->getTerminator()))
1570 assert(ExitBlockSet.
count(ExitBB) &&
"Expect the PHI is in an exit block.");
1606 while (!PredBBs.
empty()) {
1609 "Expect all predecessors are in the loop");
1612 ExitBB, PredBB,
".split.loop.exit", DT, LI, MSSAU,
true);
1616 if (!BlockColors.empty())
1634 bool Changed =
false;
1641 auto *
User = cast<Instruction>(*UI);
1642 Use &U = UI.getUse();
1680 UI =
I.user_begin();
1684 if (VisitedUsers.
empty())
1689 <<
"sinking " <<
ore::NV(
"Inst", &
I);
1691 if (isa<LoadInst>(
I))
1693 else if (isa<CallInst>(
I))
1713 for (
auto *UI :
Users) {
1714 auto *
User = cast<Instruction>(UI);
1721 "The LCSSA PHI is not in an exit block!");
1725 PN, &
I, LI, SunkCopies, SafetyInfo, CurLoop, MSSAU);
1727 New->dropLocation();
1757 if ((
I.hasMetadataOtherThanDebugLoc() || isa<CallInst>(
I)) &&
1762 I.dropUBImplyingAttrsAndMetadata();
1764 if (isa<PHINode>(
I))
1772 I.updateLocationAfterHoist();
1774 if (isa<LoadInst>(
I))
1776 else if (isa<CallInst>(
I))
1789 if (AllowSpeculation &&
1793 bool GuaranteedToExecute =
1796 if (!GuaranteedToExecute) {
1797 auto *LI = dyn_cast<LoadInst>(&Inst);
1801 DEBUG_TYPE,
"LoadWithLoopInvariantAddressCondExecuted", LI)
1802 <<
"failed to hoist load with loop-invariant address "
1803 "because load is conditionally executed";
1807 return GuaranteedToExecute;
1821 bool UnorderedAtomic;
1824 bool CanInsertStoresInExitBlocks;
1838 I->getName() +
".lcssa");
1854 LoopInsertPts(LIP), MSSAInsertPts(MSSAIP), PredCache(
PIC), MSSAU(MSSAU),
1855 LI(li),
DL(
std::
move(dl)), Alignment(Alignment),
1856 UnorderedAtomic(UnorderedAtomic), AATags(AATags),
1857 SafetyInfo(SafetyInfo),
1858 CanInsertStoresInExitBlocks(CanInsertStoresInExitBlocks),
Uses(Insts) {}
1860 void insertStoresInLoopExitBlocks() {
1866 for (
unsigned i = 0, e = LoopExitBlocks.
size(); i != e; ++i) {
1868 Value *LiveInValue =
SSA.GetValueInMiddleOfBlock(ExitBlock);
1869 LiveInValue = maybeInsertLCSSAPHI(LiveInValue, ExitBlock);
1870 Value *
Ptr = maybeInsertLCSSAPHI(SomePtr, ExitBlock);
1873 if (UnorderedAtomic)
1884 NewID = cast_or_null<DIAssignID>(
1889 NewSI->
setMetadata(LLVMContext::MD_DIAssignID, NewID);
1897 if (!MSSAInsertPoint) {
1904 MSSAInsertPts[i] = NewMemAcc;
1905 MSSAU.
insertDef(cast<MemoryDef>(NewMemAcc),
true);
1910 void doExtraRewritesBeforeFinalDeletion()
override {
1911 if (CanInsertStoresInExitBlocks)
1912 insertStoresInLoopExitBlocks();
1915 void instructionDeleted(
Instruction *
I)
const override {
1921 if (isa<StoreInst>(
I))
1922 return CanInsertStoresInExitBlocks;
1927bool isNotCapturedBeforeOrInLoop(
const Value *V,
const Loop *L,
1935 L->getHeader()->getTerminator(), DT);
1940bool isNotVisibleOnUnwindInLoop(
const Value *Object,
const Loop *L,
1942 bool RequiresNoCaptureBeforeUnwind;
1946 return !RequiresNoCaptureBeforeUnwind ||
1947 isNotCapturedBeforeOrInLoop(Object, L, DT);
1955 isNotCapturedBeforeOrInLoop(Object, L, DT)) ||
1975 bool HasReadsOutsideSet) {
1977 assert(LI !=
nullptr && DT !=
nullptr && CurLoop !=
nullptr &&
1978 SafetyInfo !=
nullptr &&
1979 "Unexpected Input to promoteLoopAccessesToScalars");
1982 dbgs() <<
"Trying to promote set of must-aliased pointers:\n";
1983 for (
Value *
Ptr : PointerMustAliases)
1984 dbgs() <<
" " << *
Ptr <<
"\n";
1986 ++NumPromotionCandidates;
1988 Value *SomePtr = *PointerMustAliases.
begin();
2028 bool DereferenceableInPH =
false;
2029 bool StoreIsGuanteedToExecute =
false;
2030 bool FoundLoadToPromote =
false;
2036 } StoreSafety = StoreSafetyUnknown;
2044 bool SawUnorderedAtomic =
false;
2045 bool SawNotAtomic =
false;
2052 if (HasReadsOutsideSet)
2053 StoreSafety = StoreUnsafe;
2062 if (!isNotVisibleOnUnwindInLoop(Object, CurLoop, DT))
2063 StoreSafety = StoreUnsafe;
2069 Type *AccessTy =
nullptr;
2070 for (
Value *ASIV : PointerMustAliases) {
2071 for (
Use &U : ASIV->uses()) {
2073 Instruction *UI = dyn_cast<Instruction>(U.getUser());
2079 if (
LoadInst *Load = dyn_cast<LoadInst>(UI)) {
2080 if (!Load->isUnordered())
2083 SawUnorderedAtomic |= Load->isAtomic();
2084 SawNotAtomic |= !Load->isAtomic();
2085 FoundLoadToPromote =
true;
2087 Align InstAlignment = Load->getAlign();
2093 if (!DereferenceableInPH || (InstAlignment > Alignment))
2095 *Load, DT, TLI, CurLoop, SafetyInfo, ORE,
2097 DereferenceableInPH =
true;
2098 Alignment = std::max(Alignment, InstAlignment);
2100 }
else if (
const StoreInst *Store = dyn_cast<StoreInst>(UI)) {
2105 if (!Store->isUnordered())
2108 SawUnorderedAtomic |= Store->isAtomic();
2109 SawNotAtomic |= !Store->isAtomic();
2116 Align InstAlignment = Store->getAlign();
2117 bool GuaranteedToExecute =
2119 StoreIsGuanteedToExecute |= GuaranteedToExecute;
2120 if (GuaranteedToExecute) {
2121 DereferenceableInPH =
true;
2122 if (StoreSafety == StoreSafetyUnknown)
2123 StoreSafety = StoreSafe;
2124 Alignment = std::max(Alignment, InstAlignment);
2133 if (StoreSafety == StoreSafetyUnknown &&
2135 return DT->
dominates(Store->getParent(), Exit);
2137 StoreSafety = StoreSafe;
2141 if (!DereferenceableInPH) {
2143 Store->getPointerOperand(), Store->getValueOperand()->getType(),
2144 Store->getAlign(), MDL, Preheader->
getTerminator(), AC, DT, TLI);
2155 if (LoopUses.
empty()) {
2158 }
else if (AATags) {
2170 if (SawUnorderedAtomic && SawNotAtomic)
2180 if (!DereferenceableInPH) {
2181 LLVM_DEBUG(
dbgs() <<
"Not promoting: Not dereferenceable in preheader\n");
2189 if (StoreSafety == StoreSafetyUnknown) {
2191 bool ExplicitlyDereferenceableOnly;
2193 (!ExplicitlyDereferenceableOnly ||
2195 isThreadLocalObject(Object, CurLoop, DT,
TTI))
2196 StoreSafety = StoreSafe;
2201 if (StoreSafety != StoreSafe && !FoundLoadToPromote)
2206 if (StoreSafety == StoreSafe) {
2207 LLVM_DEBUG(
dbgs() <<
"LICM: Promoting load/store of the value: " << *SomePtr
2209 ++NumLoadStorePromoted;
2211 LLVM_DEBUG(
dbgs() <<
"LICM: Promoting load of the value: " << *SomePtr
2219 <<
"Moving accesses to memory location out of the loop";
2223 std::vector<DILocation *> LoopUsesLocs;
2224 for (
auto *U : LoopUses)
2225 LoopUsesLocs.push_back(U->getDebugLoc().get());
2231 LoopPromoter Promoter(SomePtr, LoopUses,
SSA, ExitBlocks, InsertPts,
2232 MSSAInsertPts,
PIC, MSSAU, *LI,
DL, Alignment,
2233 SawUnorderedAtomic, AATags, *SafetyInfo,
2234 StoreSafety == StoreSafe);
2239 if (FoundLoadToPromote || !StoreIsGuanteedToExecute) {
2243 if (SawUnorderedAtomic)
2252 MemoryUse *NewMemUse = cast<MemoryUse>(PreheaderLoadMemoryAccess);
2254 SSA.AddAvailableValue(Preheader, PreheaderLoad);
2263 Promoter.run(LoopUses);
2268 if (PreheaderLoad && PreheaderLoad->
use_empty())
2278 for (
const auto &Access : *Accesses)
2279 if (
const auto *MUD = dyn_cast<MemoryUseOrDef>(&Access))
2280 Fn(MUD->getMemoryInst());
2290 auto IsPotentiallyPromotable = [L](
const Instruction *
I) {
2291 if (
const auto *SI = dyn_cast<StoreInst>(
I))
2292 return L->isLoopInvariant(SI->getPointerOperand());
2293 if (
const auto *LI = dyn_cast<LoadInst>(
I))
2294 return L->isLoopInvariant(LI->getPointerOperand());
2301 if (IsPotentiallyPromotable(
I)) {
2302 AttemptingPromotion.
insert(
I);
2310 if (!AS.isForwardingAliasSet() && AS.isMod() && AS.isMustAlias())
2322 ModRefInfo MR = Pair.getPointer()->aliasesUnknownInst(I, BatchAA);
2331 return !Pair.getPointer()->isRef();
2338 for (
auto [Set, HasReadsOutsideSet] : Sets) {
2340 for (
const auto &MemLoc : *Set)
2341 PointerMustAliases.
insert(
const_cast<Value *
>(MemLoc.Ptr));
2342 Result.emplace_back(std::move(PointerMustAliases), HasReadsOutsideSet);
2351 bool InvariantGroup) {
2353 if (!Flags.getIsSink()) {
2366 CurLoop->
contains(Source->getBlock()) &&
2367 !(InvariantGroup && Source->getBlock() == CurLoop->
getHeader() && isa<MemoryPhi>(Source));
2387 if (Flags.tooManyMemoryAccesses())
2401 for (
const auto &MA : *Accesses)
2402 if (
const auto *MD = dyn_cast<MemoryDef>(&MA))
2414 using namespace PatternMatch;
2415 Value *Cond1, *Cond2;
2431 if (L.isLoopInvariant(
LHS)) {
2435 if (L.isLoopInvariant(
LHS) || !L.isLoopInvariant(
RHS))
2442 Value *LHS1, *LHS2, *RHS1, *RHS2;
2443 if (!MatchICmpAgainstInvariant(Cond1, P1, LHS1, RHS1) ||
2444 !MatchICmpAgainstInvariant(Cond2, P2, LHS2, RHS2))
2446 if (P1 != P2 || LHS1 != LHS2)
2453 "Relational predicate is either less (or equal) or greater (or equal)!");
2455 ? (UseMin ? Intrinsic::smin : Intrinsic::smax)
2456 : (UseMin ? Intrinsic::umin : Intrinsic::umax);
2457 auto *Preheader = L.getLoopPreheader();
2458 assert(Preheader &&
"Loop is not in simplify form?");
2464 if (isa<SelectInst>(
I))
2467 id, RHS1, RHS2,
nullptr,
StringRef(
"invariant.") +
2469 (UseMin ?
"min" :
"max"));
2476 I.replaceAllUsesWith(NewCond);
2488 auto *
GEP = dyn_cast<GetElementPtrInst>(&
I);
2492 auto *Src = dyn_cast<GetElementPtrInst>(
GEP->getPointerOperand());
2493 if (!Src || !Src->hasOneUse() || !L.contains(Src))
2496 Value *SrcPtr = Src->getPointerOperand();
2497 auto LoopInvariant = [&](
Value *V) {
return L.isLoopInvariant(V); };
2498 if (!L.isLoopInvariant(SrcPtr) || !
all_of(
GEP->indices(), LoopInvariant))
2505 if (
all_of(Src->indices(), LoopInvariant))
2515 bool IsInBounds = Src->isInBounds() &&
GEP->isInBounds() &&
2519 BasicBlock *Preheader = L.getLoopPreheader();
2523 "invariant.gep", IsInBounds);
2525 Value *NewGEP = Builder.
CreateGEP(Src->getSourceElementType(), NewSrc,
2528 GEP->replaceAllUsesWith(NewGEP);
2541 assert(!L.isLoopInvariant(VariantLHS) &&
"Precondition.");
2542 assert(L.isLoopInvariant(InvariantRHS) &&
"Precondition.");
2545 using namespace PatternMatch;
2546 Value *VariantOp, *InvariantOp;
2552 if (L.isLoopInvariant(VariantOp))
2554 if (L.isLoopInvariant(VariantOp) || !L.isLoopInvariant(InvariantOp))
2561 auto &
DL = L.getHeader()->getDataLayout();
2562 bool ProvedNoOverflowAfterReassociate =
2566 if (!ProvedNoOverflowAfterReassociate)
2568 auto *Preheader = L.getLoopPreheader();
2569 assert(Preheader &&
"Loop is not in simplify form?");
2571 Value *NewCmpOp = Builder.
CreateSub(InvariantRHS, InvariantOp,
"invariant.op",
2588 assert(!L.isLoopInvariant(VariantLHS) &&
"Precondition.");
2589 assert(L.isLoopInvariant(InvariantRHS) &&
"Precondition.");
2592 using namespace PatternMatch;
2593 Value *VariantOp, *InvariantOp;
2597 bool VariantSubtracted =
false;
2601 if (L.isLoopInvariant(VariantOp)) {
2603 VariantSubtracted =
true;
2606 if (L.isLoopInvariant(VariantOp) || !L.isLoopInvariant(InvariantOp))
2614 auto &
DL = L.getHeader()->getDataLayout();
2616 if (VariantSubtracted) {
2627 auto *Preheader = L.getLoopPreheader();
2628 assert(Preheader &&
"Loop is not in simplify form?");
2632 ? Builder.
CreateSub(InvariantOp, InvariantRHS,
"invariant.op",
2634 : Builder.
CreateAdd(InvariantOp, InvariantRHS,
"invariant.op",
2647 using namespace PatternMatch;
2658 if (L.isLoopInvariant(
LHS)) {
2669 if (
hoistAdd(Pred,
LHS,
RHS, cast<ICmpInst>(
I), L, SafetyInfo, MSSAU, AC, DT))
2672 if (
hoistSub(Pred,
LHS,
RHS, cast<ICmpInst>(
I), L, SafetyInfo, MSSAU, AC, DT))
2679 unsigned FPOpcode) {
2680 if (
I->getOpcode() == IntOpcode)
2682 if (
I->getOpcode() == FPOpcode &&
I->hasAllowReassoc() &&
2683 I->hasNoSignedZeros())
2699 Value *VariantOp =
I.getOperand(0);
2700 Value *InvariantOp =
I.getOperand(1);
2701 if (L.isLoopInvariant(VariantOp))
2703 if (L.isLoopInvariant(VariantOp) || !L.isLoopInvariant(InvariantOp))
2705 Value *Factor = InvariantOp;
2711 if (
BinaryOperator *VariantBinOp = dyn_cast<BinaryOperator>(VariantOp))
2713 while (!Worklist.
empty()) {
2726 L.isLoopInvariant(BO))
2730 if (L.isLoopInvariant(U0))
2732 else if (L.isLoopInvariant(U1))
2736 unsigned Limit =
I.getType()->isIntOrIntVectorTy()
2739 if (Changes.
size() > Limit)
2742 if (Changes.
empty())
2746 if (
I.getType()->isIntOrIntVectorTy()) {
2747 for (
auto *
Add : Adds)
2748 Add->dropPoisonGeneratingFlags();
2752 auto *Preheader = L.getLoopPreheader();
2753 assert(Preheader &&
"Loop is not in simplify form?");
2755 for (
auto *U : Changes) {
2756 assert(L.isLoopInvariant(U->get()));
2757 auto *Ins = cast<BinaryOperator>(U->getUser());
2759 if (
I.getType()->isIntOrIntVectorTy()) {
2760 Mul = Builder.
CreateMul(U->get(), Factor,
"factor.op.mul");
2762 Ins->dropPoisonGeneratingFlags();
2767 unsigned OpIdx = U->getOperandNo();
2768 auto *
LHS = OpIdx == 0 ?
Mul : Ins->getOperand(0);
2769 auto *
RHS = OpIdx == 1 ?
Mul : Ins->getOperand(1);
2772 Ins->getName() +
".reass", Ins->getIterator());
2773 NewBO->copyIRFlags(Ins);
2774 if (VariantOp == Ins)
2776 Ins->replaceAllUsesWith(NewBO);
2780 I.replaceAllUsesWith(VariantOp);
2800 auto *BO = dyn_cast<BinaryOperator>(&
I);
2801 if (!BO || !BO->isAssociative())
2806 if (Opcode != Instruction::Add)
2809 auto *BO0 = dyn_cast<BinaryOperator>(BO->getOperand(0));
2810 if (!BO0 || BO0->getOpcode() != Opcode || !BO0->isAssociative() ||
2811 BO0->hasNUsesOrMore(3))
2815 Value *LV = BO0->getOperand(0);
2816 Value *C1 = BO0->getOperand(1);
2817 Value *C2 = BO->getOperand(1);
2819 if (L.isLoopInvariant(LV) || !L.isLoopInvariant(C1) || !L.isLoopInvariant(C2))
2822 auto *Preheader = L.getLoopPreheader();
2823 assert(Preheader &&
"Loop is not in simplify form?");
2828 Opcode, LV, Inv, BO->
getName() +
".reass", BO->getIterator());
2831 if (Opcode == Instruction::Add && BO->hasNoUnsignedWrap() &&
2832 BO0->hasNoUnsignedWrap()) {
2833 Inv->setHasNoUnsignedWrap(
true);
2834 NewBO->setHasNoUnsignedWrap(
true);
2837 BO->replaceAllUsesWith(NewBO);
2842 if (BO0->use_empty())
2862 if (
hoistGEP(
I, L, SafetyInfo, MSSAU, AC, DT)) {
2875 bool IsInt =
I.getType()->isIntOrIntVectorTy();
2879 ++NumIntAssociationsHoisted;
2881 ++NumFPAssociationsHoisted;
2887 ++NumBOAssociationsHoisted;
2898 assert(CurLoop->
contains(BB) &&
"Only valid if BB is IN the loop");
unsigned const MachineRegisterInfo * MRI
static msgpack::DocNode getNode(msgpack::DocNode DN, msgpack::Type Type, MCValue Val)
MachineBasicBlock MachineBasicBlock::iterator DebugLoc DL
This file contains the declarations for the subclasses of Constant, which represent the different fla...
Rewrite Partial Register Uses
This file provides various utilities for inspecting and working with the control flow graph in LLVM I...
iv Induction Variable Users
static bool isReassociableOp(Instruction *I, unsigned IntOpcode, unsigned FPOpcode)
static bool isNotUsedOrFoldableInLoop(const Instruction &I, const Loop *CurLoop, const LoopSafetyInfo *SafetyInfo, TargetTransformInfo *TTI, bool &FoldableInLoop, bool LoopNestMode)
Return true if the only users of this instruction are outside of the loop.
static bool hoistGEP(Instruction &I, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Reassociate gep (gep ptr, idx1), idx2 to gep (gep ptr, idx2), idx1 if this allows hoisting the inner ...
static cl::opt< bool > SingleThread("licm-force-thread-model-single", cl::Hidden, cl::init(false), cl::desc("Force thread model single in LICM pass"))
static void splitPredecessorsOfLoopExit(PHINode *PN, DominatorTree *DT, LoopInfo *LI, const Loop *CurLoop, LoopSafetyInfo *SafetyInfo, MemorySSAUpdater *MSSAU)
static cl::opt< unsigned > FPAssociationUpperLimit("licm-max-num-fp-reassociations", cl::init(5U), cl::Hidden, cl::desc("Set upper limit for the number of transformations performed " "during a single round of hoisting the reassociated expressions."))
static bool isFoldableInLoop(const Instruction &I, const Loop *CurLoop, const TargetTransformInfo *TTI)
Return true if the instruction is foldable in the loop.
static bool hoistMinMax(Instruction &I, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU)
Try to simplify things like (A < INV_1 AND icmp A < INV_2) into (A < min(INV_1, INV_2)),...
static void moveInstructionBefore(Instruction &I, BasicBlock::iterator Dest, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, ScalarEvolution *SE)
static Instruction * cloneInstructionInExitBlock(Instruction &I, BasicBlock &ExitBlock, PHINode &PN, const LoopInfo *LI, const LoopSafetyInfo *SafetyInfo, MemorySSAUpdater &MSSAU)
static cl::opt< bool > ControlFlowHoisting("licm-control-flow-hoisting", cl::Hidden, cl::init(false), cl::desc("Enable control flow (and PHI) hoisting in LICM"))
static bool pointerInvalidatedByLoop(MemorySSA *MSSA, MemoryUse *MU, Loop *CurLoop, Instruction &I, SinkAndHoistLICMFlags &Flags, bool InvariantGroup)
static bool hoistAdd(ICmpInst::Predicate Pred, Value *VariantLHS, Value *InvariantRHS, ICmpInst &ICmp, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Try to turn things like "LV + C1 < C2" into "LV < C2 - C1".
static MemoryAccess * getClobberingMemoryAccess(MemorySSA &MSSA, BatchAAResults &BAA, SinkAndHoistLICMFlags &Flags, MemoryUseOrDef *MA)
static SmallVector< PointersAndHasReadsOutsideSet, 0 > collectPromotionCandidates(MemorySSA *MSSA, AliasAnalysis *AA, Loop *L)
static void hoist(Instruction &I, const DominatorTree *DT, const Loop *CurLoop, BasicBlock *Dest, ICFLoopSafetyInfo *SafetyInfo, MemorySSAUpdater &MSSAU, ScalarEvolution *SE, OptimizationRemarkEmitter *ORE)
When an instruction is found to only use loop invariant operands that is safe to hoist,...
static bool canSplitPredecessors(PHINode *PN, LoopSafetyInfo *SafetyInfo)
static bool sink(Instruction &I, LoopInfo *LI, DominatorTree *DT, const Loop *CurLoop, ICFLoopSafetyInfo *SafetyInfo, MemorySSAUpdater &MSSAU, OptimizationRemarkEmitter *ORE)
When an instruction is found to only be used outside of the loop, this function moves it to the exit ...
static bool hoistAddSub(Instruction &I, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Reassociate and hoist add/sub expressions.
static bool hoistMulAddAssociation(Instruction &I, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Try to reassociate expressions like ((A1 * B1) + (A2 * B2) + ...) * C where A1, A2,...
static cl::opt< uint32_t > MaxNumUsesTraversed("licm-max-num-uses-traversed", cl::Hidden, cl::init(8), cl::desc("Max num uses visited for identifying load " "invariance in loop using invariant start (default = 8)"))
cl::opt< unsigned > IntAssociationUpperLimit("licm-max-num-int-reassociations", cl::init(5U), cl::Hidden, cl::desc("Set upper limit for the number of transformations performed " "during a single round of hoisting the reassociated expressions."))
static void foreachMemoryAccess(MemorySSA *MSSA, Loop *L, function_ref< void(Instruction *)> Fn)
static bool isLoadInvariantInLoop(LoadInst *LI, DominatorTree *DT, Loop *CurLoop)
static Instruction * sinkThroughTriviallyReplaceablePHI(PHINode *TPN, Instruction *I, LoopInfo *LI, SmallDenseMap< BasicBlock *, Instruction *, 32 > &SunkCopies, const LoopSafetyInfo *SafetyInfo, const Loop *CurLoop, MemorySSAUpdater &MSSAU)
static bool inSubLoop(BasicBlock *BB, Loop *CurLoop, LoopInfo *LI)
Little predicate that returns true if the specified basic block is in a subloop of the current one,...
static bool hoistSub(ICmpInst::Predicate Pred, Value *VariantLHS, Value *InvariantRHS, ICmpInst &ICmp, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Try to reassociate and hoist the following two patterns: LV - C1 < C2 --> LV < C1 + C2,...
static void eraseInstruction(Instruction &I, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU)
static bool isSafeToExecuteUnconditionally(Instruction &Inst, const DominatorTree *DT, const TargetLibraryInfo *TLI, const Loop *CurLoop, const LoopSafetyInfo *SafetyInfo, OptimizationRemarkEmitter *ORE, const Instruction *CtxI, AssumptionCache *AC, bool AllowSpeculation)
Only sink or hoist an instruction if it is not a trapping instruction, or if the instruction is known...
static bool hoistArithmetics(Instruction &I, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Aggregates various functions for hoisting computations out of loop.
static bool isTriviallyReplaceablePHI(const PHINode &PN, const Instruction &I)
Returns true if a PHINode is a trivially replaceable with an Instruction.
std::pair< SmallSetVector< Value *, 8 >, bool > PointersAndHasReadsOutsideSet
static cl::opt< bool > DisablePromotion("disable-licm-promotion", cl::Hidden, cl::init(false), cl::desc("Disable memory promotion in LICM pass"))
Memory promotion is enabled by default.
static bool hoistBOAssociation(Instruction &I, Loop &L, ICFLoopSafetyInfo &SafetyInfo, MemorySSAUpdater &MSSAU, AssumptionCache *AC, DominatorTree *DT)
Reassociate associative binary expressions of the form.
static bool pointerInvalidatedByBlock(BasicBlock &BB, MemorySSA &MSSA, MemoryUse &MU)
This file defines the interface for the loop nest analysis.
Machine Loop Invariant Code Motion
This file exposes an interface to building/using memory SSA to walk memory instructions using a use/d...
Contains a collection of routines for determining if a given instruction is guaranteed to execute if ...
uint64_t IntrinsicInst * II
PassInstrumentationCallbacks PIC
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
This file provides a priority worklist.
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file defines generic set operations that may be used on set's of different types,...
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
ModRefInfo getModRefInfoMask(const MemoryLocation &Loc, bool IgnoreLocals=false)
Returns a bitmask that should be unconditionally applied to the ModRef info of a memory location.
MemoryEffects getMemoryEffects(const CallBase *Call)
Return the behavior of the given call site.
void add(const MemoryLocation &Loc)
These methods are used to add different types of instructions to the alias sets.
A container for analyses that lazily runs them and caches their results.
Represent the analysis usage information of a pass.
AnalysisUsage & addRequired()
AnalysisUsage & addPreserved()
Add the specified Pass class to the set of analyses preserved by this pass.
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
An immutable pass that tracks lazily created AssumptionCache objects.
A cache of @llvm.assume calls within a function.
LLVM Basic Block Representation.
void replaceSuccessorsPhiUsesWith(BasicBlock *Old, BasicBlock *New)
Update all phi nodes in this basic block's successors to refer to basic block New instead of basic bl...
iterator begin()
Instruction iterator methods.
const_iterator getFirstInsertionPt() const
Returns an iterator to the first instruction in this block that is suitable for inserting a non-PHI i...
InstListType::const_iterator getFirstNonPHIIt() const
Iterator returning form of getFirstNonPHI.
const Instruction * getFirstNonPHI() const
Returns a pointer to the first instruction in this block that is not a PHINode instruction.
static BasicBlock * Create(LLVMContext &Context, const Twine &Name="", Function *Parent=nullptr, BasicBlock *InsertBefore=nullptr)
Creates a new BasicBlock.
const BasicBlock * getSingleSuccessor() const
Return the successor of this block if it has a single successor.
const Function * getParent() const
Return the enclosing method, or null if none.
const DataLayout & getDataLayout() const
Get the data layout of the module this basic block belongs to.
InstListType::iterator iterator
Instruction iterators...
LLVMContext & getContext() const
Get the context in which this basic block lives.
void moveBefore(BasicBlock *MovePos)
Unlink this basic block from its current function and insert it into the function that MovePos lives ...
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 canSplitPredecessors() const
This class is a wrapper over an AAResults, and it is intended to be used only when there are no IR ch...
ModRefInfo getModRefInfo(const Instruction *I, const std::optional< MemoryLocation > &OptLoc)
static BinaryOperator * Create(BinaryOps Op, Value *S1, Value *S2, const Twine &Name=Twine(), InsertPosition InsertBefore=nullptr)
Construct a binary instruction, given the opcode and the two operands.
Conditional or Unconditional Branch instruction.
bool isConditional() const
static BranchInst * Create(BasicBlock *IfTrue, InsertPosition InsertBefore=nullptr)
BasicBlock * getSuccessor(unsigned i) const
Value * getCondition() const
This class represents a function call, abstracting a target machine's calling convention.
static CallInst * Create(FunctionType *Ty, Value *F, const Twine &NameStr="", InsertPosition InsertBefore=nullptr)
void setPredicate(Predicate P)
Set the predicate for this instruction to the specified value.
Predicate
This enumeration lists the possible predicates for CmpInst subclasses.
Predicate getSwappedPredicate() const
For example, EQ->EQ, SLE->SGE, ULT->UGT, OEQ->OEQ, ULE->UGE, OLT->OGT, etc.
Predicate getInversePredicate() const
For example, EQ -> NE, UGT -> ULE, SLT -> SGE, OEQ -> UNE, UGT -> OLE, OLT -> UGE,...
This is the shared class of boolean and integer constants.
int64_t getSExtValue() const
Return the constant as a 64-bit integer value after it has been sign extended as appropriate for the ...
static DILocation * getMergedLocations(ArrayRef< DILocation * > Locs)
Try to combine the vector of locations passed as input in a single one.
This class represents an Operation in the Expression.
A parsed version of the target data layout string in and methods for querying it.
TypeSize getTypeStoreSize(Type *Ty) const
Returns the maximum number of bytes that may be overwritten by storing the specified type.
iterator find(const_arg_type_t< KeyT > Val)
size_type count(const_arg_type_t< KeyT > Val) const
Return 1 if the specified key is in the map, 0 otherwise.
DomTreeNodeBase * getIDom() const
Analysis pass which computes a DominatorTree.
bool verify(VerificationLevel VL=VerificationLevel::Full) const
verify - checks if the tree is correct.
void changeImmediateDominator(DomTreeNodeBase< NodeT > *N, DomTreeNodeBase< NodeT > *NewIDom)
changeImmediateDominator - This method is used to update the dominator tree information when a node's...
DomTreeNodeBase< NodeT > * addNewBlock(NodeT *BB, NodeT *DomBB)
Add a new node to the dominator tree information.
DomTreeNodeBase< NodeT > * getNode(const NodeT *BB) const
getNode - return the (Post)DominatorTree node for the specified basic block.
bool properlyDominates(const DomTreeNodeBase< NodeT > *A, const DomTreeNodeBase< NodeT > *B) const
properlyDominates - Returns true iff A dominates B and A != B.
Legacy analysis pass which computes a DominatorTree.
Concrete subclass of DominatorTreeBase that is used to compute a normal dominator tree.
bool isReachableFromEntry(const Use &U) const
Provide an overload for a Use.
bool dominates(const BasicBlock *BB, const Use &U) const
Return true if the (end of the) basic block BB dominates the use U.
This implementation of LoopSafetyInfo use ImplicitControlFlowTracking to give precise answers on "may...
bool doesNotWriteMemoryBefore(const BasicBlock *BB, const Loop *CurLoop) const
Returns true if we could not execute a memory-modifying instruction before we enter BB under assumpti...
void removeInstruction(const Instruction *Inst)
Inform safety info that we are planning to remove the instruction Inst from its block.
bool isGuaranteedToExecute(const Instruction &Inst, const DominatorTree *DT, const Loop *CurLoop) const override
Returns true if the instruction in a loop is guaranteed to execute at least once (under the assumptio...
bool anyBlockMayThrow() const override
Returns true iff any block of the loop for which this info is contains an instruction that may throw ...
void computeLoopSafetyInfo(const Loop *CurLoop) override
Computes safety information for a loop checks loop body & header for the possibility of may throw exc...
void insertInstructionTo(const Instruction *Inst, const BasicBlock *BB)
Inform the safety info that we are planning to insert a new instruction Inst into the basic block BB.
This instruction compares its operands according to the predicate given to the constructor.
static bool isGE(Predicate P)
Return true if the predicate is SGE or UGE.
static bool isLT(Predicate P)
Return true if the predicate is SLT or ULT.
static bool isGT(Predicate P)
Return true if the predicate is SGT or UGT.
bool isRelational() const
Return true if the predicate is relational (not EQ or NE).
static bool isLE(Predicate P)
Return true if the predicate is SLE or ULE.
Value * CreateBinaryIntrinsic(Intrinsic::ID ID, Value *LHS, Value *RHS, Instruction *FMFSource=nullptr, const Twine &Name="")
Create a call to intrinsic ID with 2 operands which is mangled on the first type.
Value * CreateFMulFMF(Value *L, Value *R, Instruction *FMFSource, const Twine &Name="")
Copy fast-math-flags from an instruction rather than using the builder's default FMF.
Value * CreateFreeze(Value *V, const Twine &Name="")
Value * CreateGEP(Type *Ty, Value *Ptr, ArrayRef< Value * > IdxList, const Twine &Name="", GEPNoWrapFlags NW=GEPNoWrapFlags::none())
Value * CreateSub(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
Value * CreateAdd(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
void SetInsertPoint(BasicBlock *TheBB)
This specifies that created instructions should be appended to the end of the specified block.
Value * CreateICmp(CmpInst::Predicate P, Value *LHS, Value *RHS, const Twine &Name="")
Value * CreateMul(Value *LHS, Value *RHS, const Twine &Name="", bool HasNUW=false, bool HasNSW=false)
This provides a uniform API for creating instructions and inserting them into a basic block: either a...
void mergeDIAssignID(ArrayRef< const Instruction * > SourceInstructions)
Merge the DIAssignID metadata from this instruction and those attached to instructions in SourceInstr...
void insertBefore(Instruction *InsertPos)
Insert an unlinked instruction into a basic block immediately before the specified instruction.
void setAAMetadata(const AAMDNodes &N)
Sets the AA metadata on this instruction from the AAMDNodes structure.
bool isEHPad() const
Return true if the instruction is a variety of EH-block.
Instruction * user_back()
Specialize the methods defined in Value, as we know that an instruction can only be used by other ins...
MDNode * getMetadata(unsigned KindID) const
Get the metadata of given kind attached to this Instruction.
void setMetadata(unsigned KindID, MDNode *Node)
Set the metadata of the specified kind to the specified node.
AAMDNodes getAAMetadata() const
Returns the AA metadata for this instruction.
void setDebugLoc(DebugLoc Loc)
Set the debug location information for this instruction.
const DataLayout & getDataLayout() const
Get the data layout of the module this instruction belongs to.
A wrapper class for inspecting calls to intrinsic functions.
void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
PreservedAnalyses run(Loop &L, LoopAnalysisManager &AM, LoopStandardAnalysisResults &AR, LPMUpdater &U)
This is an important class for using LLVM in a threaded context.
PreservedAnalyses run(LoopNest &L, LoopAnalysisManager &AM, LoopStandardAnalysisResults &AR, LPMUpdater &U)
void printPipeline(raw_ostream &OS, function_ref< StringRef(StringRef)> MapClassName2PassName)
This class provides an interface for updating the loop pass manager based on mutations to the loop ne...
This is an alternative analysis pass to BlockFrequencyInfoWrapperPass.
static void getLazyBFIAnalysisUsage(AnalysisUsage &AU)
Helper for client passes to set up the analysis usage on behalf of this pass.
This is an alternative analysis pass to BranchProbabilityInfoWrapperPass.
An instruction for reading from memory.
void setAlignment(Align Align)
Value * getPointerOperand()
void setOrdering(AtomicOrdering Ordering)
Sets the ordering constraint of this load instruction.
Analysis pass that exposes the LoopInfo for a function.
bool contains(const LoopT *L) const
Return true if the specified loop is contained within in this loop.
BlockT * getHeader() const
void addBasicBlockToLoop(BlockT *NewBB, LoopInfoBase< BlockT, LoopT > &LI)
This method is used by other analyses to update loop information.
BlockT * getLoopPreheader() const
If there is a preheader for this loop, return it.
ArrayRef< BlockT * > getBlocks() const
Get a list of the basic blocks which make up this loop.
void getUniqueExitBlocks(SmallVectorImpl< BlockT * > &ExitBlocks) const
Return all unique successor blocks of this loop.
LoopT * getParentLoop() const
Return the parent loop if it exists or nullptr for top level loops.
Wrapper class to LoopBlocksDFS that provides a standard begin()/end() interface for the DFS reverse p...
void perform(const LoopInfo *LI)
Traverse the loop blocks and store the DFS result.
void verify(const DominatorTreeBase< BlockT, false > &DomTree) const
LoopT * getLoopFor(const BlockT *BB) const
Return the inner most loop that BB lives in.
The legacy pass manager's analysis pass to compute loop information.
bool wouldBeOutOfLoopUseRequiringLCSSA(const Value *V, const BasicBlock *ExitBB) const
This class represents a loop nest and can be used to query its properties.
Function * getParent() const
Return the function to which the loop-nest belongs.
Loop & getOutermostLoop() const
Return the outermost loop in the loop nest.
Captures loop safety information.
void copyColors(BasicBlock *New, BasicBlock *Old)
Copy colors of block Old into the block New.
const DenseMap< BasicBlock *, ColorVector > & getBlockColors() const
Returns block colors map that is used to update funclet operand bundles.
virtual bool isGuaranteedToExecute(const Instruction &Inst, const DominatorTree *DT, const Loop *CurLoop) const =0
Returns true if the instruction in a loop is guaranteed to execute at least once (under the assumptio...
Represents a single loop in the control flow graph.
bool hasLoopInvariantOperands(const Instruction *I) const
Return true if all the operands of the specified instruction are loop invariant.
bool isLoopInvariant(const Value *V) const
Return true if the specified value is loop invariant.
BasicBlock * getBlock() const
bool doesNotAccessMemory() const
Whether this function accesses no memory.
bool onlyAccessesArgPointees() const
Whether this function only (at most) accesses argument memory.
bool onlyReadsMemory() const
Whether this function only (at most) reads memory.
static MemoryLocation get(const LoadInst *LI)
Return a location with information about the memory reference by the given instruction.
An analysis that produces MemorySSA for a function.
MemorySSA * getMemorySSA() const
Get handle on MemorySSA.
void insertDef(MemoryDef *Def, bool RenameUses=false)
Insert a definition into the MemorySSA IR.
MemoryAccess * createMemoryAccessInBB(Instruction *I, MemoryAccess *Definition, const BasicBlock *BB, MemorySSA::InsertionPlace Point)
Create a MemoryAccess in MemorySSA at a specified point in a block.
void insertUse(MemoryUse *Use, bool RenameUses=false)
void removeMemoryAccess(MemoryAccess *, bool OptimizePhis=false)
Remove a MemoryAccess from MemorySSA, including updating all definitions and uses.
MemoryUseOrDef * createMemoryAccessAfter(Instruction *I, MemoryAccess *Definition, MemoryAccess *InsertPt)
Create a MemoryAccess in MemorySSA after an existing MemoryAccess.
void moveToPlace(MemoryUseOrDef *What, BasicBlock *BB, MemorySSA::InsertionPlace Where)
void wireOldPredecessorsToNewImmediatePredecessor(BasicBlock *Old, BasicBlock *New, ArrayRef< BasicBlock * > Preds, bool IdenticalEdgesWereMerged=true)
A new empty BasicBlock (New) now branches directly to Old.
MemoryAccess * getClobberingMemoryAccess(const Instruction *I, BatchAAResults &AA)
Given a memory Mod/Ref/ModRef'ing instruction, calling this will give you the nearest dominating Memo...
Legacy analysis pass which computes MemorySSA.
Encapsulates MemorySSA, including all data associated with memory accesses.
const AccessList * getBlockAccesses(const BasicBlock *BB) const
Return the list of MemoryAccess's for a given basic block.
MemorySSAWalker * getSkipSelfWalker()
bool dominates(const MemoryAccess *A, const MemoryAccess *B) const
Given two memory accesses in potentially different blocks, determine whether MemoryAccess A dominates...
void verifyMemorySSA(VerificationLevel=VerificationLevel::Fast) const
Verify that MemorySSA is self consistent (IE definitions dominate all uses, uses appear in the right ...
MemoryUseOrDef * getMemoryAccess(const Instruction *I) const
Given a memory Mod/Ref'ing instruction, get the MemorySSA access associated with it.
const DefsList * getBlockDefs(const BasicBlock *BB) const
Return the list of MemoryDef's and MemoryPhi's for a given basic block.
bool locallyDominates(const MemoryAccess *A, const MemoryAccess *B) const
Given two memory accesses in the same basic block, determine whether MemoryAccess A dominates MemoryA...
bool isLiveOnEntryDef(const MemoryAccess *MA) const
Return true if MA represents the live on entry value.
Class that has the common methods + fields of memory uses/defs.
MemoryAccess * getDefiningAccess() const
Get the access that produces the memory state used by this Use.
Represents read-only accesses to memory.
void addIncoming(Value *V, BasicBlock *BB)
Add an incoming value to the end of the PHI list.
op_range incoming_values()
void setIncomingBlock(unsigned i, BasicBlock *BB)
BasicBlock * getIncomingBlock(unsigned i) const
Return incoming basic block number i.
int getBasicBlockIndex(const BasicBlock *BB) const
Return the first index of the specified basic block in the value list for this PHI.
unsigned getNumIncomingValues() const
Return the number of incoming edges.
static PHINode * Create(Type *Ty, unsigned NumReservedValues, const Twine &NameStr="", InsertPosition InsertBefore=nullptr)
Constructors - NumReservedValues is a hint for the number of incoming edges that this phi node will h...
static PassRegistry * getPassRegistry()
getPassRegistry - Access the global registry object, which is automatically initialized at applicatio...
Pass interface - Implemented by all 'passes'.
PointerIntPair - This class implements a pair of a pointer and small integer.
static PoisonValue * get(Type *T)
Static factory methods - Return an 'poison' object of the specified type.
PredIteratorCache - This class is an extremely trivial cache for predecessor iterator queries.
size_t size(BasicBlock *BB)
ArrayRef< BasicBlock * > get(BasicBlock *BB)
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.
bool empty() const
Determine if the PriorityWorklist is empty or not.
bool insert(const T &X)
Insert a new element into the PriorityWorklist.
Helper class for SSA formation on a set of values defined in multiple blocks.
The main scalar evolution driver.
void forgetBlockAndLoopDispositions(Value *V=nullptr)
Called when the client has changed the disposition of values in a loop or block.
void forgetLoopDispositions()
Called when the client has changed the disposition of values in this loop.
bool remove(const value_type &X)
Remove an item from the set vector.
bool empty() const
Determine if the SetVector is empty or not.
iterator begin()
Get an iterator to the beginning of the SetVector.
bool insert(const value_type &X)
Insert a new element into the SetVector.
Flags controlling how much is checked when sinking or hoisting instructions.
SinkAndHoistLICMFlags(unsigned LicmMssaOptCap, unsigned LicmMssaNoAccForPromotionCap, bool IsSink, Loop &L, MemorySSA &MSSA)
unsigned LicmMssaNoAccForPromotionCap
A version of PriorityWorklist that selects small size optimized data structures for the vector and ma...
bool erase(PtrType Ptr)
Remove pointer from the set.
size_type count(ConstPtrType Ptr) const
count - Return 1 if the specified pointer is in the set, 0 otherwise.
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
bool contains(ConstPtrType Ptr) const
SmallPtrSet - This class implements a set which is optimized for holding SmallSize or less elements.
A SetVector that performs no allocations if smaller than a certain size.
This class consists of common code factored out of the SmallVector class to reduce code duplication b...
reference emplace_back(ArgTypes &&... Args)
void reserve(size_type N)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
An instruction for storing to memory.
void setAlignment(Align Align)
void setOrdering(AtomicOrdering Ordering)
Sets the ordering constraint of this store instruction.
static unsigned getPointerOperandIndex()
StringRef - Represent a constant reference to a string, i.e.
Provides information about what library functions are available for the current target.
TinyPtrVector - This class is specialized for cases where there are normally 0 or 1 element in a vect...
The instances of the Type class are immutable: once they are created, they are never changed.
bool isIntegerTy() const
True if this is an instance of IntegerType.
A Use represents the edge between a Value definition and its users.
const Use & getOperandUse(unsigned i) const
void setOperand(unsigned i, Value *Val)
Value * getOperand(unsigned i) const
unsigned getNumOperands() const
LLVM Value Representation.
Type * getType() const
All values are typed, get the type of this value.
bool hasOneUser() const
Return true if there is exactly one user of this value.
std::string getNameOrAsOperand() const
bool hasOneUse() const
Return true if there is exactly one use of this value.
void replaceAllUsesWith(Value *V)
Change all uses of this to point to a new Value.
user_iterator_impl< User > user_iterator
StringRef getName() const
Return a constant reference to the value's name.
void takeName(Value *V)
Transfer the name from V to this value.
constexpr ScalarTy getFixedValue() const
constexpr bool isScalable() const
Returns whether the quantity is scaled by a runtime quantity (vscale).
An efficient, type-erasing, non-owning reference to a callable.
const ParentTy * getParent() const
self_iterator getIterator()
This class implements an extremely fast bulk output stream that can only output to a stream.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
@ C
The default llvm calling convention, compatible with C.
OverflowingBinaryOp_match< LHS, RHS, Instruction::Sub, OverflowingBinaryOperator::NoSignedWrap > m_NSWSub(const LHS &L, const RHS &R)
bool match(Val *V, const Pattern &P)
CmpClass_match< LHS, RHS, ICmpInst, ICmpInst::Predicate > m_ICmp(ICmpInst::Predicate &Pred, const LHS &L, const RHS &R)
OneUse_match< T > m_OneUse(const T &SubPattern)
auto m_LogicalOr()
Matches L || R where L and R are arbitrary values.
class_match< Value > m_Value()
Match an arbitrary value and ignore it.
OverflowingBinaryOp_match< LHS, RHS, Instruction::Add, OverflowingBinaryOperator::NoSignedWrap > m_NSWAdd(const LHS &L, const RHS &R)
auto m_LogicalAnd()
Matches L && R where L and R are arbitrary values.
initializer< Ty > init(const Ty &Val)
DiagnosticInfoOptimizationBase::Argument NV
This is an optimization pass for GlobalISel generic memory operations.
void ReplaceInstWithInst(BasicBlock *BB, BasicBlock::iterator &BI, Instruction *I)
Replace the instruction specified by BI with the instruction specified by I.
pred_iterator pred_end(BasicBlock *BB)
SmallVector< DomTreeNode *, 16 > collectChildrenInLoop(DomTreeNode *N, const Loop *CurLoop)
Does a BFS from a given node to all of its children inside a given loop.
@ NeverOverflows
Never overflows.
bool all_of(R &&range, UnaryPredicate P)
Provide wrappers to std::all_of which take ranges instead of having to pass begin/end explicitly.
bool canSinkOrHoistInst(Instruction &I, AAResults *AA, DominatorTree *DT, Loop *CurLoop, MemorySSAUpdater &MSSAU, bool TargetExecutesOncePerLoop, SinkAndHoistLICMFlags &LICMFlags, OptimizationRemarkEmitter *ORE=nullptr)
Returns true if is legal to hoist or sink this instruction disregarding the possible introduction of ...
void set_intersect(S1Ty &S1, const S2Ty &S2)
set_intersect(A, B) - Compute A := A ^ B Identical to set_intersection, except that it works on set<>...
void salvageDebugInfo(const MachineRegisterInfo &MRI, MachineInstr &MI)
Assuming the instruction MI is going to be deleted, attempt to salvage debug users of MI by writing t...
void initializeLegacyLICMPassPass(PassRegistry &)
bool isDereferenceableAndAlignedPointer(const Value *V, Type *Ty, Align Alignment, const DataLayout &DL, const Instruction *CtxI=nullptr, AssumptionCache *AC=nullptr, const DominatorTree *DT=nullptr, const TargetLibraryInfo *TLI=nullptr)
Returns true if V is always a dereferenceable pointer with alignment greater or equal than requested.
bool formLCSSARecursively(Loop &L, const DominatorTree &DT, const LoopInfo *LI, ScalarEvolution *SE)
Put a loop nest into LCSSA form.
bool PointerMayBeCapturedBefore(const Value *V, bool ReturnCaptures, bool StoreCaptures, const Instruction *I, const DominatorTree *DT, bool IncludeI=false, unsigned MaxUsesToExplore=0, const LoopInfo *LI=nullptr)
PointerMayBeCapturedBefore - Return true if this pointer value may be captured by the enclosing funct...
const Value * getUnderlyingObject(const Value *V, unsigned MaxLookup=6)
This method strips off any GEP address adjustments, pointer casts or llvm.threadlocal....
iterator_range< early_inc_iterator_impl< detail::IterOfRange< RangeT > > > make_early_inc_range(RangeT &&Range)
Make a range that does early increment to allow mutation of the underlying range without disrupting i...
bool hoistRegion(DomTreeNode *, AAResults *, LoopInfo *, DominatorTree *, AssumptionCache *, TargetLibraryInfo *, Loop *, MemorySSAUpdater &, ScalarEvolution *, ICFLoopSafetyInfo *, SinkAndHoistLICMFlags &, OptimizationRemarkEmitter *, bool, bool AllowSpeculation)
Walk the specified region of the CFG (defined by all blocks dominated by the specified block,...
bool any_of(R &&range, UnaryPredicate P)
Provide wrappers to std::any_of which take ranges instead of having to pass begin/end explicitly.
bool isInstructionTriviallyDead(Instruction *I, const TargetLibraryInfo *TLI=nullptr)
Return true if the result produced by the instruction is not used, and the instruction will return.
pred_iterator pred_begin(BasicBlock *BB)
bool isGuard(const User *U)
Returns true iff U has semantics of a guard expressed in a form of call of llvm.experimental....
auto reverse(ContainerTy &&C)
OverflowResult computeOverflowForSignedSub(const Value *LHS, const Value *RHS, const SimplifyQuery &SQ)
bool isModSet(const ModRefInfo MRI)
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
void report_fatal_error(Error Err, bool gen_crash_diag=true)
Report a serious error, calling any installed error handler.
bool isModOrRefSet(const ModRefInfo MRI)
bool isSafeToSpeculativelyExecute(const Instruction *I, const Instruction *CtxI=nullptr, AssumptionCache *AC=nullptr, const DominatorTree *DT=nullptr, const TargetLibraryInfo *TLI=nullptr, bool UseVariableInfo=true)
Return true if the instruction does not have any effects besides calculating the result and does not ...
bool isNotVisibleOnUnwind(const Value *Object, bool &RequiresNoCaptureBeforeUnwind)
Return true if Object memory is not visible after an unwind, in the sense that program semantics cann...
void getLoopAnalysisUsage(AnalysisUsage &AU)
Helper to consistently add the set of standard passes to a loop pass's AnalysisUsage.
RNSuccIterator< NodeRef, BlockT, RegionT > succ_begin(NodeRef Node)
BasicBlock * SplitBlockPredecessors(BasicBlock *BB, ArrayRef< BasicBlock * > Preds, const char *Suffix, DominatorTree *DT, LoopInfo *LI=nullptr, MemorySSAUpdater *MSSAU=nullptr, bool PreserveLCSSA=false)
This method introduces at least one new basic block into the function and moves some of the predecess...
ModRefInfo
Flags indicating whether a memory access modifies or references memory.
bool VerifyMemorySSA
Enables verification of MemorySSA.
RNSuccIterator< NodeRef, BlockT, RegionT > succ_end(NodeRef Node)
bool salvageKnowledge(Instruction *I, AssumptionCache *AC=nullptr, DominatorTree *DT=nullptr)
Calls BuildAssumeFromInst and if the resulting llvm.assume is valid insert if before I.
bool hasDisableLICMTransformsHint(const Loop *L)
Look for the loop attribute that disables the LICM transformation heuristics.
OverflowResult computeOverflowForSignedAdd(const WithCache< const Value * > &LHS, const WithCache< const Value * > &RHS, const SimplifyQuery &SQ)
@ Mul
Product of integers.
void appendLoopsToWorklist(RangeT &&, SmallPriorityWorklist< Loop *, 4 > &)
Utility that implements appending of loops onto a worklist given a range.
bool isIdentifiedFunctionLocal(const Value *V)
Return true if V is umabigously identified at the function-level.
bool isDereferenceablePointer(const Value *V, Type *Ty, const DataLayout &DL, const Instruction *CtxI=nullptr, AssumptionCache *AC=nullptr, const DominatorTree *DT=nullptr, const TargetLibraryInfo *TLI=nullptr)
Return true if this is always a dereferenceable pointer.
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
PreservedAnalyses getLoopPassPreservedAnalyses()
Returns the minimum set of Analyses that all loop passes must preserve.
auto find_if(R &&Range, UnaryPredicate P)
Provide wrappers to std::find_if which take ranges instead of having to pass begin/end explicitly.
void erase_if(Container &C, UnaryPredicate P)
Provide a container algorithm similar to C++ Library Fundamentals v2's erase_if which is equivalent t...
auto predecessors(const MachineBasicBlock *BB)
Type * getLoadStoreType(const Value *I)
A helper function that returns the type of a load or store instruction.
bool sinkRegion(DomTreeNode *, AAResults *, LoopInfo *, DominatorTree *, TargetLibraryInfo *, TargetTransformInfo *, Loop *CurLoop, MemorySSAUpdater &, ICFLoopSafetyInfo *, SinkAndHoistLICMFlags &, OptimizationRemarkEmitter *, Loop *OutermostLoop=nullptr)
Walk the specified region of the CFG (defined by all blocks dominated by the specified block,...
cl::opt< unsigned > SetLicmMssaNoAccForPromotionCap
unsigned pred_size(const MachineBasicBlock *BB)
bool isKnownNonNegative(const Value *V, const SimplifyQuery &SQ, unsigned Depth=0)
Returns true if the give value is known to be non-negative.
bool promoteLoopAccessesToScalars(const SmallSetVector< Value *, 8 > &, SmallVectorImpl< BasicBlock * > &, SmallVectorImpl< BasicBlock::iterator > &, SmallVectorImpl< MemoryAccess * > &, PredIteratorCache &, LoopInfo *, DominatorTree *, AssumptionCache *AC, const TargetLibraryInfo *, TargetTransformInfo *, Loop *, MemorySSAUpdater &, ICFLoopSafetyInfo *, OptimizationRemarkEmitter *, bool AllowSpeculation, bool HasReadsOutsideSet)
Try to promote memory values to scalars by sinking stores out of the loop and moving loads to before ...
cl::opt< unsigned > SetLicmMssaOptCap
bool sinkRegionForLoopNest(DomTreeNode *, AAResults *, LoopInfo *, DominatorTree *, TargetLibraryInfo *, TargetTransformInfo *, Loop *, MemorySSAUpdater &, ICFLoopSafetyInfo *, SinkAndHoistLICMFlags &, OptimizationRemarkEmitter *)
Call sinkRegion on loops contained within the specified loop in order from innermost to outermost.
bool isWritableObject(const Value *Object, bool &ExplicitlyDereferenceableOnly)
Return true if the Object is writable, in the sense that any location based on this pointer that can ...
Implement std::hash so that hash_code can be used in STL containers.
void swap(llvm::BitVector &LHS, llvm::BitVector &RHS)
Implement std::swap in terms of BitVector swap.
A collection of metadata nodes that might be associated with a memory access used by the alias-analys...
AAMDNodes merge(const AAMDNodes &Other) const
Given two sets of AAMDNodes applying to potentially different locations, determine the best AAMDNodes...
This struct is a compact representation of a valid (non-zero power of two) alignment.
unsigned MssaNoAccForPromotionCap
The adaptor from a function pass to a loop pass computes these analyses and makes them available to t...
TargetTransformInfo & TTI
A lightweight accessor for an operand bundle meant to be passed around by value.
uint32_t getTagID() const
Return the tag of this operand bundle as an integer.
A CRTP mix-in to automatically provide informational APIs needed for passes.