Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / llvm / test / Analysis / MustExecute / loop-header.ll
blob50efd74b61b1111432f9f29532c429c84ccbb6a8
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt -disable-output -passes=print-mustexecute %s 2>&1 | FileCheck %s
4 define i1 @header_with_icf(ptr noalias %p, i32 %high) {
5 ; CHECK-LABEL: @header_with_icf(
6 ; CHECK-LABEL:       loop:
7 ; CHECK:         %iv = phi i32 [ 0, %entry ], [ %iv.next, %loop ] ; (mustexec in: loop)
8 ; CHECK:          %v = load i32, ptr %p, align 4 ; (mustexec in: loop)
9 ; CHECK:          call void @maythrow_and_use(i32 %v)   ; (mustexec in: loop)
10 ; CHECK-NOT: mustexec
12 entry:
13   br label %loop
15 loop:
16   %iv = phi i32 [0, %entry], [%iv.next, %loop]
17   %v = load i32, ptr %p
18   call void @maythrow_and_use(i32 %v)
19   %iv.next = add nsw nuw i32 %iv, 1
20   %exit.test = icmp slt i32 %iv, %high
21   br i1 %exit.test, label %exit, label %loop
23 exit:
24   ret i1 false
27 define i1 @split_header(ptr noalias %p, i32 %high) {
28 ; CHECK-LABEL: @split_header(
29 ; CHECK-LABEL:       loop:
30 ; CHECK:          %iv = phi i32 [ 0, %entry ], [ %iv.next, %next ]      ; (mustexec in: loop)
31 ; CHECK:          %v = load i32, ptr %p, align 4 ; (mustexec in: loop)
32 ; CHECK:          br label %next ; (mustexec in: loop)
33 ; CHECK-NOT: mustexec
34 entry:
35   br label %loop
37 loop:
38   %iv = phi i32 [0, %entry], [%iv.next, %next]
39   %v = load i32, ptr %p
40   br label %next
41 next:
42   call void @maythrow_and_use(i32 %v)
43   %iv.next = add nsw nuw i32 %iv, 1
44   %exit.test = icmp slt i32 %iv, %high
45   br i1 %exit.test, label %exit, label %loop
47 exit:
48   ret i1 false
51 ; FIXME: everything in inner loop header should be must execute
52 ; for outer as well
53 define i1 @nested(ptr noalias %p, i32 %high) {
54 ; CHECK-LABEL: @nested
55 ; CHECK-LABEL: loop:                                             ; preds = %next
56 ; CHECK:         %iv = phi i32 [ 0, %entry ], [ %iv.next, %next ]       ; (mustexec in: loop)
57 ; CHECK:         br label %inner_loop   ; (mustexec in: loop)
58 ; CHECK-LABEL: inner_loop:
59 ; CHECK:         %v = load i32, ptr %p, align 4 ; (mustexec in: inner_loop)
60 ; CHECK:         %inner.test = icmp eq i32 %v, 0        ; (mustexec in: inner_loop)
61 ; CHECK:         br i1 %inner.test, label %inner_loop, label %next      ; (mustexec in: inner_loop)
62 ; CHECK-NOT: mustexec
64 entry:
65   br label %loop
67 loop:
68   %iv = phi i32 [0, %entry], [%iv.next, %next]
69   br label %inner_loop
71 inner_loop:
72   %v = load i32, ptr %p
73   %inner.test = icmp eq i32 %v, 0
74   br i1 %inner.test, label %inner_loop, label %next
76 next:
77   call void @maythrow_and_use(i32 %v)
78   %iv.next = add nsw nuw i32 %iv, 1
79   %exit.test = icmp slt i32 %iv, %high
80   br i1 %exit.test, label %exit, label %loop
82 exit:
83   ret i1 false
86 define i1 @nested_no_throw(ptr noalias %p, i32 %high) {
87 ; CHECK-LABEL: @nested_no_throw
88 ; CHECK-LABEL: loop:                                             ; preds = %next
89 ; CHECK:         %iv = phi i32 [ 0, %entry ], [ %iv.next, %next ]       ; (mustexec in: loop)
90 ; CHECK:         br label %inner_loop   ; (mustexec in: loop)
91 ; CHECK-LABEL: inner_loop:
92 ; CHECK:         %v = load i32, ptr %p, align 4 ; (mustexec in 2 loops: inner_loop, loop)
93 ; CHECK:         %inner.test = icmp eq i32 %v, 0        ; (mustexec in 2 loops: inner_loop, loop)
94 ; CHECK:         br i1 %inner.test, label %inner_loop, label %next      ; (mustexec in 2 loops: inner_loop, loop)
95 ; CHECK-LABEL: next:
96 ; CHECK:         %iv.next = add nuw nsw i32 %iv, 1 ; (mustexec in: loop)
97 ; CHECK:         %exit.test = icmp slt i32 %iv, %high ; (mustexec in: loop)
98 ; CHECK:         br i1 %exit.test, label %exit, label %loop ; (mustexec in: loop)
100 entry:
101   br label %loop
103 loop:
104   %iv = phi i32 [0, %entry], [%iv.next, %next]
105   br label %inner_loop
107 inner_loop:
108   %v = load i32, ptr %p
109   %inner.test = icmp eq i32 %v, 0
110   br i1 %inner.test, label %inner_loop, label %next
112 next:
113   %iv.next = add nsw nuw i32 %iv, 1
114   %exit.test = icmp slt i32 %iv, %high
115   br i1 %exit.test, label %exit, label %loop
117 exit:
118   ret i1 false
121 ; Since all the instructions in the loop dominate the only exit
122 ; and there's no implicit control flow in the loop, all must execute
123 ; FIXME: handled by loop safety info, test it
124 define i1 @nothrow_loop(ptr noalias %p, i32 %high) {
125 ; CHECK-LABEL: @nothrow_loop(
126 ; CHECK-LABEL:  loop:
127 ; CHECK:         %iv = phi i32 [ 0, %entry ], [ %iv.next, %next ] ; (mustexec in: loop)
128 ; CHECK:          br label %next ; (mustexec in: loop)
129 ; CHECK-LABEL: next:
130 ; CHECK:          %v = load i32, ptr %p, align 4 ; (mustexec in: loop)
131 ; CHECK:          %iv.next = add nuw nsw i32 %iv, 1 ; (mustexec in: loop)
132 ; CHECK:          %exit.test = icmp slt i32 %iv, %high ; (mustexec in: loop)
133 ; CHECK:          br i1 %exit.test, label %exit, label %loop ; (mustexec in: loop)
134 ; CHECK-NOT: mustexec
136 entry:
137   br label %loop
139 loop:
140   %iv = phi i32 [0, %entry], [%iv.next, %next]
141   br label %next
142 next:
143   %v = load i32, ptr %p
144   %iv.next = add nsw nuw i32 %iv, 1
145   %exit.test = icmp slt i32 %iv, %high
146   br i1 %exit.test, label %exit, label %loop
148 exit:
149   ret i1 false
153 declare void @maythrow_and_use(i32)