Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 1 | ; RUN: llc < %s -asm-verbose=false | FileCheck %s |
| 2 | |
| 3 | ; Test the CFG stackifier pass. |
| 4 | |
| 5 | target datalayout = "e-p:32:32-i64:64-n32:64-S128" |
| 6 | target triple = "wasm32-unknown-unknown" |
| 7 | |
| 8 | declare void @something() |
| 9 | |
| 10 | ; Test that loops are made contiguous, even in the presence of split backedges. |
| 11 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 12 | ; CHECK-LABEL: test0: |
| 13 | ; CHECK: loop |
Dan Gohman | 3cb66c8 | 2015-11-06 21:32:42 +0000 | [diff] [blame] | 14 | ; CHECK: i32.add |
JF Bastien | 3b0177c | 2015-10-20 00:37:42 +0000 | [diff] [blame] | 15 | ; CHECK: br_if |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 16 | ; CHECK: call |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 17 | ; CHECK: br BB0_1{{$}} |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 18 | ; CHECK: return{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 19 | define void @test0(i32 %n) { |
| 20 | entry: |
| 21 | br label %header |
| 22 | |
| 23 | header: |
| 24 | %i = phi i32 [ 0, %entry ], [ %i.next, %back ] |
| 25 | %i.next = add i32 %i, 1 |
| 26 | |
| 27 | %c = icmp slt i32 %i.next, %n |
| 28 | br i1 %c, label %back, label %exit |
| 29 | |
| 30 | exit: |
| 31 | ret void |
| 32 | |
| 33 | back: |
| 34 | call void @something() |
| 35 | br label %header |
| 36 | } |
| 37 | |
| 38 | ; Same as test0, but the branch condition is reversed. |
| 39 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 40 | ; CHECK-LABEL: test1: |
| 41 | ; CHECK: loop |
Dan Gohman | 3cb66c8 | 2015-11-06 21:32:42 +0000 | [diff] [blame] | 42 | ; CHECK: i32.add |
JF Bastien | 3b0177c | 2015-10-20 00:37:42 +0000 | [diff] [blame] | 43 | ; CHECK: br_if |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 44 | ; CHECK: call |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 45 | ; CHECK: br BB1_1{{$}} |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 46 | ; CHECK: return{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 47 | define void @test1(i32 %n) { |
| 48 | entry: |
| 49 | br label %header |
| 50 | |
| 51 | header: |
| 52 | %i = phi i32 [ 0, %entry ], [ %i.next, %back ] |
| 53 | %i.next = add i32 %i, 1 |
| 54 | |
| 55 | %c = icmp sge i32 %i.next, %n |
| 56 | br i1 %c, label %exit, label %back |
| 57 | |
| 58 | exit: |
| 59 | ret void |
| 60 | |
| 61 | back: |
| 62 | call void @something() |
| 63 | br label %header |
| 64 | } |
| 65 | |
| 66 | ; Test that a simple loop is handled as expected. |
| 67 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 68 | ; CHECK-LABEL: test2: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 69 | ; CHECK: block BB2_2{{$}} |
Derek Schuff | 4ed4778 | 2015-11-16 21:04:51 +0000 | [diff] [blame] | 70 | ; CHECK: br_if {{.*}}, BB2_2{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 71 | ; CHECK: BB2_1: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 72 | ; CHECK: br_if $pop{{[0-9]+}}, BB2_1{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 73 | ; CHECK: BB2_2: |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 74 | ; CHECK: return{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 75 | define void @test2(double* nocapture %p, i32 %n) { |
| 76 | entry: |
| 77 | %cmp.4 = icmp sgt i32 %n, 0 |
| 78 | br i1 %cmp.4, label %for.body.preheader, label %for.end |
| 79 | |
| 80 | for.body.preheader: |
| 81 | br label %for.body |
| 82 | |
| 83 | for.body: |
| 84 | %i.05 = phi i32 [ %inc, %for.body ], [ 0, %for.body.preheader ] |
| 85 | %arrayidx = getelementptr inbounds double, double* %p, i32 %i.05 |
| 86 | %0 = load double, double* %arrayidx, align 8 |
| 87 | %mul = fmul double %0, 3.200000e+00 |
| 88 | store double %mul, double* %arrayidx, align 8 |
| 89 | %inc = add nuw nsw i32 %i.05, 1 |
| 90 | %exitcond = icmp eq i32 %inc, %n |
| 91 | br i1 %exitcond, label %for.end.loopexit, label %for.body |
| 92 | |
| 93 | for.end.loopexit: |
| 94 | br label %for.end |
| 95 | |
| 96 | for.end: |
| 97 | ret void |
| 98 | } |
| 99 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 100 | ; CHECK-LABEL: doublediamond: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 101 | ; CHECK: block BB3_5{{$}} |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 102 | ; CHECK: block BB3_2{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 103 | ; CHECK: br_if $pop{{[0-9]+}}, BB3_2{{$}} |
Dan Gohman | 3280793 | 2015-11-23 16:19:56 +0000 | [diff] [blame] | 104 | ; CHECK: block BB3_4{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 105 | ; CHECK: br_if $pop{{[0-9]+}}, BB3_4{{$}} |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 106 | ; CHECK: br BB3_5{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 107 | ; CHECK: BB3_4: |
| 108 | ; CHECK: BB3_5: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 109 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 110 | define i32 @doublediamond(i32 %a, i32 %b, i32* %p) { |
| 111 | entry: |
| 112 | %c = icmp eq i32 %a, 0 |
| 113 | %d = icmp eq i32 %b, 0 |
| 114 | store volatile i32 0, i32* %p |
| 115 | br i1 %c, label %true, label %false |
| 116 | true: |
| 117 | store volatile i32 1, i32* %p |
| 118 | br label %exit |
| 119 | false: |
| 120 | store volatile i32 2, i32* %p |
| 121 | br i1 %d, label %ft, label %ff |
| 122 | ft: |
| 123 | store volatile i32 3, i32* %p |
| 124 | br label %exit |
| 125 | ff: |
| 126 | store volatile i32 4, i32* %p |
| 127 | br label %exit |
| 128 | exit: |
| 129 | store volatile i32 5, i32* %p |
| 130 | ret i32 0 |
| 131 | } |
| 132 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 133 | ; CHECK-LABEL: triangle: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 134 | ; CHECK: block BB4_2{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 135 | ; CHECK: br_if $pop{{[0-9]+}}, BB4_2{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 136 | ; CHECK: BB4_2: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 137 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 138 | define i32 @triangle(i32* %p, i32 %a) { |
| 139 | entry: |
| 140 | %c = icmp eq i32 %a, 0 |
| 141 | store volatile i32 0, i32* %p |
| 142 | br i1 %c, label %true, label %exit |
| 143 | true: |
| 144 | store volatile i32 1, i32* %p |
| 145 | br label %exit |
| 146 | exit: |
| 147 | store volatile i32 2, i32* %p |
| 148 | ret i32 0 |
| 149 | } |
| 150 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 151 | ; CHECK-LABEL: diamond: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 152 | ; CHECK: block BB5_3{{$}} |
| 153 | ; CHECK: block BB5_2{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 154 | ; CHECK: br_if $pop{{[0-9]+}}, BB5_2{{$}} |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 155 | ; CHECK: br BB5_3{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 156 | ; CHECK: BB5_2: |
| 157 | ; CHECK: BB5_3: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 158 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 159 | define i32 @diamond(i32* %p, i32 %a) { |
| 160 | entry: |
| 161 | %c = icmp eq i32 %a, 0 |
| 162 | store volatile i32 0, i32* %p |
| 163 | br i1 %c, label %true, label %false |
| 164 | true: |
| 165 | store volatile i32 1, i32* %p |
| 166 | br label %exit |
| 167 | false: |
| 168 | store volatile i32 2, i32* %p |
| 169 | br label %exit |
| 170 | exit: |
| 171 | store volatile i32 3, i32* %p |
| 172 | ret i32 0 |
| 173 | } |
| 174 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 175 | ; CHECK-LABEL: single_block: |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 176 | ; CHECK-NOT: br |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 177 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 178 | define i32 @single_block(i32* %p) { |
| 179 | entry: |
| 180 | store volatile i32 0, i32* %p |
| 181 | ret i32 0 |
| 182 | } |
| 183 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 184 | ; CHECK-LABEL: minimal_loop: |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 185 | ; CHECK-NOT: br |
| 186 | ; CHECK: BB7_1: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 187 | ; CHECK: i32.store $0, $pop{{[0-9]+}}{{$}} |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 188 | ; CHECK: br BB7_1{{$}} |
Dan Gohman | f685722 | 2015-11-23 19:12:37 +0000 | [diff] [blame^] | 189 | ; CHECK: BB7_2: |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 190 | define i32 @minimal_loop(i32* %p) { |
| 191 | entry: |
| 192 | store volatile i32 0, i32* %p |
| 193 | br label %loop |
| 194 | loop: |
| 195 | store volatile i32 1, i32* %p |
| 196 | br label %loop |
| 197 | } |
| 198 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 199 | ; CHECK-LABEL: simple_loop: |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 200 | ; CHECK-NOT: br |
| 201 | ; CHECK: BB8_1: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 202 | ; CHECK: loop BB8_2{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 203 | ; CHECK: br_if $pop{{[0-9]+}}, BB8_1{{$}} |
Dan Gohman | f685722 | 2015-11-23 19:12:37 +0000 | [diff] [blame^] | 204 | ; CHECK: BB8_2: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 205 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 206 | define i32 @simple_loop(i32* %p, i32 %a) { |
| 207 | entry: |
| 208 | %c = icmp eq i32 %a, 0 |
| 209 | store volatile i32 0, i32* %p |
| 210 | br label %loop |
| 211 | loop: |
| 212 | store volatile i32 1, i32* %p |
| 213 | br i1 %c, label %loop, label %exit |
| 214 | exit: |
| 215 | store volatile i32 2, i32* %p |
| 216 | ret i32 0 |
| 217 | } |
| 218 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 219 | ; CHECK-LABEL: doubletriangle: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 220 | ; CHECK: block BB9_4{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 221 | ; CHECK: br_if $pop{{[0-9]+}}, BB9_4{{$}} |
Dan Gohman | 3280793 | 2015-11-23 16:19:56 +0000 | [diff] [blame] | 222 | ; CHECK: block BB9_3{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 223 | ; CHECK: br_if $pop{{[0-9]+}}, BB9_3{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 224 | ; CHECK: BB9_3: |
| 225 | ; CHECK: BB9_4: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 226 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 227 | define i32 @doubletriangle(i32 %a, i32 %b, i32* %p) { |
| 228 | entry: |
| 229 | %c = icmp eq i32 %a, 0 |
| 230 | %d = icmp eq i32 %b, 0 |
| 231 | store volatile i32 0, i32* %p |
| 232 | br i1 %c, label %true, label %exit |
| 233 | true: |
| 234 | store volatile i32 2, i32* %p |
| 235 | br i1 %d, label %tt, label %tf |
| 236 | tt: |
| 237 | store volatile i32 3, i32* %p |
| 238 | br label %tf |
| 239 | tf: |
| 240 | store volatile i32 4, i32* %p |
| 241 | br label %exit |
| 242 | exit: |
| 243 | store volatile i32 5, i32* %p |
| 244 | ret i32 0 |
| 245 | } |
| 246 | |
Dan Gohman | e51c058 | 2015-10-06 00:27:55 +0000 | [diff] [blame] | 247 | ; CHECK-LABEL: ifelse_earlyexits: |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 248 | ; CHECK: block BB10_4{{$}} |
| 249 | ; CHECK: block BB10_2{{$}} |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 250 | ; CHECK: br_if $pop{{[0-9]+}}, BB10_2{{$}} |
Dan Gohman | cf4748f | 2015-11-12 17:04:33 +0000 | [diff] [blame] | 251 | ; CHECK: br BB10_4{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 252 | ; CHECK: BB10_2: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 253 | ; CHECK: br_if $pop{{[0-9]+}}, BB10_4{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 254 | ; CHECK: BB10_4: |
Dan Gohman | 4ba4816 | 2015-11-18 16:12:01 +0000 | [diff] [blame] | 255 | ; CHECK: return ${{[0-9]+}}{{$}} |
Dan Gohman | 950a13c | 2015-09-16 16:51:30 +0000 | [diff] [blame] | 256 | define i32 @ifelse_earlyexits(i32 %a, i32 %b, i32* %p) { |
| 257 | entry: |
| 258 | %c = icmp eq i32 %a, 0 |
| 259 | %d = icmp eq i32 %b, 0 |
| 260 | store volatile i32 0, i32* %p |
| 261 | br i1 %c, label %true, label %false |
| 262 | true: |
| 263 | store volatile i32 1, i32* %p |
| 264 | br label %exit |
| 265 | false: |
| 266 | store volatile i32 2, i32* %p |
| 267 | br i1 %d, label %ft, label %exit |
| 268 | ft: |
| 269 | store volatile i32 3, i32* %p |
| 270 | br label %exit |
| 271 | exit: |
| 272 | store volatile i32 4, i32* %p |
| 273 | ret i32 0 |
| 274 | } |
Dan Gohman | e3e4a5f | 2015-10-02 21:11:36 +0000 | [diff] [blame] | 275 | |
Dan Gohman | 3280793 | 2015-11-23 16:19:56 +0000 | [diff] [blame] | 276 | ; CHECK-LABEL: doublediamond_in_a_loop: |
| 277 | ; CHECK: BB11_1: |
| 278 | ; CHECK: loop BB11_7{{$}} |
| 279 | ; CHECK: block BB11_6{{$}} |
| 280 | ; CHECK: block BB11_3{{$}} |
| 281 | ; CHECK: br_if $pop{{.*}}, BB11_3{{$}} |
| 282 | ; CHECK: br BB11_6{{$}} |
| 283 | ; CHECK: BB11_3: |
| 284 | ; CHECK: block BB11_5{{$}} |
| 285 | ; CHECK: br_if $pop{{.*}}, BB11_5{{$}} |
| 286 | ; CHECK: br BB11_6{{$}} |
| 287 | ; CHECK: BB11_5: |
| 288 | ; CHECK: BB11_6: |
| 289 | ; CHECK: br BB11_1{{$}} |
Dan Gohman | f685722 | 2015-11-23 19:12:37 +0000 | [diff] [blame^] | 290 | ; CHECK: BB11_7: |
Dan Gohman | 3280793 | 2015-11-23 16:19:56 +0000 | [diff] [blame] | 291 | define i32 @doublediamond_in_a_loop(i32 %a, i32 %b, i32* %p) { |
| 292 | entry: |
| 293 | br label %header |
| 294 | header: |
| 295 | %c = icmp eq i32 %a, 0 |
| 296 | %d = icmp eq i32 %b, 0 |
| 297 | store volatile i32 0, i32* %p |
| 298 | br i1 %c, label %true, label %false |
| 299 | true: |
| 300 | store volatile i32 1, i32* %p |
| 301 | br label %exit |
| 302 | false: |
| 303 | store volatile i32 2, i32* %p |
| 304 | br i1 %d, label %ft, label %ff |
| 305 | ft: |
| 306 | store volatile i32 3, i32* %p |
| 307 | br label %exit |
| 308 | ff: |
| 309 | store volatile i32 4, i32* %p |
| 310 | br label %exit |
| 311 | exit: |
| 312 | store volatile i32 5, i32* %p |
| 313 | br label %header |
| 314 | } |
| 315 | |
Dan Gohman | e3e4a5f | 2015-10-02 21:11:36 +0000 | [diff] [blame] | 316 | ; Test that nested loops are handled. |
| 317 | |
| 318 | declare void @bar() |
| 319 | |
| 320 | define void @test3(i32 %w) { |
| 321 | entry: |
| 322 | br i1 undef, label %outer.ph, label %exit |
| 323 | |
| 324 | outer.ph: |
| 325 | br label %outer |
| 326 | |
| 327 | outer: |
| 328 | %tobool = icmp eq i32 undef, 0 |
| 329 | br i1 %tobool, label %inner, label %unreachable |
| 330 | |
| 331 | unreachable: |
| 332 | unreachable |
| 333 | |
| 334 | inner: |
| 335 | %c = icmp eq i32 undef, %w |
| 336 | br i1 %c, label %if.end, label %inner |
| 337 | |
| 338 | exit: |
| 339 | ret void |
| 340 | |
| 341 | if.end: |
| 342 | call void @bar() |
| 343 | br label %outer |
| 344 | } |