[InstCombine] Fold Select with binary op - FP opcodes

Summary:
Follow up for https://reviews.llvm.org/rL339520 and https://reviews.llvm.org/rL338300

Alive:

```
%A = fcmp oeq float %x, 0.0
%B = fadd nsz float %x, %z
%C = select i1 %A, float %B, float %y
=>
%C = select i1 %A, float %z, float %y
----------                                                                      
  %A = fcmp oeq float %x, 0.0
  %B = fadd nsz float %x, %z
  %C = select %A, float %B, float %y
=>
  %C = select %A, float %z, float %y

Done: 1                                                                         
Optimization is correct

%A = fcmp une float %x, -0.0
%B = fadd nsz float %x, %z
%C = select i1 %A, float %y, float %B
=>
%C = select i1 %A, float %y, float %z
----------                                                                      
  %A = fcmp une float %x, -0.0
  %B = fadd nsz float %x, %z
  %C = select %A, float %y, float %B
=>
  %C = select %A, float %y, float %z

Done: 1                                                                         
Optimization is correct
```


Reviewers: spatel, lebedev.ri

Reviewed By: spatel

Subscribers: llvm-commits

Differential Revision: https://reviews.llvm.org/D50714

llvm-svn: 340538
This commit is contained in:
David Bolvansky 2018-08-23 15:22:15 +00:00
parent 3ecabd709f
commit 43b0e25847
2 changed files with 304 additions and 41 deletions

View File

@ -56,19 +56,26 @@ static Value *createMinMax(InstCombiner::BuilderTy &Builder,
/// Replace a select operand based on an equality comparison with the identity /// Replace a select operand based on an equality comparison with the identity
/// constant of a binop. /// constant of a binop.
static Instruction *foldSelectBinOpIdentity(SelectInst &Sel) { static Instruction *foldSelectBinOpIdentity(SelectInst &Sel, const TargetLibraryInfo &TLI) {
// The select condition must be an equality compare with a constant operand. // The select condition must be an equality compare with a constant operand.
// TODO: Support FP compares.
Value *X; Value *X;
Constant *C; Constant *C;
CmpInst::Predicate Pred; CmpInst::Predicate Pred;
if (!match(Sel.getCondition(), m_ICmp(Pred, m_Value(X), m_Constant(C))) || if (!match(Sel.getCondition(), m_Cmp(Pred, m_Value(X), m_Constant(C))))
!ICmpInst::isEquality(Pred)) return nullptr;
bool IsEq;
if (ICmpInst::isEquality(Pred))
IsEq = Pred == ICmpInst::ICMP_EQ;
else if (Pred == FCmpInst::FCMP_OEQ)
IsEq = true;
else if (Pred == FCmpInst::FCMP_UNE)
IsEq = false;
else
return nullptr; return nullptr;
// A select operand must be a binop, and the compare constant must be the // A select operand must be a binop, and the compare constant must be the
// identity constant for that binop. // identity constant for that binop.
bool IsEq = Pred == ICmpInst::ICMP_EQ;
BinaryOperator *BO; BinaryOperator *BO;
if (!match(Sel.getOperand(IsEq ? 1 : 2), m_BinOp(BO)) || if (!match(Sel.getOperand(IsEq ? 1 : 2), m_BinOp(BO)) ||
ConstantExpr::getBinOpIdentity(BO->getOpcode(), BO->getType(), true) != C) ConstantExpr::getBinOpIdentity(BO->getOpcode(), BO->getType(), true) != C)
@ -81,6 +88,12 @@ static Instruction *foldSelectBinOpIdentity(SelectInst &Sel) {
if (!match(BO, m_c_BinOp(m_Value(Y), m_Specific(X)))) if (!match(BO, m_c_BinOp(m_Value(Y), m_Specific(X))))
return nullptr; return nullptr;
// +0.0 compares equal to -0.0, and so it does not behave as required for this
// transform. Bail out if we can not exclude that possibility.
if (isa<FPMathOperator>(BO))
if (!BO->hasNoSignedZeros() && !CannotBeNegativeZero(Y, &TLI))
return nullptr;
// BO = binop Y, X // BO = binop Y, X
// S = { select (cmp eq X, C), BO, ? } or { select (cmp ne X, C), ?, BO } // S = { select (cmp eq X, C), BO, ? } or { select (cmp ne X, C), ?, BO }
// => // =>
@ -2000,7 +2013,7 @@ Instruction *InstCombiner::visitSelectInst(SelectInst &SI) {
if (Instruction *Select = foldSelectCmpXchg(SI)) if (Instruction *Select = foldSelectCmpXchg(SI))
return Select; return Select;
if (Instruction *Select = foldSelectBinOpIdentity(SI)) if (Instruction *Select = foldSelectBinOpIdentity(SI, TLI))
return Select; return Select;
return nullptr; return nullptr;

View File

@ -143,38 +143,37 @@ define i32 @select_xor_inv_icmp2(i32 %x, i32 %y, i32 %z) {
define float @select_fadd_fcmp(float %x, float %y, float %z) { define float @select_fadd_fcmp(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp( ; CHECK-LABEL: @select_fadd_fcmp(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Z:%.*]], float [[Y:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, -0.0 %A = fcmp oeq float %x, -0.0
%B = fadd float %x, %z %B = fadd nsz float %x, %z
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
define float @select_fadd_fcmp_2(float %x, float %y, float %z) { define float @select_fadd_fcmp_2(float %x, float %y, float %v) {
; CHECK-LABEL: @select_fadd_fcmp_2( ; CHECK-LABEL: @select_fadd_fcmp_2(
; CHECK-NEXT: [[A:%.*]] = fcmp ueq float [[X:%.*]], -0.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[Z:%.*]] = fadd float [[V:%.*]], 0.000000e+00
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[Z]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp ueq float %x, -0.0 %A = fcmp une float %x, -0.0
%B = fadd float %x, %z %z = fadd float %v, 0.0 ; cannot produce -0.0
%C = select i1 %A, float %B, float %y %B = fadd float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C ret float %C
} }
define float @select_fadd_fcmp_3(float %x, float %y, float %z) { define float @select_fadd_fcmp_3(float %x, float %y) {
; CHECK-LABEL: @select_fadd_fcmp_3( ; CHECK-LABEL: @select_fadd_fcmp_3(
; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], -0.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[Z:%.*]], [[X]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float 6.000000e+00
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp one float %x, -0.0 %A = fcmp une float %x, -0.0
%B = fadd float %z, %x %B = fadd float 6.0, %x
%C = select i1 %A, float %y, float %B %C = select i1 %A, float %y, float %B
ret float %C ret float %C
} }
@ -182,25 +181,49 @@ define float @select_fadd_fcmp_3(float %x, float %y, float %z) {
define float @select_fadd_fcmp_4(float %x, float %y, float %z) { define float @select_fadd_fcmp_4(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_4( ; CHECK-LABEL: @select_fadd_fcmp_4(
; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[Z:%.*]], [[X]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp une float %x, -0.0 %A = fcmp une float %x, -0.0
%B = fadd float %z, %x %B = fadd nsz float %z, %x
%C = select i1 %A, float %y, float %B %C = select i1 %A, float %y, float %B
ret float %C ret float %C
} }
define float @select_fadd_fcmp_5(float %x, float %y, float %v) {
; CHECK-LABEL: @select_fadd_fcmp_5(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[Z:%.*]] = fadd float [[V:%.*]], 0.000000e+00
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Z]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, -0.0
%z = fadd float %v, 0.0 ; cannot produce -0.0
%B = fadd float %z, %x
%C = select i1 %A, float %B, float %y
ret float %C
}
define float @select_fadd_fcmp_6(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_6(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float 6.000000e+00, float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, -0.0
%B = fadd float %x, 6.0
%C = select i1 %A, float %B, float %y
ret float %C
}
define float @select_fmul_fcmp(float %x, float %y, float %z) { define float @select_fmul_fcmp(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fmul_fcmp( ; CHECK-LABEL: @select_fmul_fcmp(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fmul float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Z:%.*]], float [[Y:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, 1.0 %A = fcmp oeq float %x, 1.0
%B = fmul float %x, %z %B = fmul nsz float %x, %z
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
@ -208,12 +231,11 @@ define float @select_fmul_fcmp(float %x, float %y, float %z) {
define float @select_fsub_fcmp(float %x, float %y, float %z) { define float @select_fsub_fcmp(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fsub_fcmp( ; CHECK-LABEL: @select_fsub_fcmp(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 0.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fsub float [[Z:%.*]], [[X]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Z:%.*]], float [[Y:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, 0.0 %A = fcmp oeq float %x, 0.0
%B = fsub float %z, %x %B = fsub nsz float %z, %x
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
@ -221,12 +243,11 @@ define float @select_fsub_fcmp(float %x, float %y, float %z) {
define float @select_fdiv_fcmp(float %x, float %y, float %z) { define float @select_fdiv_fcmp(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fdiv_fcmp( ; CHECK-LABEL: @select_fdiv_fcmp(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fdiv float [[Z:%.*]], [[X]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Z:%.*]], float [[Y:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, 1.0 %A = fcmp oeq float %x, 1.0
%B = fdiv float %z, %x %B = fdiv nsz float %z, %x
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
@ -507,19 +528,21 @@ define i32 @select_or_icmp_bad(i32 %x, i32 %y, i32 %z) {
ret i32 %C ret i32 %C
} }
; Invalid identity constant for FP op
define float @select_fadd_fcmp_bad(float %x, float %y, float %z) { define float @select_fadd_fcmp_bad(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad( ; CHECK-LABEL: @select_fadd_fcmp_bad(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -1.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, -1.0 %A = fcmp oeq float %x, -1.0
%B = fadd float %x, %z %B = fadd nsz float %x, %z
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
; Invalid comparison type
define float @select_fadd_fcmp_bad_2(float %x, float %y, float %z) { define float @select_fadd_fcmp_bad_2(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_2( ; CHECK-LABEL: @select_fadd_fcmp_bad_2(
; CHECK-NEXT: [[A:%.*]] = fcmp ueq float [[X:%.*]], -1.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp ueq float [[X:%.*]], -1.000000e+00
@ -533,6 +556,7 @@ define float @select_fadd_fcmp_bad_2(float %x, float %y, float %z) {
ret float %C ret float %C
} }
; Invalid comparison type
define float @select_fadd_fcmp_bad_3(float %x, float %y, float %z, float %k) { define float @select_fadd_fcmp_bad_3(float %x, float %y, float %z, float %k) {
; CHECK-LABEL: @select_fadd_fcmp_bad_3( ; CHECK-LABEL: @select_fadd_fcmp_bad_3(
; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], [[K:%.*]] ; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], [[K:%.*]]
@ -546,6 +570,7 @@ define float @select_fadd_fcmp_bad_3(float %x, float %y, float %z, float %k) {
ret float %C ret float %C
} }
; Invalid order of operands of select
define float @select_fadd_fcmp_bad_4(float %x, float %y, float %z) { define float @select_fadd_fcmp_bad_4(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_4( ; CHECK-LABEL: @select_fadd_fcmp_bad_4(
; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00
@ -559,14 +584,200 @@ define float @select_fadd_fcmp_bad_4(float %x, float %y, float %z) {
ret float %C ret float %C
} }
; Invalid comparison type
define float @select_fadd_fcmp_bad_5(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_5(
; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[Z:%.*]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp one float %x, -0.0
%B = fadd nsz float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid order of operands of select
define float @select_fadd_fcmp_bad_6(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_6(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[Z:%.*]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, -0.0
%B = fadd nsz float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Do not transform if we have signed zeros and if Z is possibly negative zero
define float @select_fadd_fcmp_bad_7(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_7(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, -0.0
%B = fadd float %x, %z
%C = select i1 %A, float %B, float %y
ret float %C
}
; Invalid comparison type
define float @select_fadd_fcmp_bad_8(float %x, float %y, float %v) {
; CHECK-LABEL: @select_fadd_fcmp_bad_8(
; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[Z:%.*]] = fadd float [[V:%.*]], -1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[Z]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp one float %x, -0.0
%z = fadd float %v, -1.0
%B = fadd float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid comparison type
define float @select_fadd_fcmp_bad_9(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_9(
; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[Z:%.*]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp one float %x, -0.0
%B = fadd nsz float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid comparison type
define float @select_fadd_fcmp_bad_10(float %x, float %y, float %v) {
; CHECK-LABEL: @select_fadd_fcmp_bad_10(
; CHECK-NEXT: [[A:%.*]] = fcmp one float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[Z:%.*]] = fadd float [[V:%.*]], 0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[Z]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp one float %x, -0.0
%z = fadd float %v, 0.0 ; cannot produce -0.0
%B = fadd float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Do not transform if Z is possibly negative zero
define float @select_fadd_fcmp_bad_11(float %x, float %y, float %v) {
; CHECK-LABEL: @select_fadd_fcmp_bad_11(
; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[Z:%.*]] = fadd float [[V:%.*]], -1.000000e+00
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[Z]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp une float %x, -0.0
%z = fadd float %v, -1.0
%B = fadd nsz float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Do not transform if we have signed zeros and if Z is possibly negative zero
define float @select_fadd_fcmp_bad_12(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_12(
; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd float [[Z:%.*]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp une float %x, -0.0
%B = fadd float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid order of operands of select
define float @select_fadd_fcmp_bad_13(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_13(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], -0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, -0.0
%B = fadd nsz float %x, %z
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid identity constant for FP op
define float @select_fadd_fcmp_bad_14(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_14(
; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], -1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp une float %x, -1.0
%B = fadd nsz float %x, %z
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid identity constant for FP op
define float @select_fadd_fcmp_bad_15(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_15(
; CHECK-NEXT: [[A:%.*]] = fcmp une float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[Z:%.*]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[Y:%.*]], float [[B]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp une float %x, 0.0
%B = fadd nsz float %z, %x
%C = select i1 %A, float %y, float %B
ret float %C
}
; Invalid identity constant for FP op
define float @select_fadd_fcmp_bad_16(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fadd_fcmp_bad_16(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fadd nsz float [[Z:%.*]], [[X]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, 0.0
%B = fadd nsz float %z, %x
%C = select i1 %A, float %B, float %y
ret float %C
}
define float @select_fmul_fcmp_bad(float %x, float %y, float %z) { define float @select_fmul_fcmp_bad(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fmul_fcmp_bad( ; CHECK-LABEL: @select_fmul_fcmp_bad(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 3.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 3.000000e+00
; CHECK-NEXT: [[B:%.*]] = fmul float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[B:%.*]] = fmul nsz float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, 3.0 %A = fcmp oeq float %x, 3.0
%B = fmul nsz float %x, %z
%C = select i1 %A, float %B, float %y
ret float %C
}
define float @select_fmul_fcmp_bad_2(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fmul_fcmp_bad_2(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fmul float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, 1.0
%B = fmul float %x, %z %B = fmul float %x, %z
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
@ -585,32 +796,71 @@ define float @select_fmul_icmp_bad(float %x, float %y, float %z, i32 %k) {
ret float %C ret float %C
} }
define float @select_fmul_icmp_bad_2(float %x, float %y, float %z, i32 %k) {
; CHECK-LABEL: @select_fmul_icmp_bad_2(
; CHECK-NEXT: [[A:%.*]] = icmp eq i32 [[K:%.*]], 0
; CHECK-NEXT: [[B:%.*]] = fmul nsz float [[X:%.*]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = icmp eq i32 %k, 0
%B = fmul nsz float %x, %z
%C = select i1 %A, float %B, float %y
ret float %C
}
define float @select_fdiv_fcmp_bad(float %x, float %y, float %z) { define float @select_fdiv_fcmp_bad(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fdiv_fcmp_bad( ; CHECK-LABEL: @select_fdiv_fcmp_bad(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 3.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fdiv float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[B:%.*]] = fdiv float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, 3.0 %A = fcmp oeq float %x, 1.0
%B = fdiv float %x, %z %B = fdiv float %x, %z
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
define float @select_fdiv_fcmp_bad_2(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fdiv_fcmp_bad_2(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 3.000000e+00
; CHECK-NEXT: [[B:%.*]] = fdiv nsz float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, 3.0
%B = fdiv nsz float %x, %z
%C = select i1 %A, float %B, float %y
ret float %C
}
define float @select_fsub_fcmp_bad(float %x, float %y, float %z) { define float @select_fsub_fcmp_bad(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fsub_fcmp_bad( ; CHECK-LABEL: @select_fsub_fcmp_bad(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00 ; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: [[B:%.*]] = fsub float [[X]], [[Z:%.*]] ; CHECK-NEXT: [[B:%.*]] = fsub float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]] ; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]] ; CHECK-NEXT: ret float [[C]]
; ;
%A = fcmp oeq float %x, 1.0 %A = fcmp oeq float %x, 0.0
%B = fsub float %x, %z %B = fsub float %x, %z
%C = select i1 %A, float %B, float %y %C = select i1 %A, float %B, float %y
ret float %C ret float %C
} }
define float @select_fsub_fcmp_bad_2(float %x, float %y, float %z) {
; CHECK-LABEL: @select_fsub_fcmp_bad_2(
; CHECK-NEXT: [[A:%.*]] = fcmp oeq float [[X:%.*]], 1.000000e+00
; CHECK-NEXT: [[B:%.*]] = fsub nsz float [[X]], [[Z:%.*]]
; CHECK-NEXT: [[C:%.*]] = select i1 [[A]], float [[B]], float [[Y:%.*]]
; CHECK-NEXT: ret float [[C]]
;
%A = fcmp oeq float %x, 1.0
%B = fsub nsz float %x, %z
%C = select i1 %A, float %B, float %y
ret float %C
}
define i32 @select_sub_icmp_bad(i32 %x, i32 %y, i32 %z) { define i32 @select_sub_icmp_bad(i32 %x, i32 %y, i32 %z) {
; CHECK-LABEL: @select_sub_icmp_bad( ; CHECK-LABEL: @select_sub_icmp_bad(
; CHECK-NEXT: [[A:%.*]] = icmp eq i32 [[X:%.*]], 0 ; CHECK-NEXT: [[A:%.*]] = icmp eq i32 [[X:%.*]], 0