| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 4 |
| ; RUN: opt -S -passes=instcombine %s | FileCheck %s |
| |
| declare void @use(i1) |
| |
| define i1 @test_and_olt(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_ole(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_ole( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ole float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ole float %x, 0x3C00000000000000 |
| %cmp2 = fcmp oge float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_ogt(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_ogt( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ogt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ogt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp olt float %x, 0xBC00000000000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_oge(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_oge( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp oge float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp oge float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ole float %x, 0xBC00000000000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_ult(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_ult( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ult float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ult float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ugt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_ule(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_ule( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ule float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ule float %x, 0x3C00000000000000 |
| %cmp2 = fcmp uge float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_ugt(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_ugt( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ugt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ugt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ult float %x, 0xBC00000000000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_uge(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_uge( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp uge float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp uge float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ule float %x, 0xBC00000000000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_commuted(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_commuted( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp2, %cmp1 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_subnormal(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_subnormal( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp olt float [[TMP1]], 0x36A0000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x36A0000000000000 |
| %cmp2 = fcmp ogt float %x, 0xB6A0000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_infinity(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_infinity( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp one float [[TMP1]], 0x7FF0000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x7FF0000000000000 |
| %cmp2 = fcmp ogt float %x, 0xFFF0000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_zero(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_zero( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 false |
| ; |
| %cmp1 = fcmp olt float %x, 0x0000000000000000 |
| %cmp2 = fcmp ogt float %x, 0x8000000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_ole_zero(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_ole_zero( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp oeq float [[X]], 0.000000e+00 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ole float %x, 0x0000000000000000 |
| %cmp2 = fcmp oge float %x, 0x8000000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_logical(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_logical( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = select i1 %cmp1, i1 %cmp2, i1 false |
| ret i1 %cond |
| } |
| |
| define <2 x i1> @test_and_olt_poison(<2 x float> %x) { |
| ; CHECK-LABEL: define <2 x i1> @test_and_olt_poison( |
| ; CHECK-SAME: <2 x float> [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call <2 x float> @llvm.fabs.v2f32(<2 x float> [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp olt <2 x float> [[TMP1]], splat (float 0x3C00000000000000) |
| ; CHECK-NEXT: ret <2 x i1> [[COND]] |
| ; |
| %cmp1 = fcmp olt <2 x float> %x, <float 0x3C00000000000000, float poison> |
| %cmp2 = fcmp ogt <2 x float> %x, <float 0xBC00000000000000, float poison> |
| %cond = and <2 x i1> %cmp1, %cmp2 |
| ret <2 x i1> %cond |
| } |
| |
| define i1 @test_and_olt_nan(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_nan( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 false |
| ; |
| %cmp1 = fcmp olt float %x, 0x7FF8000000000000 |
| %cmp2 = fcmp ogt float %x, 0xFFF8000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_ult_nan(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_ult_nan( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 true |
| ; |
| %cmp1 = fcmp ult float %x, 0x7FF0000020000000 |
| %cmp2 = fcmp ugt float %x, 0xFFF0000020000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_ogt_nan(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_ogt_nan( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 false |
| ; |
| %cmp1 = fcmp ogt float %x, 0x7FF0000020000000 |
| %cmp2 = fcmp olt float %x, 0xFFF0000020000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_ugt_nan(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_ugt_nan( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 true |
| ; |
| %cmp1 = fcmp ugt float %x, 0x7FF0000020000000 |
| %cmp2 = fcmp ult float %x, 0xFFF0000020000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_ogt(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_ogt( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 false |
| ; |
| %cmp1 = fcmp ogt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp olt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_or_olt(float %x) { |
| ; CHECK-LABEL: define i1 @test_or_olt( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ogt float [[TMP1]], 0xBC00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = or i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| ; Negative tests |
| |
| define i1 @test_and_olt_multiuse(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_multiuse( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 0x3C00000000000000 |
| ; CHECK-NEXT: call void @use(i1 [[CMP1]]) |
| ; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[X]], 0xBC00000000000000 |
| ; CHECK-NEXT: [[COND:%.*]] = and i1 [[CMP1]], [[CMP2]] |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| call void @use(i1 %cmp1) |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_mismatched_lhs(float %x, float %y) { |
| ; CHECK-LABEL: define i1 @test_and_olt_mismatched_lhs( |
| ; CHECK-SAME: float [[X:%.*]], float [[Y:%.*]]) { |
| ; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 0x3C00000000000000 |
| ; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[Y]], 0xBC00000000000000 |
| ; CHECK-NEXT: [[COND:%.*]] = and i1 [[CMP1]], [[CMP2]] |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %y, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_same_sign(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_same_sign( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: ret i1 false |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %x, 0x3C00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_mismatched_mag(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_mismatched_mag( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 0x3C80000000000000 |
| ; CHECK-NEXT: [[CMP2:%.*]] = fcmp ogt float [[X]], 0xBC00000000000000 |
| ; CHECK-NEXT: [[COND:%.*]] = and i1 [[CMP1]], [[CMP2]] |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C80000000000000 |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_wrong_pred2(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_wrong_pred2( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[CMP1:%.*]] = fcmp olt float [[X]], 0x3C00000000000000 |
| ; CHECK-NEXT: [[CMP2:%.*]] = fcmp oge float [[X]], 0xBC00000000000000 |
| ; CHECK-NEXT: [[COND:%.*]] = and i1 [[CMP1]], [[CMP2]] |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp oge float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_fmf_propagation(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_fmf_propagation( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call nnan ninf nsz float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp nnan ninf nsz olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp nsz nnan ninf olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp nsz nnan ninf ogt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_fmf_propagation_union(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_fmf_propagation_union( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call nnan ninf nsz float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp nnan ninf nsz olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp nnan ninf olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp nsz nnan ogt float %x, 0xBC00000000000000 |
| %cond = and i1 %cmp1, %cmp2 |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_fmf_propagation_union_logical_rhs_poison(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_fmf_propagation_union_logical_rhs_poison( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ninf olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ogt float %x, 0xBC00000000000000 |
| %cond = select i1 %cmp2, i1 %cmp1, i1 false |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_fmf_propagation_union_logical_lhs_poison(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_fmf_propagation_union_logical_lhs_poison( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call ninf float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ninf olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ninf ogt float %x, 0xBC00000000000000 |
| %cond = select i1 %cmp2, i1 %cmp1, i1 false |
| ret i1 %cond |
| } |
| |
| define i1 @test_and_olt_fmf_propagation_union_logical_both_poison(float %x) { |
| ; CHECK-LABEL: define i1 @test_and_olt_fmf_propagation_union_logical_both_poison( |
| ; CHECK-SAME: float [[X:%.*]]) { |
| ; CHECK-NEXT: [[TMP1:%.*]] = call ninf float @llvm.fabs.f32(float [[X]]) |
| ; CHECK-NEXT: [[COND:%.*]] = fcmp ninf olt float [[TMP1]], 0x3C00000000000000 |
| ; CHECK-NEXT: ret i1 [[COND]] |
| ; |
| %cmp1 = fcmp ninf olt float %x, 0x3C00000000000000 |
| %cmp2 = fcmp ninf ogt float %x, 0xBC00000000000000 |
| %cond = select i1 %cmp2, i1 %cmp1, i1 false |
| ret i1 %cond |
| } |