; RUN: opt -correlated-propagation -S < %s | FileCheck %s declare i32 @foo() define i32 @test1(i32 %a) nounwind { %a.off = add i32 %a, -8 %cmp = icmp ult i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, 7 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 ; CHECK-LABEL: @test1( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else } define i32 @test2(i32 %a) nounwind { %a.off = add i32 %a, -8 %cmp = icmp ult i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead = icmp ugt i32 %a, 15 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 ; CHECK-LABEL: @test2( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else } ; CHECK-LABEL: @test3( define i32 @test3(i32 %c) nounwind { %cmp = icmp slt i32 %c, 2 br i1 %cmp, label %if.then, label %if.end if.then: ret i32 1 if.end: %cmp1 = icmp slt i32 %c, 3 br i1 %cmp1, label %if.then2, label %if.end8 ; CHECK: if.then2 if.then2: %cmp2 = icmp eq i32 %c, 2 ; CHECK: br i1 true br i1 %cmp2, label %if.then4, label %if.end6 ; CHECK: if.end6 if.end6: ret i32 2 if.then4: ret i32 3 if.end8: ret i32 4 } ; CHECK-LABEL: @test4( define i32 @test4(i32 %c) nounwind { switch i32 %c, label %sw.default [ i32 1, label %sw.bb i32 2, label %sw.bb i32 4, label %sw.bb ] ; CHECK: sw.bb sw.bb: %cmp = icmp sge i32 %c, 1 ; CHECK: br i1 true br i1 %cmp, label %if.then, label %if.end if.then: br label %return if.end: br label %return sw.default: br label %return return: %retval.0 = phi i32 [ 42, %sw.default ], [ 4, %if.then ], [ 9, %if.end ] ret i32 %retval.0 } ; CHECK-LABEL: @test5( define i1 @test5(i32 %c) nounwind { %cmp = icmp slt i32 %c, 5 br i1 %cmp, label %if.then, label %if.end if.then: %cmp1 = icmp eq i32 %c, 4 br i1 %cmp1, label %if.end, label %if.end8 if.end: ret i1 true if.end8: %cmp2 = icmp eq i32 %c, 3 %cmp3 = icmp eq i32 %c, 4 %cmp4 = icmp eq i32 %c, 6 ; CHECK: %or = or i1 false, false %or = or i1 %cmp3, %cmp4 ; CHECK: ret i1 %cmp2 ret i1 %cmp2 } ; CHECK-LABEL: @test6( define i1 @test6(i32 %c) nounwind { %cmp = icmp ule i32 %c, 7 br i1 %cmp, label %if.then, label %if.end if.then: ; CHECK: icmp eq i32 %c, 6 ; CHECK: br i1 switch i32 %c, label %if.end [ i32 6, label %sw.bb i32 8, label %sw.bb ] if.end: ret i1 true sw.bb: %cmp2 = icmp eq i32 %c, 6 ; CHECK: ret i1 true ret i1 %cmp2 } ; CHECK-LABEL: @test7( define i1 @test7(i32 %c) nounwind { entry: switch i32 %c, label %sw.default [ i32 6, label %sw.bb i32 7, label %sw.bb ] sw.bb: ret i1 true sw.default: %cmp5 = icmp eq i32 %c, 5 %cmp6 = icmp eq i32 %c, 6 %cmp7 = icmp eq i32 %c, 7 %cmp8 = icmp eq i32 %c, 8 ; CHECK: %or = or i1 %cmp5, false %or = or i1 %cmp5, %cmp6 ; CHECK: %or2 = or i1 false, %cmp8 %or2 = or i1 %cmp7, %cmp8 ret i1 false } define i1 @test8(i64* %p) { ; CHECK-LABEL: @test8 ; CHECK: ret i1 false %a = load i64, i64* %p, !range !{i64 4, i64 255} %res = icmp eq i64 %a, 0 ret i1 %res } define i1 @test9(i64* %p) { ; CHECK-LABEL: @test9 ; CHECK: ret i1 true %a = load i64, i64* %p, !range !{i64 0, i64 1} %res = icmp eq i64 %a, 0 ret i1 %res } define i1 @test10(i64* %p) { ; CHECK-LABEL: @test10 ; CHECK: ret i1 false %a = load i64, i64* %p, !range !{i64 4, i64 8, i64 15, i64 20} %res = icmp eq i64 %a, 0 ret i1 %res } @g = external global i32 define i1 @test11() { ; CHECK: @test11 ; CHECK: ret i1 true %positive = load i32, i32* @g, !range !{i32 1, i32 2048} %add = add i32 %positive, 1 %test = icmp sgt i32 %add, 0 br label %next next: ret i1 %test } define i32 @test12(i32 %a, i32 %b) { ; CHECK-LABEL: @test12( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else %cmp = icmp ult i32 %a, %b br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, -1 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 } define i32 @test12_swap(i32 %a, i32 %b) { ; CHECK-LABEL: @test12_swap( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else %cmp = icmp ugt i32 %b, %a br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, -1 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 } define i32 @test12_neg(i32 %a, i32 %b) { ; The same as @test12 but the second check is on the false path ; CHECK-LABEL: @test12_neg( ; CHECK: else: ; CHECK-NEXT: %alive = icmp eq i32 %a, -1 %cmp = icmp ult i32 %a, %b br i1 %cmp, label %then, label %else else: %alive = icmp eq i32 %a, -1 br i1 %alive, label %end, label %then then: ret i32 1 end: ret i32 2 } define i32 @test12_signed(i32 %a, i32 %b) { ; The same as @test12 but with signed comparison ; CHECK-LABEL: @test12_signed( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else %cmp = icmp slt i32 %a, %b br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, 2147483647 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 } define i32 @test13(i32 %a, i32 %b) { ; CHECK-LABEL: @test13( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else %a.off = add i32 %a, -8 %cmp = icmp ult i32 %a.off, %b br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, 7 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 } define i32 @test13_swap(i32 %a, i32 %b) { ; CHECK-LABEL: @test13_swap( ; CHECK: then: ; CHECK-NEXT: br i1 false, label %end, label %else %a.off = add i32 %a, -8 %cmp = icmp ugt i32 %b, %a.off br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, 7 br i1 %dead, label %end, label %else else: ret i32 1 end: ret i32 2 } define i1 @test14_slt(i32 %a) { ; CHECK-LABEL: @test14_slt( ; CHECK: then: ; CHECK-NEXT: %result = or i1 false, false %a.off = add i32 %a, -8 %cmp = icmp slt i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead.1 = icmp eq i32 %a, -2147483641 %dead.2 = icmp eq i32 %a, 16 %result = or i1 %dead.1, %dead.2 ret i1 %result else: ret i1 false } define i1 @test14_sle(i32 %a) { ; CHECK-LABEL: @test14_sle( ; CHECK: then: ; CHECK-NEXT: %alive = icmp eq i32 %a, 16 ; CHECK-NEXT: %result = or i1 false, %alive %a.off = add i32 %a, -8 %cmp = icmp sle i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, -2147483641 %alive = icmp eq i32 %a, 16 %result = or i1 %dead, %alive ret i1 %result else: ret i1 false } define i1 @test14_sgt(i32 %a) { ; CHECK-LABEL: @test14_sgt( ; CHECK: then: ; CHECK-NEXT: %result = or i1 false, false %a.off = add i32 %a, -8 %cmp = icmp sgt i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead.1 = icmp eq i32 %a, -2147483640 %dead.2 = icmp eq i32 %a, 16 %result = or i1 %dead.1, %dead.2 ret i1 %result else: ret i1 false } define i1 @test14_sge(i32 %a) { ; CHECK-LABEL: @test14_sge( ; CHECK: then: ; CHECK-NEXT: %alive = icmp eq i32 %a, 16 ; CHECK-NEXT: %result = or i1 false, %alive %a.off = add i32 %a, -8 %cmp = icmp sge i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, -2147483640 %alive = icmp eq i32 %a, 16 %result = or i1 %dead, %alive ret i1 %result else: ret i1 false } define i1 @test14_ule(i32 %a) { ; CHECK-LABEL: @test14_ule( ; CHECK: then: ; CHECK-NEXT: %alive = icmp eq i32 %a, 16 ; CHECK-NEXT: %result = or i1 false, %alive %a.off = add i32 %a, -8 %cmp = icmp ule i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, 7 %alive = icmp eq i32 %a, 16 %result = or i1 %dead, %alive ret i1 %result else: ret i1 false } define i1 @test14_ugt(i32 %a) { ; CHECK-LABEL: @test14_ugt( ; CHECK: then: ; CHECK-NEXT: %result = or i1 false, false %a.off = add i32 %a, -8 %cmp = icmp ugt i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead.1 = icmp eq i32 %a, 8 %dead.2 = icmp eq i32 %a, 16 %result = or i1 %dead.1, %dead.2 ret i1 %result else: ret i1 false } define i1 @test14_uge(i32 %a) { ; CHECK-LABEL: @test14_uge( ; CHECK: then: ; CHECK-NEXT: %alive = icmp eq i32 %a, 16 ; CHECK-NEXT: %result = or i1 false, %alive %a.off = add i32 %a, -8 %cmp = icmp uge i32 %a.off, 8 br i1 %cmp, label %then, label %else then: %dead = icmp eq i32 %a, 8 %alive = icmp eq i32 %a, 16 %result = or i1 %dead, %alive ret i1 %result else: ret i1 false } @limit = external global i32 define i1 @test15(i32 %a) { ; CHECK-LABEL: @test15( ; CHECK: then: ; CHECK-NEXT: ret i1 false %limit = load i32, i32* @limit, !range !{i32 0, i32 256} %cmp = icmp ult i32 %a, %limit br i1 %cmp, label %then, label %else then: %result = icmp eq i32 %a, 255 ret i1 %result else: ret i1 false } define i32 @test16(i8 %a) { entry: %b = zext i8 %a to i32 br label %dispatch dispatch: %cmp = icmp eq i8 %a, 93 br i1 %cmp, label %target93, label %dispatch ; CHECK-LABEL: @test16( ; CHECK: target93: ; CHECK-NEXT: ret i32 93 target93: ret i32 %b } define i32 @test16_i1(i1 %a) { entry: %b = zext i1 %a to i32 br label %dispatch dispatch: br i1 %a, label %true, label %dispatch ; CHECK-LABEL: @test16_i1( ; CHECK: true: ; CHECK-NEXT: ret i32 1 true: ret i32 %b } define i8 @test17(i8 %a) { entry: %c = add i8 %a, 3 br label %dispatch dispatch: %cmp = icmp eq i8 %a, 93 br i1 %cmp, label %target93, label %dispatch ; CHECK-LABEL: @test17( ; CHECK: target93: ; CHECK-NEXT: ret i8 96 target93: ret i8 %c } define i8 @test17_2(i8 %a) { entry: %c = add i8 %a, %a br label %dispatch dispatch: %cmp = icmp eq i8 %a, 93 br i1 %cmp, label %target93, label %dispatch ; CHECK-LABEL: @test17_2( ; CHECK: target93: ; CHECK-NEXT: ret i8 -70 target93: ret i8 %c } define i1 @test17_i1(i1 %a) { entry: %c = and i1 %a, true br label %dispatch dispatch: br i1 %a, label %true, label %dispatch ; CHECK-LABEL: @test17_i1( ; CHECK: true: ; CHECK-NEXT: ret i1 true true: ret i1 %c } define i32 @test18(i8 %a) { entry: %b = zext i8 %a to i32 br label %dispatch dispatch: switch i8 %a, label %dispatch [ i8 93, label %target93 i8 -111, label %dispatch ] ; CHECK-LABEL: @test18( ; CHECK: target93: ; CHECK-NEXT: ret i32 93 target93: ret i32 %b } define i8 @test19(i8 %a) { entry: %c = add i8 %a, 3 br label %dispatch dispatch: switch i8 %a, label %dispatch [ i8 93, label %target93 i8 -111, label %dispatch ] ; CHECK-LABEL: @test19( ; CHECK: target93: ; CHECK-NEXT: ret i8 96 target93: ret i8 %c } define i1 @test20(i64 %a) { entry: %b = and i64 %a, 7 br label %dispatch dispatch: switch i64 %a, label %default [ i64 0, label %exit2 i64 -2147483647, label %exit2 ] default: %c = icmp eq i64 %b, 0 br label %exit exit: ; Negative test. Shouldn't be incorrectly optimized to "ret i1 false". ; CHECK-LABEL: @test20( ; CHECK: exit: ; CHECK-NOT: ret i1 false ; CHECK: exit2: ret i1 %c exit2: ret i1 false }