| ; RUN: opt -jump-threading -S < %s | FileCheck %s |
| |
| declare void @side_effect(i32) |
| |
| define void @test0(i32 %i, i32 %len) { |
| ; CHECK-LABEL: @test0( |
| entry: |
| call void @side_effect(i32 0) |
| %i.inc = add nuw i32 %i, 1 |
| %c0 = icmp ult i32 %i.inc, %len |
| br i1 %c0, label %left, label %right |
| |
| left: |
| ; CHECK: entry: |
| ; CHECK: br i1 %c0, label %left0, label %right |
| |
| ; CHECK: left0: |
| ; CHECK: call void @side_effect |
| ; CHECK-NOT: br i1 %c1 |
| ; CHECK: call void @side_effect |
| call void @side_effect(i32 0) |
| %c1 = icmp ult i32 %i, %len |
| br i1 %c1, label %left0, label %right |
| |
| left0: |
| call void @side_effect(i32 0) |
| ret void |
| |
| right: |
| %t = phi i32 [ 1, %left ], [ 2, %entry ] |
| call void @side_effect(i32 %t) |
| ret void |
| } |
| |
| define void @test1(i32 %i, i32 %len) { |
| ; CHECK-LABEL: @test1( |
| entry: |
| call void @side_effect(i32 0) |
| %i.inc = add nsw i32 %i, 1 |
| %c0 = icmp slt i32 %i.inc, %len |
| br i1 %c0, label %left, label %right |
| |
| left: |
| ; CHECK: entry: |
| ; CHECK: br i1 %c0, label %left0, label %right |
| |
| ; CHECK: left0: |
| ; CHECK: call void @side_effect |
| ; CHECK-NOT: br i1 %c1 |
| ; CHECK: call void @side_effect |
| call void @side_effect(i32 0) |
| %c1 = icmp slt i32 %i, %len |
| br i1 %c1, label %left0, label %right |
| |
| left0: |
| call void @side_effect(i32 0) |
| ret void |
| |
| right: |
| %t = phi i32 [ 1, %left ], [ 2, %entry ] |
| call void @side_effect(i32 %t) |
| ret void |
| } |
| |
| define void @test2(i32 %i, i32 %len, i1* %c.ptr) { |
| ; CHECK-LABEL: @test2( |
| |
| ; CHECK: entry: |
| ; CHECK: br i1 %c0, label %cont, label %right |
| ; CHECK: cont: |
| ; CHECK: br i1 %c, label %left0, label %right |
| ; CHECK: left0: |
| ; CHECK: call void @side_effect(i32 0) |
| ; CHECK: call void @side_effect(i32 0) |
| entry: |
| call void @side_effect(i32 0) |
| %i.inc = add nsw i32 %i, 1 |
| %c0 = icmp slt i32 %i.inc, %len |
| br i1 %c0, label %cont, label %right |
| |
| cont: |
| %c = load i1, i1* %c.ptr |
| br i1 %c, label %left, label %right |
| |
| left: |
| call void @side_effect(i32 0) |
| %c1 = icmp slt i32 %i, %len |
| br i1 %c1, label %left0, label %right |
| |
| left0: |
| call void @side_effect(i32 0) |
| ret void |
| |
| right: |
| %t = phi i32 [ 1, %left ], [ 2, %entry ], [ 3, %cont ] |
| call void @side_effect(i32 %t) |
| ret void |
| } |
| |
| ; A s<= B implies A s> B is false. |
| ; CHECK-LABEL: @test3( |
| ; CHECK: entry: |
| ; CHECK: br i1 %cmp, label %if.end, label %if.end3 |
| ; CHECK-NOT: br i1 %cmp1, label %if.then2, label %if.end |
| ; CHECK-NOT: call void @side_effect(i32 0) |
| ; CHECK: br label %if.end3 |
| ; CHECK: ret void |
| |
| define void @test3(i32 %a, i32 %b) { |
| entry: |
| %cmp = icmp sle i32 %a, %b |
| br i1 %cmp, label %if.then, label %if.end3 |
| |
| if.then: |
| %cmp1 = icmp sgt i32 %a, %b |
| br i1 %cmp1, label %if.then2, label %if.end |
| |
| if.then2: |
| call void @side_effect(i32 0) |
| br label %if.end |
| |
| if.end: |
| br label %if.end3 |
| |
| if.end3: |
| ret void |
| } |
| |
| declare void @is(i1) |
| |
| ; If A >=s B is false then A <=s B is implied true. |
| ; CHECK-LABEL: @test_sge_sle |
| ; CHECK: call void @is(i1 true) |
| ; CHECK-NOT: call void @is(i1 false) |
| define void @test_sge_sle(i32 %a, i32 %b) { |
| %cmp1 = icmp sge i32 %a, %b |
| br i1 %cmp1, label %untaken, label %taken |
| |
| taken: |
| %cmp2 = icmp sle i32 %a, %b |
| br i1 %cmp2, label %istrue, label %isfalse |
| |
| istrue: |
| call void @is(i1 true) |
| ret void |
| |
| isfalse: |
| call void @is(i1 false) |
| ret void |
| |
| untaken: |
| ret void |
| } |
| |
| ; If A <=s B is false then A <=s B is implied false. |
| ; CHECK-LABEL: @test_sle_sle |
| ; CHECK-NOT: call void @is(i1 true) |
| ; CHECK: call void @is(i1 false) |
| define void @test_sle_sle(i32 %a, i32 %b) { |
| %cmp1 = icmp sle i32 %a, %b |
| br i1 %cmp1, label %untaken, label %taken |
| |
| taken: |
| %cmp2 = icmp sle i32 %a, %b |
| br i1 %cmp2, label %istrue, label %isfalse |
| |
| istrue: |
| call void @is(i1 true) |
| ret void |
| |
| isfalse: |
| call void @is(i1 false) |
| ret void |
| |
| untaken: |
| ret void |
| } |