diff options
Diffstat (limited to 'test/Transforms/JumpThreading')
-rw-r--r-- | test/Transforms/JumpThreading/assume-edge-dom.ll | 39 | ||||
-rw-r--r-- | test/Transforms/JumpThreading/assume.ll | 68 | ||||
-rw-r--r-- | test/Transforms/JumpThreading/conservative-lvi.ll | 58 | ||||
-rw-r--r-- | test/Transforms/JumpThreading/pr22086.ll | 28 | ||||
-rw-r--r-- | test/Transforms/JumpThreading/thread-loads.ll | 39 |
5 files changed, 228 insertions, 4 deletions
diff --git a/test/Transforms/JumpThreading/assume-edge-dom.ll b/test/Transforms/JumpThreading/assume-edge-dom.ll new file mode 100644 index 000000000000..f1d0f41e2505 --- /dev/null +++ b/test/Transforms/JumpThreading/assume-edge-dom.ll @@ -0,0 +1,39 @@ +; RUN: opt -S -jump-threading < %s | FileCheck %s + +declare i8* @escape() +declare void @llvm.assume(i1) + +define i1 @test1(i1 %cond) { +entry: + br i1 %cond, label %taken, label %not_taken + +; CHECK-LABEL: @test1 +; CHECK: br i1 %cond, label %no, label %yes +; CHECK: ret i1 true + +taken: + %res1 = call i8* @escape() + %a = icmp eq i8* %res1, null + tail call void @llvm.assume(i1 %a) + br label %done +not_taken: + %res2 = call i8* @escape() + %b = icmp ne i8* %res2, null + tail call void @llvm.assume(i1 %b) + br label %done + +; An assume that can be used to simplify this comparison dominates each +; predecessor branch (although no assume dominates the cmp itself). Make sure +; this still can be simplified. + +done: + %res = phi i8* [ %res1, %taken ], [ %res2, %not_taken ] + %cnd = icmp ne i8* %res, null + br i1 %cnd, label %yes, label %no + +yes: + ret i1 true +no: + ret i1 false +} + diff --git a/test/Transforms/JumpThreading/assume.ll b/test/Transforms/JumpThreading/assume.ll new file mode 100644 index 000000000000..89dd0a954c01 --- /dev/null +++ b/test/Transforms/JumpThreading/assume.ll @@ -0,0 +1,68 @@ +; RUN: opt -S -jump-threading -dce < %s | FileCheck %s +target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128" +target triple = "x86_64-unknown-linux-gnu" + +; Function Attrs: nounwind uwtable +define i32 @test1(i32 %a, i32 %b) #0 { +entry: + %cmp = icmp sgt i32 %a, 5 + tail call void @llvm.assume(i1 %cmp) + %cmp1 = icmp sgt i32 %b, 1234 + br i1 %cmp1, label %if.then, label %if.else + +; CHECK-LABEL: @test1 +; CHECK: icmp sgt i32 %a, 5 +; CHECK: call void @llvm.assume +; CHECK-NOT: icmp sgt i32 %a, 3 +; CHECK: ret i32 + +if.then: ; preds = %entry + %cmp2 = icmp sgt i32 %a, 3 + br i1 %cmp2, label %if.then3, label %return + +if.then3: ; preds = %if.then + tail call void (...)* @bar() #1 + br label %return + +if.else: ; preds = %entry + tail call void (...)* @car() #1 + br label %return + +return: ; preds = %if.else, %if.then, %if.then3 + %retval.0 = phi i32 [ 1, %if.then3 ], [ 0, %if.then ], [ 0, %if.else ] + ret i32 %retval.0 +} + +define i32 @test2(i32 %a) #0 { +entry: + %cmp = icmp sgt i32 %a, 5 + tail call void @llvm.assume(i1 %cmp) + %cmp1 = icmp sgt i32 %a, 3 + br i1 %cmp1, label %if.then, label %return + +; CHECK-LABEL: @test2 +; CHECK: icmp sgt i32 %a, 5 +; CHECK: tail call void @llvm.assume +; CHECK: tail call void (...)* @bar() +; CHECK: ret i32 1 + + +if.then: ; preds = %entry + tail call void (...)* @bar() #1 + br label %return + +return: ; preds = %entry, %if.then + %retval.0 = phi i32 [ 1, %if.then ], [ 0, %entry ] + ret i32 %retval.0 +} + +; Function Attrs: nounwind +declare void @llvm.assume(i1) #1 + +declare void @bar(...) + +declare void @car(...) + +attributes #0 = { nounwind uwtable } +attributes #1 = { nounwind } + diff --git a/test/Transforms/JumpThreading/conservative-lvi.ll b/test/Transforms/JumpThreading/conservative-lvi.ll new file mode 100644 index 000000000000..1ea8cdc4693a --- /dev/null +++ b/test/Transforms/JumpThreading/conservative-lvi.ll @@ -0,0 +1,58 @@ +; RUN: opt -jump-threading -S %s | FileCheck %s + +; Check that we thread arg2neg -> checkpos -> end. +; +; LazyValueInfo would previously fail to analyze the value of %arg in arg2neg +; because its predecessing blocks (checkneg) hadn't been processed yet (PR21238) + +; CHECK-LABEL: @test_jump_threading +; CHECK: arg2neg: +; CHECK-NEXT: br i1 %arg1, label %end, label %checkpos.thread +; CHECK: checkpos.thread: +; CHECK-NEXT: br label %end + +define i32 @test_jump_threading(i1 %arg1, i32 %arg2) { +checkneg: + %cmp = icmp slt i32 %arg2, 0 + br i1 %cmp, label %arg2neg, label %checkpos + +arg2neg: + br i1 %arg1, label %end, label %checkpos + +checkpos: + %cmp2 = icmp sgt i32 %arg2, 0 + br i1 %cmp2, label %arg2pos, label %end + +arg2pos: + br label %end + +end: + %0 = phi i32 [ 1, %arg2neg ], [ 2, %checkpos ], [ 3, %arg2pos ] + ret i32 %0 +} + + +; arg2neg has an edge back to itself. If LazyValueInfo is not careful when +; visiting predecessors, it could get into an infinite loop. + +; CHECK-LABEL: test_infinite_loop + +define i32 @test_infinite_loop(i1 %arg1, i32 %arg2) { +checkneg: + %cmp = icmp slt i32 %arg2, 0 + br i1 %cmp, label %arg2neg, label %checkpos + +arg2neg: + br i1 %arg1, label %arg2neg, label %checkpos + +checkpos: + %cmp2 = icmp sgt i32 %arg2, 0 + br i1 %cmp2, label %arg2pos, label %end + +arg2pos: + br label %end + +end: + %0 = phi i32 [ 2, %checkpos ], [ 3, %arg2pos ] + ret i32 %0 +} diff --git a/test/Transforms/JumpThreading/pr22086.ll b/test/Transforms/JumpThreading/pr22086.ll new file mode 100644 index 000000000000..35d9aa5b1841 --- /dev/null +++ b/test/Transforms/JumpThreading/pr22086.ll @@ -0,0 +1,28 @@ +; RUN: opt -S -jump-threading < %s | FileCheck %s + + +; CHECK-LABEL: @f( +; CHECK-LABEL: entry: +; CHECK-NEXT: br label %[[loop:.*]] +; CHECK: [[loop]]: +; CHECK-NEXT: br label %[[loop]] + +define void @f() { +entry: + br label %for.cond1 + +if.end16: + %phi1 = phi i32 [ undef, %for.cond1 ] + %g.3 = phi i32 [ %g.1, %for.cond1 ] + %sext = shl i32 %g.3, 16 + %conv20 = ashr exact i32 %sext, 16 + %tobool21 = icmp eq i32 %phi1, 0 + br i1 %tobool21, label %lor.rhs, label %for.cond1 + +for.cond1: + %g.1 = phi i32 [ 0, %entry ], [ 0, %lor.rhs ], [ %g.3, %if.end16 ] + br i1 undef, label %lor.rhs, label %if.end16 + +lor.rhs: + br label %for.cond1 +} diff --git a/test/Transforms/JumpThreading/thread-loads.ll b/test/Transforms/JumpThreading/thread-loads.ll index e5bf64b9e256..4351f9918109 100644 --- a/test/Transforms/JumpThreading/thread-loads.ll +++ b/test/Transforms/JumpThreading/thread-loads.ll @@ -75,7 +75,38 @@ bb3: ; preds = %bb1 ret i32 %res.0 } -!0 = metadata !{metadata !3, metadata !3, i64 0} -!1 = metadata !{metadata !"omnipotent char", metadata !2} -!2 = metadata !{metadata !"Simple C/C++ TBAA", null} -!3 = metadata !{metadata !"int", metadata !1} +define i32 @test3(i8** %x, i1 %f) { +; Correctly thread loads of different (but compatible) types, placing bitcasts +; as necessary in the predecessors. This is especially tricky because the same +; predecessor ends up with two entries in the PHI node and they must share +; a single cast. +; CHECK-LABEL: @test3( +entry: + %0 = bitcast i8** %x to i32** + %1 = load i32** %0, align 8 + br i1 %f, label %if.end57, label %if.then56 +; CHECK: %[[LOAD:.*]] = load i32** +; CHECK: %[[CAST:.*]] = bitcast i32* %[[LOAD]] to i8* + +if.then56: + br label %if.end57 + +if.end57: + %2 = load i8** %x, align 8 + %tobool59 = icmp eq i8* %2, null + br i1 %tobool59, label %return, label %if.then60 +; CHECK: %[[PHI:.*]] = phi i8* [ %[[CAST]], %[[PRED:[^ ]+]] ], [ %[[CAST]], %[[PRED]] ] +; CHECK-NEXT: %[[CMP:.*]] = icmp eq i8* %[[PHI]], null +; CHECK-NEXT: br i1 %[[CMP]] + +if.then60: + ret i32 42 + +return: + ret i32 13 +} + +!0 = !{!3, !3, i64 0} +!1 = !{!"omnipotent char", !2} +!2 = !{!"Simple C/C++ TBAA", null} +!3 = !{!"int", !1} |