diff options
Diffstat (limited to 'llvm/lib/Transforms')
| -rw-r--r-- | llvm/lib/Transforms/Scalar/LoopIdiomRecognize.cpp | 25 | 
1 files changed, 15 insertions, 10 deletions
| diff --git a/llvm/lib/Transforms/Scalar/LoopIdiomRecognize.cpp b/llvm/lib/Transforms/Scalar/LoopIdiomRecognize.cpp index 3e9546fda8e..d8692198f7a 100644 --- a/llvm/lib/Transforms/Scalar/LoopIdiomRecognize.cpp +++ b/llvm/lib/Transforms/Scalar/LoopIdiomRecognize.cpp @@ -1405,16 +1405,21 @@ bool LoopIdiomRecognize::recognizeAndInsertCTLZ() {    if (DefX->getOpcode() == Instruction::AShr && !isKnownNonNegative(InitX, *DL))      return false; -  // If we check X != 0 before entering the loop we don't need a zero -  // check in CTLZ intrinsic, but only if Cnt Phi is not used outside of the -  // loop (if it is used we count CTLZ(X >> 1)). -  if (!IsCntPhiUsedOutsideLoop) -    if (BasicBlock *PreCondBB = PH->getSinglePredecessor()) -      if (BranchInst *PreCondBr = -          dyn_cast<BranchInst>(PreCondBB->getTerminator())) { -        if (matchCondition(PreCondBr, PH) == InitX) -          ZeroCheck = true; -      } +  // If we are using the count instruction outside the loop, make sure we +  // have a zero check as a precondition. Without the check the loop would run +  // one iteration for before any check of the input value. This means 0 and 1 +  // would have identical behavior in the original loop and thus +  if (!IsCntPhiUsedOutsideLoop) { +    auto *PreCondBB = PH->getSinglePredecessor(); +    if (!PreCondBB) +      return false; +    auto *PreCondBI = dyn_cast<BranchInst>(PreCondBB->getTerminator()); +    if (!PreCondBI) +      return false; +    if (matchCondition(PreCondBI, PH) != InitX) +      return false; +    ZeroCheck = true; +  }    // Check if CTLZ intrinsic is profitable. Assume it is always profitable    // if we delete the loop (the loop has only 6 instructions): | 

