Index: llvm/include/llvm/IR/PatternMatch.h =================================================================== --- llvm/include/llvm/IR/PatternMatch.h +++ llvm/include/llvm/IR/PatternMatch.h @@ -187,11 +187,35 @@ return false; } }; +// Either constexpr if or renaming ConstantFP::getValueAPF to +// ConstantFP::getValue is needed to do it via single template +// function for both apint/apfloat. +struct apfloat_match { + const APFloat *&Res; + apfloat_match(const APFloat *&R) : Res(R) {} + template bool match(ITy *V) { + if (auto *CI = dyn_cast(V)) { + Res = &CI->getValueAPF(); + return true; + } + if (V->getType()->isVectorTy()) + if (const auto *C = dyn_cast(V)) + if (auto *CI = dyn_cast_or_null(C->getSplatValue())) { + Res = &CI->getValueAPF(); + return true; + } + return false; + } +}; /// \brief Match a ConstantInt or splatted ConstantVector, binding the /// specified pointer to the contained APInt. inline apint_match m_APInt(const APInt *&Res) { return Res; } +/// \brief Match a ConstantFP or splatted ConstantVector, binding the +/// specified pointer to the contained APFloat. +inline apfloat_match m_APFloat(const APFloat *&Res) { return Res; } + template struct constantint_match { template bool match(ITy *V) { if (const auto *CI = dyn_cast(V)) { Index: llvm/lib/Analysis/ValueTracking.cpp =================================================================== --- llvm/lib/Analysis/ValueTracking.cpp +++ llvm/lib/Analysis/ValueTracking.cpp @@ -3826,6 +3826,61 @@ return false; } +/// Match clamp pattern for float types without care about NaNs or signed zeros. +/// Given non-min/max outer cmp/select from the clamp pattern this +/// function recognizes if it can be substitued by a "canonical" min/max +/// pattern. +static SelectPatternResult matchFastFloatClamp(CmpInst::Predicate Pred, + Value *CmpLHS, Value *CmpRHS, + Value *TrueVal, Value *FalseVal, + Value *&LHS, Value *&RHS) { + // Try to match X `pred` C1 ? C1 : ProperMinMax(X, C2) into + // MaxMin(C1, ProperMinMax(X, C2) and return description of the + // outer one. + + // First, check if select has inversed order: + if (CmpRHS == FalseVal) { + std::swap(TrueVal, FalseVal); + Pred = CmpInst::getInversePredicate(Pred); + } + + // Assume success now. If there's no match, callers should not use these anyway. + LHS = TrueVal; + RHS = FalseVal; + + const APFloat *FC1; + if (CmpRHS != TrueVal || !match(CmpRHS, m_APFloat(FC1)) || !FC1->isFinite()) + return {SPF_UNKNOWN, SPNB_NA, false}; + + const APFloat *FC2; + switch (Pred) { + case CmpInst::FCMP_OLT: + case CmpInst::FCMP_OLE: + case CmpInst::FCMP_ULT: + case CmpInst::FCMP_ULE: + if (match(FalseVal, + m_CombineOr(m_OrdFMin(m_Specific(CmpLHS), m_APFloat(FC2)), + m_UnordFMin(m_Specific(CmpLHS), m_APFloat(FC2)))) && + FC1->compare(*FC2) == APFloat::cmpResult::cmpLessThan) + return {SPF_FMAXNUM, SPNB_RETURNS_ANY, false}; + break; + case CmpInst::FCMP_OGT: + case CmpInst::FCMP_OGE: + case CmpInst::FCMP_UGT: + case CmpInst::FCMP_UGE: + if (match(FalseVal, + m_CombineOr(m_OrdFMax(m_Specific(CmpLHS), m_APFloat(FC2)), + m_UnordFMax(m_Specific(CmpLHS), m_APFloat(FC2)))) && + FC1->compare(*FC2) == APFloat::cmpResult::cmpGreaterThan) + return {SPF_FMINNUM, SPNB_RETURNS_ANY, false}; + break; + default: + break; + } + + return {SPF_UNKNOWN, SPNB_NA, false}; +} + /// Match non-obvious integer minimum and maximum sequences. static SelectPatternResult matchMinMax(CmpInst::Predicate Pred, Value *CmpLHS, Value *CmpRHS, @@ -4033,7 +4088,15 @@ } } - return matchMinMax(Pred, CmpLHS, CmpRHS, TrueVal, FalseVal, LHS, RHS); + if (CmpInst::isIntPredicate(Pred)) + return matchMinMax(Pred, CmpLHS, CmpRHS, TrueVal, FalseVal, LHS, RHS); + + if (NaNBehavior != SPNB_RETURNS_ANY || + (!FMF.noSignedZeros() && !isKnownNonZero(CmpLHS) && + !isKnownNonZero(CmpRHS))) + return {SPF_UNKNOWN, SPNB_NA, false}; + + return matchFastFloatClamp(Pred, CmpLHS, CmpRHS, TrueVal, FalseVal, LHS, RHS); } static Value *lookThroughCast(CmpInst *CmpI, Value *V1, Value *V2, Index: llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp =================================================================== --- llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp +++ llvm/lib/Transforms/InstCombine/InstCombineSelect.cpp @@ -1317,9 +1317,15 @@ auto SPF = SPR.Flavor; if (SelectPatternResult::isMinOrMax(SPF)) { - // Canonicalize so that type casts are outside select patterns. - if (LHS->getType()->getPrimitiveSizeInBits() != - SelType->getPrimitiveSizeInBits()) { + // Canonicalize so that + // - type casts are outside select patterns. + // - clamp is transformed to min/max pattern + Value *CmpLHS = cast(CondVal)->getOperand(0); + Value *CmpRHS = cast(CondVal)->getOperand(1); + if ((LHS->getType()->getPrimitiveSizeInBits() != + SelType->getPrimitiveSizeInBits()) || + (CmpLHS != LHS && CmpLHS != RHS) || + (CmpRHS != LHS && CmpRHS != RHS)) { CmpInst::Predicate Pred = getCmpPredicateForMinMax(SPF, SPR.Ordered); Value *Cmp; Index: llvm/test/Transforms/InstCombine/clamp-to-minmax.ll =================================================================== --- /dev/null +++ llvm/test/Transforms/InstCombine/clamp-to-minmax.ll @@ -0,0 +1,501 @@ +; NOTE: Assertions have been autogenerated by utils/update_test_checks.py +; RUN: opt < %s -instcombine -S | FileCheck %s + +; (X < C1) ? C1 : MIN(X, C2) +define float @clamp_float_fast_ordered_strict_maxmin(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_ordered_strict_maxmin( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp fast olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast oge float [[MIN]], 1.000000e+00 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MIN]], float 1.000000e+00 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast olt float %x, 255.0 + %min = select i1 %cmp2, float %x, float 255.0 + %cmp1 = fcmp fast olt float %x, 1.0 + %r = select i1 %cmp1, float 1.0, float %min + ret float %r +} + +; (X <= C1) ? C1 : MIN(X, C2) +define float @clamp_float_fast_ordered_nonstrict_maxmin(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_ordered_nonstrict_maxmin( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp fast olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast oge float [[MIN]], 1.000000e+00 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MIN]], float 1.000000e+00 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast olt float %x, 255.0 + %min = select i1 %cmp2, float %x, float 255.0 + %cmp1 = fcmp fast ole float %x, 1.0 + %r = select i1 %cmp1, float 1.0, float %min + ret float %r +} + +; (X > C1) ? C1 : MAX(X, C2) +define float @clamp_float_fast_ordered_strict_minmax(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_ordered_strict_minmax( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp fast ogt float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2]], float [[X]], float 1.000000e+00 +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast ole float [[MAX]], 2.550000e+02 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MAX]], float 2.550000e+02 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast ogt float %x, 1.0 + %max = select i1 %cmp2, float %x, float 1.0 + %cmp1 = fcmp fast ogt float %x, 255.0 + %r = select i1 %cmp1, float 255.0, float %max + ret float %r +} + +; (X >= C1) ? C1 : MAX(X, C2) +define float @clamp_float_fast_ordered_nonstrict_minmax(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_ordered_nonstrict_minmax( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp fast ogt float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2]], float [[X]], float 1.000000e+00 +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast ole float [[MAX]], 2.550000e+02 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MAX]], float 2.550000e+02 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast ogt float %x, 1.0 + %max = select i1 %cmp2, float %x, float 1.0 + %cmp1 = fcmp fast oge float %x, 255.0 + %r = select i1 %cmp1, float 255.0, float %max + ret float %r +} + + +; The same for unordered + +; (X < C1) ? C1 : MIN(X, C2) +define float @clamp_float_fast_unordered_strict_maxmin(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_unordered_strict_maxmin( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp fast oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast oge float [[MIN]], 1.000000e+00 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MIN]], float 1.000000e+00 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast ult float %x, 255.0 + %min = select i1 %cmp2, float %x, float 255.0 + %cmp1 = fcmp fast ult float %x, 1.0 + %r = select i1 %cmp1, float 1.0, float %min + ret float %r +} + +; (X <= C1) ? C1 : MIN(X, C2) +define float @clamp_float_fast_unordered_nonstrict_maxmin(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_unordered_nonstrict_maxmin( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp fast oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast oge float [[MIN]], 1.000000e+00 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MIN]], float 1.000000e+00 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast ult float %x, 255.0 + %min = select i1 %cmp2, float %x, float 255.0 + %cmp1 = fcmp fast ule float %x, 1.0 + %r = select i1 %cmp1, float 1.0, float %min + ret float %r +} + +; (X > C1) ? C1 : MAX(X, C2) +define float @clamp_float_fast_unordered_strict_minmax(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_unordered_strict_minmax( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp fast ole float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2_INV]], float 1.000000e+00, float [[X]] +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast ole float [[MAX]], 2.550000e+02 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MAX]], float 2.550000e+02 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast ugt float %x, 1.0 + %max = select i1 %cmp2, float %x, float 1.0 + %cmp1 = fcmp fast ugt float %x, 255.0 + %r = select i1 %cmp1, float 255.0, float %max + ret float %r +} + +; (X >= C1) ? C1 : MAX(X, C2) +define float @clamp_float_fast_unordered_nonstrict_minmax(float %x) { +; +; CHECK-LABEL: @clamp_float_fast_unordered_nonstrict_minmax( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp fast ole float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2_INV]], float 1.000000e+00, float [[X]] +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast ole float [[MAX]], 2.550000e+02 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[MAX]], float 2.550000e+02 +; CHECK-NEXT: ret float [[R1]] +; + %cmp2 = fcmp fast ugt float %x, 1.0 + %max = select i1 %cmp2, float %x, float 1.0 + %cmp1 = fcmp fast uge float %x, 255.0 + %r = select i1 %cmp1, float 255.0, float %max + ret float %r +} + +; Some more checks with fast + +; (X > 1.0) ? min(x, 255.0) : 1.0 +; That dit not match because select was in inversed order +define float @clamp_test_1(float %x) { +; CHECK-LABEL: @clamp_test_1( +; CHECK-NEXT: [[INNER_CMP_INV:%.*]] = fcmp fast oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[INNER_SEL:%.*]] = select i1 [[INNER_CMP_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[DOTINV:%.*]] = fcmp fast oge float [[INNER_SEL]], 1.000000e+00 +; CHECK-NEXT: [[R1:%.*]] = select i1 [[DOTINV]], float [[INNER_SEL]], float 1.000000e+00 +; CHECK-NEXT: ret float [[R1]] +; + %inner_cmp = fcmp fast ult float %x, 255.0 + %inner_sel = select i1 %inner_cmp, float %x, float 255.0 + %outer_cmp = fcmp fast ugt float %x, 1.0 + %r = select i1 %outer_cmp, float %inner_sel, float 1.0 + ret float %r +} + +; And something negative + +; Like @clamp_test_1 but HighConst < LowConst +define float @clamp_negative_wrong_const(float %x) { +; CHECK-LABEL: @clamp_negative_wrong_const( +; CHECK-NEXT: [[INNER_CMP_INV:%.*]] = fcmp fast oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[INNER_SEL:%.*]] = select i1 [[INNER_CMP_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[OUTER_CMP:%.*]] = fcmp fast ugt float [[X]], 5.120000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[OUTER_CMP]], float [[INNER_SEL]], float 5.120000e+02 +; CHECK-NEXT: ret float [[R]] +; + %inner_cmp = fcmp fast ult float %x, 255.0 + %inner_sel = select i1 %inner_cmp, float %x, float 255.0 + %outer_cmp = fcmp fast ugt float %x, 512.0 + %r = select i1 %outer_cmp, float %inner_sel, float 512.0 + ret float %r +} + +; Like @clamp_test_1 but both are min +define float @clamp_negative_same_op(float %x) { +; CHECK-LABEL: @clamp_negative_same_op( +; CHECK-NEXT: [[INNER_CMP_INV:%.*]] = fcmp fast oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[INNER_SEL:%.*]] = select i1 [[INNER_CMP_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[OUTER_CMP:%.*]] = fcmp fast ult float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[OUTER_CMP]], float [[INNER_SEL]], float 1.000000e+00 +; CHECK-NEXT: ret float [[R]] +; + %inner_cmp = fcmp fast ult float %x, 255.0 + %inner_sel = select i1 %inner_cmp, float %x, float 255.0 + %outer_cmp = fcmp fast ult float %x, 1.0 + %r = select i1 %outer_cmp, float %inner_sel, float 1.0 + ret float %r +} + + +; And now without fast. + +; First, check that we don't do bad things in the presence of signed zeros +define float @clamp_float_with_zero1(float %x) { +; CHECK-LABEL: @clamp_float_with_zero1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp fast olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ole float [[X]], 0.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 0.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp fast olt float %x, 255.0 + %min = select i1 %cmp2, float %x, float 255.0 + %cmp1 = fcmp ole float %x, 0.0 + %r = select i1 %cmp1, float 0.0, float %min + ret float %r +} + +define float @clamp_float_with_zero2(float %x) { +; CHECK-LABEL: @clamp_float_with_zero2( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp fast olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 0.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 0.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp fast olt float %x, 255.0 + %min = select i1 %cmp2, float %x, float 255.0 + %cmp1 = fcmp olt float %x, 0.0 + %r = select i1 %cmp1, float 0.0, float %min + ret float %r +} + +; Also, here we care more about the ordering of the inner min/max, so +; two times more cases. +; TODO: that is not implemented yet, so these checks are for the +; future. This means that checks below can just check that +; "fcmp.*%x" happens twice for each label. + +; (X < C1) ? C1 : MIN(X, C2) +define float @clamp_float_ordered_strict_maxmin1(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_strict_maxmin1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp olt float %x, 255.0 ; X is NaN => false + %min = select i1 %cmp2, float %x, float 255.0 ; 255.0 + %cmp1 = fcmp olt float %x, 1.0 ; false + %r = select i1 %cmp1, float 1.0, float %min ; min (255.0) + ret float %r +} + +define float @clamp_float_ordered_strict_maxmin2(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_strict_maxmin2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ult float %x, 255.0 ; X is NaN => true + %min = select i1 %cmp2, float %x, float 255.0 ; NaN + %cmp1 = fcmp olt float %x, 1.0 ; false + %r = select i1 %cmp1, float 1.0, float %min ; min (NaN) + ret float %r +} + +; (X <= C1) ? C1 : MIN(X, C2) +define float @clamp_float_ordered_nonstrict_maxmin1(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_nonstrict_maxmin1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ole float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp olt float %x, 255.0 ; X is NaN => false + %min = select i1 %cmp2, float %x, float 255.0 ; 255.0 + %cmp1 = fcmp ole float %x, 1.0 ; false + %r = select i1 %cmp1, float 1.0, float %min ; min (255.0) + ret float %r +} + +define float @clamp_float_ordered_nonstrict_maxmin2(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_nonstrict_maxmin2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ole float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ult float %x, 255.0 ; x is NaN => true + %min = select i1 %cmp2, float %x, float 255.0 ; NaN + %cmp1 = fcmp ole float %x, 1.0 ; false + %r = select i1 %cmp1, float 1.0, float %min ; min (NaN) + ret float %r +} + +; (X > C1) ? C1 : MAX(X, C2) +define float @clamp_float_ordered_strict_minmax1(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_strict_minmax1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2]], float [[X]], float 1.000000e+00 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ogt float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ogt float %x, 1.0 ; x is NaN => false + %max = select i1 %cmp2, float %x, float 1.0 ; 1.0 + %cmp1 = fcmp ogt float %x, 255.0 ; false + %r = select i1 %cmp1, float 255.0, float %max ; max (1.0) + ret float %r +} + +define float @clamp_float_ordered_strict_minmax2(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_strict_minmax2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp ole float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2_INV]], float 1.000000e+00, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ogt float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ugt float %x, 1.0 ; x is NaN => true + %max = select i1 %cmp2, float %x, float 1.0 ; NaN + %cmp1 = fcmp ogt float %x, 255.0 ; false + %r = select i1 %cmp1, float 255.0, float %max ; max (NaN) + ret float %r +} + +; (X >= C1) ? C1 : MAX(X, C2) +define float @clamp_float_ordered_nonstrict_minmax1(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_nonstrict_minmax1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2]], float [[X]], float 1.000000e+00 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp oge float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ogt float %x, 1.0 ; x is NaN => false + %max = select i1 %cmp2, float %x, float 1.0 ; 1.0 + %cmp1 = fcmp oge float %x, 255.0 ; false + %r = select i1 %cmp1, float 255.0, float %max ; max (1.0) + ret float %r +} + +define float @clamp_float_ordered_nonstrict_minmax2(float %x) { +; +; CHECK-LABEL: @clamp_float_ordered_nonstrict_minmax2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp ole float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2_INV]], float 1.000000e+00, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp oge float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ugt float %x, 1.0 ; x is NaN => true + %max = select i1 %cmp2, float %x, float 1.0 ; NaN + %cmp1 = fcmp oge float %x, 255.0 ; false + %r = select i1 %cmp1, float 255.0, float %max ; max (NaN) + ret float %r +} + + +; The same for unordered + +; (X < C1) ? C1 : MIN(X, C2) +define float @clamp_float_unordered_strict_maxmin1(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_strict_maxmin1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ult float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp olt float %x, 255.0 ; x is NaN => false + %min = select i1 %cmp2, float %x, float 255.0 ; 255.0 + %cmp1 = fcmp ult float %x, 1.0 ; true + %r = select i1 %cmp1, float 1.0, float %min ; 1.0 + ret float %r +} + +define float @clamp_float_unordered_strict_maxmin2(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_strict_maxmin2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ult float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ult float %x, 255.0 ; x is NaN => true + %min = select i1 %cmp2, float %x, float 255.0 ; NaN + %cmp1 = fcmp ult float %x, 1.0 ; true + %r = select i1 %cmp1, float 1.0, float %min ; 1.0 + ret float %r +} + +; (X <= C1) ? C1 : MIN(X, C2) +define float @clamp_float_unordered_nonstrict_maxmin1(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_nonstrict_maxmin1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp olt float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2]], float [[X]], float 2.550000e+02 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ule float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp olt float %x, 255.0 ; x is NaN => false + %min = select i1 %cmp2, float %x, float 255.0 ; 255.0 + %cmp1 = fcmp ule float %x, 1.0 ; true + %r = select i1 %cmp1, float 1.0, float %min ; 1.0 + ret float %r +} + +define float @clamp_float_unordered_nonstrict_maxmin2(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_nonstrict_maxmin2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp oge float [[X:%.*]], 2.550000e+02 +; CHECK-NEXT: [[MIN:%.*]] = select i1 [[CMP2_INV]], float 2.550000e+02, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ule float [[X]], 1.000000e+00 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 1.000000e+00, float [[MIN]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ult float %x, 255.0 ; x is NaN => true + %min = select i1 %cmp2, float %x, float 255.0 ; NaN + %cmp1 = fcmp ule float %x, 1.0 ; true + %r = select i1 %cmp1, float 1.0, float %min ; 1.0 + ret float %r +} + +; (X > C1) ? C1 : MAX(X, C2) +define float @clamp_float_unordered_strict_minmax1(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_strict_minmax1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2]], float [[X]], float 1.000000e+00 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ugt float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ogt float %x, 1.0 ; x is NaN => false + %max = select i1 %cmp2, float %x, float 1.0 ; 1.0 + %cmp1 = fcmp ugt float %x, 255.0 ; true + %r = select i1 %cmp1, float 255.0, float %max ; 255.0 + ret float %r +} + +define float @clamp_float_unordered_strict_minmax2(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_strict_minmax2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp ole float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2_INV]], float 1.000000e+00, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp ugt float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ugt float %x, 1.0 ; x is NaN => true + %max = select i1 %cmp2, float %x, float 1.0 ; NaN + %cmp1 = fcmp ugt float %x, 255.0 ; true + %r = select i1 %cmp1, float 255.0, float %max ; 255.0 + ret float %r +} + +; (X >= C1) ? C1 : MAX(X, C2) +define float @clamp_float_unordered_nonstrict_minmax1(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_nonstrict_minmax1( +; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2]], float [[X]], float 1.000000e+00 +; CHECK-NEXT: [[CMP1:%.*]] = fcmp uge float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ogt float %x, 1.0 ; x is NaN => false + %max = select i1 %cmp2, float %x, float 1.0 ; 1.0 + %cmp1 = fcmp uge float %x, 255.0 ; true + %r = select i1 %cmp1, float 255.0, float %max ; 255.0 + ret float %r +} + +define float @clamp_float_unordered_nonstrict_minmax2(float %x) { +; +; CHECK-LABEL: @clamp_float_unordered_nonstrict_minmax2( +; CHECK-NEXT: [[CMP2_INV:%.*]] = fcmp ole float [[X:%.*]], 1.000000e+00 +; CHECK-NEXT: [[MAX:%.*]] = select i1 [[CMP2_INV]], float 1.000000e+00, float [[X]] +; CHECK-NEXT: [[CMP1:%.*]] = fcmp uge float [[X]], 2.550000e+02 +; CHECK-NEXT: [[R:%.*]] = select i1 [[CMP1]], float 2.550000e+02, float [[MAX]] +; CHECK-NEXT: ret float [[R]] +; + %cmp2 = fcmp ugt float %x, 1.0 ; x is NaN => true + %max = select i1 %cmp2, float %x, float 1.0 ; NaN + %cmp1 = fcmp uge float %x, 255.0 ; true + %r = select i1 %cmp1, float 255.0, float %max ; 255.0 + ret float %r +}