Index: include/llvm/Analysis/ScalarEvolutionExpander.h =================================================================== --- include/llvm/Analysis/ScalarEvolutionExpander.h +++ include/llvm/Analysis/ScalarEvolutionExpander.h @@ -83,6 +83,25 @@ typedef IRBuilder BuilderType; BuilderType Builder; + /// Stack of pointers to saved insert points, used to keep insert points + /// consistent when instructions are moved. + SmallVector InsertPointGuards; + class SCEVInsertPointGuard { + public: + SCEVInsertPointGuard(IRBuilderBase &B, SCEVExpander *SE) + : Guard(B), SE(SE) { + SE->InsertPointGuards.push_back(&Guard); + } + + ~SCEVInsertPointGuard() { + assert(SE->InsertPointGuards.back() == &Guard); + SE->InsertPointGuards.pop_back(); + } + + BuilderType::InsertPointGuard Guard; + SCEVExpander *SE; + }; + #ifndef NDEBUG const char *DebugType; #endif @@ -146,6 +165,8 @@ SmallVectorImpl &DeadInsts, const TargetTransformInfo *TTI = nullptr); + void fixupInsertPoints(Instruction *I); + /// \brief Insert code to directly compute the specified SCEV expression /// into the program. The inserted code is inserted into the specified /// block. Index: include/llvm/IR/IRBuilder.h =================================================================== --- include/llvm/IR/IRBuilder.h +++ include/llvm/IR/IRBuilder.h @@ -225,6 +225,9 @@ Builder.restoreIP(InsertPoint(Block, Point)); Builder.SetCurrentDebugLocation(DbgLoc); } + + BasicBlock::iterator GetInsertPoint() const { return Point; } + void SetInsertPoint(BasicBlock::iterator I) { Point = I; } }; // \brief RAII object that stores the current fast math settings and restores Index: lib/Analysis/ScalarEvolutionExpander.cpp =================================================================== --- lib/Analysis/ScalarEvolutionExpander.cpp +++ lib/Analysis/ScalarEvolutionExpander.cpp @@ -196,7 +196,7 @@ // Save the original insertion point so we can restore it when we're done. DebugLoc Loc = Builder.GetInsertPoint()->getDebugLoc(); - BuilderType::InsertPointGuard Guard(Builder); + SCEVInsertPointGuard Guard(Builder, this); // Move the insertion point out of as many loops as we can. while (const Loop *L = SE.LI.getLoopFor(Builder.GetInsertBlock())) { @@ -523,7 +523,7 @@ } // Save the original insertion point so we can restore it when we're done. - BuilderType::InsertPointGuard Guard(Builder); + SCEVInsertPointGuard Guard(Builder, this); // Move the insertion point out of as many loops as we can. while (const Loop *L = SE.LI.getLoopFor(Builder.GetInsertBlock())) { @@ -542,39 +542,37 @@ return GEP; } - // Save the original insertion point so we can restore it when we're done. - BuilderType::InsertPoint SaveInsertPt = Builder.saveIP(); + { + SCEVInsertPointGuard Guard(Builder, this); - // Move the insertion point out of as many loops as we can. - while (const Loop *L = SE.LI.getLoopFor(Builder.GetInsertBlock())) { - if (!L->isLoopInvariant(V)) break; + // Move the insertion point out of as many loops as we can. + while (const Loop *L = SE.LI.getLoopFor(Builder.GetInsertBlock())) { + if (!L->isLoopInvariant(V)) break; - bool AnyIndexNotLoopInvariant = - std::any_of(GepIndices.begin(), GepIndices.end(), - [L](Value *Op) { return !L->isLoopInvariant(Op); }); + bool AnyIndexNotLoopInvariant = + std::any_of(GepIndices.begin(), GepIndices.end(), + [L](Value *Op) { return !L->isLoopInvariant(Op); }); - if (AnyIndexNotLoopInvariant) - break; - - BasicBlock *Preheader = L->getLoopPreheader(); - if (!Preheader) break; + if (AnyIndexNotLoopInvariant) + break; - // Ok, move up a level. - Builder.SetInsertPoint(Preheader->getTerminator()); - } + BasicBlock *Preheader = L->getLoopPreheader(); + if (!Preheader) break; - // Insert a pretty getelementptr. Note that this GEP is not marked inbounds, - // because ScalarEvolution may have changed the address arithmetic to - // compute a value which is beyond the end of the allocated object. - Value *Casted = V; - if (V->getType() != PTy) - Casted = InsertNoopCastOfTo(Casted, PTy); - Value *GEP = Builder.CreateGEP(OriginalElTy, Casted, GepIndices, "scevgep"); - Ops.push_back(SE.getUnknown(GEP)); - rememberInstruction(GEP); + // Ok, move up a level. + Builder.SetInsertPoint(Preheader->getTerminator()); + } - // Restore the original insert point. - Builder.restoreIP(SaveInsertPt); + // Insert a pretty getelementptr. Note that this GEP is not marked inbounds, + // because ScalarEvolution may have changed the address arithmetic to + // compute a value which is beyond the end of the allocated object. + Value *Casted = V; + if (V->getType() != PTy) + Casted = InsertNoopCastOfTo(Casted, PTy); + Value *GEP = Builder.CreateGEP(OriginalElTy, Casted, GepIndices, "scevgep"); + Ops.push_back(SE.getUnknown(GEP)); + rememberInstruction(GEP); + } return expand(SE.getAddExpr(Ops)); } @@ -905,6 +903,23 @@ } } +/// If the insert point of the current builder or any of the builders on the +/// stack of saved builders has 'I' as its insert point, update it to point to +/// the instruction after 'I'. This is intended to be used when the instruction +/// 'I' is being moved. If this fixup is not done and 'I' is moved to a +/// different block, the inconsistent insert point (with a mismatched +/// Instruction and Block) can lead to an instruction being inserted in a block +/// other than its parent. +void SCEVExpander::fixupInsertPoints(Instruction *I) { + BasicBlock::iterator It(*I); + BasicBlock::iterator NewInsertPt = std::next(It); + if (Builder.GetInsertPoint() == It) + Builder.SetInsertPoint(Builder.GetInsertBlock(), NewInsertPt); + for (auto *InsertPtGuard : InsertPointGuards) + if (InsertPtGuard->GetInsertPoint() == It) + InsertPtGuard->SetInsertPoint(NewInsertPt); +} + /// hoistStep - Attempt to hoist a simple IV increment above InsertPos to make /// it available to other uses in this loop. Recursively hoist any operands, /// until we reach a value that dominates InsertPos. @@ -934,6 +949,7 @@ break; } for (auto I = IVIncs.rbegin(), E = IVIncs.rend(); I != E; ++I) { + fixupInsertPoints(*I); (*I)->moveBefore(InsertPos); } return true; @@ -988,12 +1004,14 @@ /// \brief Hoist the addrec instruction chain rooted in the loop phi above the /// position. This routine assumes that this is possible (has been checked). static void hoistBeforePos(DominatorTree *DT, Instruction *InstToHoist, - Instruction *Pos, PHINode *LoopPhi) { + Instruction *Pos, PHINode *LoopPhi, + SCEVExpander *SE) { do { if (DT->dominates(InstToHoist, Pos)) break; // Make sure the increment is where we want it. But don't move it // down past a potential existing post-inc user. + SE->fixupInsertPoints(InstToHoist); InstToHoist->moveBefore(Pos); Pos = InstToHoist; InstToHoist = cast(InstToHoist->getOperand(0)); @@ -1142,7 +1160,7 @@ // Potentially, move the increment. We have made sure in // isExpandedAddRecExprPHI or hoistIVInc that this is possible. if (L == IVIncInsertLoop) - hoistBeforePos(&SE.DT, IncV, IVIncInsertPos, AddRecPhiMatch); + hoistBeforePos(&SE.DT, IncV, IVIncInsertPos, AddRecPhiMatch, this); // Ok, the add recurrence looks usable. // Remember this PHI, even in post-inc mode. @@ -1154,7 +1172,7 @@ } // Save the original insertion point so we can restore it when we're done. - BuilderType::InsertPointGuard Guard(Builder); + SCEVInsertPointGuard Guard(Builder, this); // Another AddRec may need to be recursively expanded below. For example, if // this AddRec is quadratic, the StepV may itself be an AddRec in this @@ -1319,7 +1337,7 @@ Value *StepV; { // Expand the step somewhere that dominates the loop header. - BuilderType::InsertPointGuard Guard(Builder); + SCEVInsertPointGuard Guard(Builder, this); StepV = expandCodeFor(Step, IntTy, &L->getHeader()->front()); } Result = expandIVInc(PN, StepV, L, ExpandTy, IntTy, useSubtract); @@ -1667,7 +1685,7 @@ if (I != InsertedExpressions.end()) return I->second; - BuilderType::InsertPointGuard Guard(Builder); + SCEVInsertPointGuard Guard(Builder, this); Builder.SetInsertPoint(InsertPt); // Expand the expression into instructions. @@ -1708,7 +1726,7 @@ SE.getConstant(Ty, 1), L, SCEV::FlagAnyWrap); // Emit code for it. - BuilderType::InsertPointGuard Guard(Builder); + SCEVInsertPointGuard Guard(Builder, this); PHINode *V = cast(expandCodeFor(H, nullptr, &L->getHeader()->front()));