From: Jeff Cohen Date: Tue, 1 Mar 2005 03:46:11 +0000 (+0000) Subject: Fixed the following LSR bugs: X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=0456e4a079de51087978c177b1de63343731f4fb;p=oota-llvm.git Fixed the following LSR bugs: * Loop invariant code does not dominate the loop header, but rather the end of the loop preheader. * The base for a reduced GEP isn't a constant unless all of its operands (preceding the induction variable) are constant. * Allow induction variable elimination for the simple case after all. Also made changes recommended by Chris for properly deleting instructions. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@20383 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Transforms/Scalar/LoopStrengthReduce.cpp b/lib/Transforms/Scalar/LoopStrengthReduce.cpp index 6494490167f..56bc43e62f3 100644 --- a/lib/Transforms/Scalar/LoopStrengthReduce.cpp +++ b/lib/Transforms/Scalar/LoopStrengthReduce.cpp @@ -78,14 +78,9 @@ DeleteTriviallyDeadInstructions(std::set &Insts) { Instruction *I = *Insts.begin(); Insts.erase(Insts.begin()); if (isInstructionTriviallyDead(I)) { - for (unsigned i = 0, e = I->getNumOperands(); i != e; ++i) { - // Note: the PHI nodes had dropAllReferences() called on it, so its - // operands will all be NULL. - Value *V = I->getOperand(i); - if (V) - if (Instruction *U = dyn_cast(V)) - Insts.insert(U); - } + for (unsigned i = 0, e = I->getNumOperands(); i != e; ++i) + if (Instruction *U = dyn_cast(I->getOperand(i))) + Insts.insert(U); I->getParent()->getInstList().erase(I); Changed = true; } @@ -111,6 +106,8 @@ void LoopStrengthReduce::strengthReduceGEP(GetElementPtrInst *GEPI, Loop *L, std::vector inc_op_vector; Value *CanonicalIndVar = L->getCanonicalInductionVariable(); BasicBlock *Header = L->getHeader(); + BasicBlock *Preheader = L->getLoopPreheader(); + bool AllConstantOperands = true; for (unsigned op = 1, e = GEPI->getNumOperands(); op != e; ++op) { Value *operand = GEPI->getOperand(op); @@ -125,9 +122,10 @@ void LoopStrengthReduce::strengthReduceGEP(GetElementPtrInst *GEPI, Loop *L, } else if (isa(operand)) { pre_op_vector.push_back(operand); } else if (Instruction *inst = dyn_cast(operand)) { - if (!DS->dominates(inst, Header->begin())) + if (!DS->dominates(inst, Preheader->getTerminator())) return; pre_op_vector.push_back(operand); + AllConstantOperands = false; } else return; } @@ -139,7 +137,7 @@ void LoopStrengthReduce::strengthReduceGEP(GetElementPtrInst *GEPI, Loop *L, // their constituent operations so we have explicit multiplications to work // with. if (Instruction *GepPtrOp = dyn_cast(GEPI->getOperand(0))) - if (!DS->dominates(GepPtrOp, Header->begin())) + if (!DS->dominates(GepPtrOp, Preheader->getTerminator())) return; // If all operands of the GEP we are going to insert into the preheader @@ -148,9 +146,8 @@ void LoopStrengthReduce::strengthReduceGEP(GetElementPtrInst *GEPI, Loop *L, // If there is only one operand after the initial non-constant one, we know // that it was the induction variable, and has been replaced by a constant // null value. In this case, replace the GEP with a use of pointer directly. - BasicBlock *Preheader = L->getLoopPreheader(); Value *PreGEP; - if (isa(GEPI->getOperand(0))) { + if (AllConstantOperands && isa(GEPI->getOperand(0))) { Constant *C = dyn_cast(GEPI->getOperand(0)); PreGEP = ConstantExpr::getGetElementPtr(C, pre_op_vector); } else if (pre_op_vector.size() == 1) { @@ -177,7 +174,10 @@ void LoopStrengthReduce::strengthReduceGEP(GetElementPtrInst *GEPI, Loop *L, GetElementPtrInst *StrGEP = new GetElementPtrInst(NewPHI, inc_op_vector, GEPI->getName()+".inc", IncrInst); - NewPHI->addIncoming(StrGEP, IncrInst->getParent()); + pred_iterator PI = pred_begin(Header); + if (*PI == Preheader) + ++PI; + NewPHI->addIncoming(StrGEP, *PI); if (GEPI->getNumOperands() - 1 == indvar) { // If there were no operands following the induction variable, replace all @@ -242,24 +242,20 @@ void LoopStrengthReduce::runOnLoop(Loop *L) { // 4. the add is used by the cann indvar // If all four cases above are true, then we can remove both the add and // the cann indvar. -#if 0 - // FIXME: it's not clear this code is correct. An induction variable with - // but one use, an increment, implies an infinite loop. Not illegal, but - // of questionable utility. It also does not update the loop info with the - // new induction variable. + // FIXME: this needs to eliminate an induction variable even if it's being + // compared against some value to decide loop termination. if (PN->hasOneUse()) { BinaryOperator *BO = dyn_cast(*(PN->use_begin())); if (BO && BO->getOpcode() == Instruction::Add) if (BO->hasOneUse()) { - PHINode *PotentialIndvar = dyn_cast(*(BO->use_begin())); - if (PotentialIndvar && PN == PotentialIndvar) { - PN->dropAllReferences(); + if (PN == dyn_cast(*(BO->use_begin()))) { DeadInsts.insert(BO); - DeadInsts.insert(PN); + // Break the cycle, then delete the PHI. + PN->replaceAllUsesWith(UndefValue::get(PN->getType())); + PN->eraseFromParent(); DeleteTriviallyDeadInstructions(DeadInsts); } } } -#endif } }