diff options
Diffstat (limited to 'llvm/lib/Transforms/InstCombine/InstCombineCalls.cpp')
| -rw-r--r-- | llvm/lib/Transforms/InstCombine/InstCombineCalls.cpp | 35 | 
1 files changed, 29 insertions, 6 deletions
| diff --git a/llvm/lib/Transforms/InstCombine/InstCombineCalls.cpp b/llvm/lib/Transforms/InstCombine/InstCombineCalls.cpp index dab200d..8d9933b 100644 --- a/llvm/lib/Transforms/InstCombine/InstCombineCalls.cpp +++ b/llvm/lib/Transforms/InstCombine/InstCombineCalls.cpp @@ -582,6 +582,18 @@ static Instruction *foldCttzCtlz(IntrinsicInst &II, InstCombinerImpl &IC) {            IC.Builder.CreateBinaryIntrinsic(Intrinsic::ctlz, C, Op1);        return BinaryOperator::CreateSub(ConstCtlz, X);      } + +    // ctlz(~x & (x - 1)) -> bitwidth - cttz(x, false) +    if (Op0->hasOneUse() && +        match(Op0, +              m_c_And(m_Not(m_Value(X)), m_Add(m_Deferred(X), m_AllOnes())))) { +      Type *Ty = II.getType(); +      unsigned BitWidth = Ty->getScalarSizeInBits(); +      auto *Cttz = IC.Builder.CreateIntrinsic(Intrinsic::cttz, Ty, +                                              {X, IC.Builder.getFalse()}); +      auto *Bw = ConstantInt::get(Ty, APInt(BitWidth, BitWidth)); +      return IC.replaceInstUsesWith(II, IC.Builder.CreateSub(Bw, Cttz)); +    }    }    // cttz(Pow2) -> Log2(Pow2) @@ -4003,18 +4015,29 @@ Instruction *InstCombinerImpl::visitCallInst(CallInst &CI) {    // Try to fold intrinsic into select/phi operands. This is legal if:    //  * The intrinsic is speculatable. -  //  * The select condition is not a vector, or the intrinsic does not -  //    perform cross-lane operations. -  if (isSafeToSpeculativelyExecuteWithVariableReplaced(&CI) && -      isNotCrossLaneOperation(II)) +  //  * The operand is one of the following: +  //    - a phi. +  //    - a select with a scalar condition. +  //    - a select with a vector condition and II is not a cross lane operation. +  if (isSafeToSpeculativelyExecuteWithVariableReplaced(&CI)) {      for (Value *Op : II->args()) { -      if (auto *Sel = dyn_cast<SelectInst>(Op)) -        if (Instruction *R = FoldOpIntoSelect(*II, Sel)) +      if (auto *Sel = dyn_cast<SelectInst>(Op)) { +        bool IsVectorCond = Sel->getCondition()->getType()->isVectorTy(); +        if (IsVectorCond && !isNotCrossLaneOperation(II)) +          continue; +        // Don't replace a scalar select with a more expensive vector select if +        // we can't simplify both arms of the select. +        bool SimplifyBothArms = +            !Op->getType()->isVectorTy() && II->getType()->isVectorTy(); +        if (Instruction *R = FoldOpIntoSelect( +                *II, Sel, /*FoldWithMultiUse=*/false, SimplifyBothArms))            return R; +      }        if (auto *Phi = dyn_cast<PHINode>(Op))          if (Instruction *R = foldOpIntoPhi(*II, Phi))            return R;      } +  }    if (Instruction *Shuf = foldShuffledIntrinsicOperands(II))      return Shuf; | 
