1 ; RUN: opt < %s -simplifycfg -S | FileCheck %s
7 define void @test1(i32 %V) {
8 %C1 = icmp eq i32 %V, 4 ; <i1> [#uses=1]
9 %C2 = icmp eq i32 %V, 17 ; <i1> [#uses=1]
10 %CN = or i1 %C1, %C2 ; <i1> [#uses=1]
11 br i1 %CN, label %T, label %F
19 ; CHECK: switch i32 %V, label %F [
20 ; CHECK: i32 17, label %T
21 ; CHECK: i32 4, label %T
25 define void @test2(i32 %V) {
26 %C1 = icmp ne i32 %V, 4 ; <i1> [#uses=1]
27 %C2 = icmp ne i32 %V, 17 ; <i1> [#uses=1]
28 %CN = and i1 %C1, %C2 ; <i1> [#uses=1]
29 br i1 %CN, label %T, label %F
37 ; CHECK: switch i32 %V, label %T [
38 ; CHECK: i32 17, label %F
39 ; CHECK: i32 4, label %F
43 define void @test3(i32 %V) {
44 %C1 = icmp eq i32 %V, 4 ; <i1> [#uses=1]
45 br i1 %C1, label %T, label %N
47 %C2 = icmp eq i32 %V, 17 ; <i1> [#uses=1]
48 br i1 %C2, label %T, label %F
57 ; CHECK: switch i32 %V, label %F [
58 ; CHECK: i32 4, label %T
59 ; CHECK: i32 17, label %T
65 define i32 @test4(i8 zeroext %c) nounwind ssp noredzone {
67 %cmp = icmp eq i8 %c, 62
68 br i1 %cmp, label %lor.end, label %lor.lhs.false
70 lor.lhs.false: ; preds = %entry
71 %cmp4 = icmp eq i8 %c, 34
72 br i1 %cmp4, label %lor.end, label %lor.rhs
74 lor.rhs: ; preds = %lor.lhs.false
75 %cmp8 = icmp eq i8 %c, 92
78 lor.end: ; preds = %lor.rhs, %lor.lhs.false, %entry
79 %0 = phi i1 [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp8, %lor.rhs ]
80 %lor.ext = zext i1 %0 to i32
84 ; CHECK: switch i8 %c, label %lor.rhs [
85 ; CHECK: i8 62, label %lor.end
86 ; CHECK: i8 34, label %lor.end
87 ; CHECK: i8 92, label %lor.end
91 define i32 @test5(i8 zeroext %c) nounwind ssp noredzone {
93 switch i8 %c, label %lor.rhs [
99 lor.rhs: ; preds = %entry
100 %V = icmp eq i8 %c, 92
103 lor.end: ; preds = %entry, %entry, %entry, %lor.rhs
104 %0 = phi i1 [ true, %entry ], [ %V, %lor.rhs ], [ true, %entry ], [ true, %entry ]
105 %lor.ext = zext i1 %0 to i32
108 ; CHECK: switch i8 %c, label %lor.rhs [
109 ; CHECK: i8 62, label %lor.end
110 ; CHECK: i8 34, label %lor.end
111 ; CHECK: i8 92, label %lor.end
116 define i1 @test6({ i32, i32 }* %I) {
118 %tmp.1.i = getelementptr { i32, i32 }* %I, i64 0, i32 1 ; <i32*> [#uses=1]
119 %tmp.2.i = load i32* %tmp.1.i ; <i32> [#uses=6]
120 %tmp.2 = icmp eq i32 %tmp.2.i, 14 ; <i1> [#uses=1]
121 br i1 %tmp.2, label %shortcirc_done.4, label %shortcirc_next.0
122 shortcirc_next.0: ; preds = %entry
123 %tmp.6 = icmp eq i32 %tmp.2.i, 15 ; <i1> [#uses=1]
124 br i1 %tmp.6, label %shortcirc_done.4, label %shortcirc_next.1
125 shortcirc_next.1: ; preds = %shortcirc_next.0
126 %tmp.11 = icmp eq i32 %tmp.2.i, 16 ; <i1> [#uses=1]
127 br i1 %tmp.11, label %shortcirc_done.4, label %shortcirc_next.2
128 shortcirc_next.2: ; preds = %shortcirc_next.1
129 %tmp.16 = icmp eq i32 %tmp.2.i, 17 ; <i1> [#uses=1]
130 br i1 %tmp.16, label %shortcirc_done.4, label %shortcirc_next.3
131 shortcirc_next.3: ; preds = %shortcirc_next.2
132 %tmp.21 = icmp eq i32 %tmp.2.i, 18 ; <i1> [#uses=1]
133 br i1 %tmp.21, label %shortcirc_done.4, label %shortcirc_next.4
134 shortcirc_next.4: ; preds = %shortcirc_next.3
135 %tmp.26 = icmp eq i32 %tmp.2.i, 19 ; <i1> [#uses=1]
136 br label %UnifiedReturnBlock
137 shortcirc_done.4: ; preds = %shortcirc_next.3, %shortcirc_next.2, %shortcirc_next.1, %shortcirc_next.0, %entry
138 br label %UnifiedReturnBlock
139 UnifiedReturnBlock: ; preds = %shortcirc_done.4, %shortcirc_next.4
140 %UnifiedRetVal = phi i1 [ %tmp.26, %shortcirc_next.4 ], [ true, %shortcirc_done.4 ] ; <i1> [#uses=1]
141 ret i1 %UnifiedRetVal
144 ; CHECK: %tmp.2.i.off = add i32 %tmp.2.i, -14
145 ; CHECK: %switch = icmp ult i32 %tmp.2.i.off, 6
148 define void @test7(i8 zeroext %c, i32 %x) nounwind ssp noredzone {
150 %cmp = icmp ult i32 %x, 32
151 %cmp4 = icmp eq i8 %c, 97
152 %or.cond = or i1 %cmp, %cmp4
153 %cmp9 = icmp eq i8 %c, 99
154 %or.cond11 = or i1 %or.cond, %cmp9
155 br i1 %or.cond11, label %if.then, label %if.end
157 if.then: ; preds = %entry
158 tail call void @foo1() nounwind noredzone
161 if.end: ; preds = %entry
165 ; CHECK: %cmp = icmp ult i32 %x, 32
166 ; CHECK: br i1 %cmp, label %if.then, label %switch.early.test
167 ; CHECK: switch.early.test:
168 ; CHECK: switch i8 %c, label %if.end [
169 ; CHECK: i8 99, label %if.then
170 ; CHECK: i8 97, label %if.then
174 define i32 @test8(i8 zeroext %c, i32 %x, i1 %C) nounwind ssp noredzone {
176 br i1 %C, label %N, label %if.then
178 %cmp = icmp ult i32 %x, 32
179 %cmp4 = icmp eq i8 %c, 97
180 %or.cond = or i1 %cmp, %cmp4
181 %cmp9 = icmp eq i8 %c, 99
182 %or.cond11 = or i1 %or.cond, %cmp9
183 br i1 %or.cond11, label %if.then, label %if.end
185 if.then: ; preds = %entry
186 %A = phi i32 [0, %entry], [42, %N]
187 tail call void @foo1() nounwind noredzone
190 if.end: ; preds = %entry
194 ; CHECK: switch.early.test:
195 ; CHECK: switch i8 %c, label %if.end [
196 ; CHECK: i8 99, label %if.then
197 ; CHECK: i8 97, label %if.then
199 ; CHECK: %A = phi i32 [ 0, %entry ], [ 42, %switch.early.test ], [ 42, %N ], [ 42, %switch.early.test ]
202 ;; This is "Example 7" from http://blog.regehr.org/archives/320
203 define i32 @test9(i8 zeroext %c) nounwind ssp noredzone {
205 %cmp = icmp ult i8 %c, 33
206 br i1 %cmp, label %lor.end, label %lor.lhs.false
208 lor.lhs.false: ; preds = %entry
209 %cmp4 = icmp eq i8 %c, 46
210 br i1 %cmp4, label %lor.end, label %lor.lhs.false6
212 lor.lhs.false6: ; preds = %lor.lhs.false
213 %cmp9 = icmp eq i8 %c, 44
214 br i1 %cmp9, label %lor.end, label %lor.lhs.false11
216 lor.lhs.false11: ; preds = %lor.lhs.false6
217 %cmp14 = icmp eq i8 %c, 58
218 br i1 %cmp14, label %lor.end, label %lor.lhs.false16
220 lor.lhs.false16: ; preds = %lor.lhs.false11
221 %cmp19 = icmp eq i8 %c, 59
222 br i1 %cmp19, label %lor.end, label %lor.lhs.false21
224 lor.lhs.false21: ; preds = %lor.lhs.false16
225 %cmp24 = icmp eq i8 %c, 60
226 br i1 %cmp24, label %lor.end, label %lor.lhs.false26
228 lor.lhs.false26: ; preds = %lor.lhs.false21
229 %cmp29 = icmp eq i8 %c, 62
230 br i1 %cmp29, label %lor.end, label %lor.lhs.false31
232 lor.lhs.false31: ; preds = %lor.lhs.false26
233 %cmp34 = icmp eq i8 %c, 34
234 br i1 %cmp34, label %lor.end, label %lor.lhs.false36
236 lor.lhs.false36: ; preds = %lor.lhs.false31
237 %cmp39 = icmp eq i8 %c, 92
238 br i1 %cmp39, label %lor.end, label %lor.rhs
240 lor.rhs: ; preds = %lor.lhs.false36
241 %cmp43 = icmp eq i8 %c, 39
244 lor.end: ; preds = %lor.rhs, %lor.lhs.false36, %lor.lhs.false31, %lor.lhs.false26, %lor.lhs.false21, %lor.lhs.false16, %lor.lhs.false11, %lor.lhs.false6, %lor.lhs.false, %entry
245 %0 = phi i1 [ true, %lor.lhs.false36 ], [ true, %lor.lhs.false31 ], [ true, %lor.lhs.false26 ], [ true, %lor.lhs.false21 ], [ true, %lor.lhs.false16 ], [ true, %lor.lhs.false11 ], [ true, %lor.lhs.false6 ], [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp43, %lor.rhs ]
246 %conv46 = zext i1 %0 to i32
250 ; CHECK: %cmp = icmp ult i8 %c, 33
251 ; CHECK: br i1 %cmp, label %lor.end, label %switch.early.test
253 ; CHECK: switch.early.test:
254 ; CHECK: switch i8 %c, label %lor.rhs [
255 ; CHECK: i8 92, label %lor.end
256 ; CHECK: i8 62, label %lor.end
257 ; CHECK: i8 60, label %lor.end
258 ; CHECK: i8 59, label %lor.end
259 ; CHECK: i8 58, label %lor.end
260 ; CHECK: i8 46, label %lor.end
261 ; CHECK: i8 44, label %lor.end
262 ; CHECK: i8 34, label %lor.end
263 ; CHECK: i8 39, label %lor.end
267 define i32 @test10(i32 %mode, i1 %Cond) {
268 %A = icmp ne i32 %mode, 0
269 %B = icmp ne i32 %mode, 51
271 %D = and i1 %C, %Cond
272 br i1 %D, label %T, label %F
279 ; CHECK: br i1 %Cond, label %switch.early.test, label %F
280 ; CHECK:switch.early.test:
281 ; CHECK: switch i32 %mode, label %T [
282 ; CHECK: i32 51, label %F
283 ; CHECK: i32 0, label %F
288 define i32 @test11(i32 %bar) nounwind {
290 %cmp = icmp eq i32 %bar, 4
291 %cmp2 = icmp eq i32 %bar, 35
292 %or.cond = or i1 %cmp, %cmp2
293 %cmp5 = icmp eq i32 %bar, 53
294 %or.cond1 = or i1 %or.cond, %cmp5
295 %cmp8 = icmp eq i32 %bar, 24
296 %or.cond2 = or i1 %or.cond1, %cmp8
297 %cmp11 = icmp eq i32 %bar, 23
298 %or.cond3 = or i1 %or.cond2, %cmp11
299 %cmp14 = icmp eq i32 %bar, 55
300 %or.cond4 = or i1 %or.cond3, %cmp14
301 %cmp17 = icmp eq i32 %bar, 12
302 %or.cond5 = or i1 %or.cond4, %cmp17
303 %cmp20 = icmp eq i32 %bar, 35
304 %or.cond6 = or i1 %or.cond5, %cmp20
305 br i1 %or.cond6, label %if.then, label %if.end
307 if.then: ; preds = %entry
310 if.end: ; preds = %entry
313 return: ; preds = %if.end, %if.then
314 %retval.0 = phi i32 [ 1, %if.then ], [ 0, %if.end ]
318 ; CHECK: switch i32 %bar, label %if.end [
319 ; CHECK: i32 55, label %return
320 ; CHECK: i32 53, label %return
321 ; CHECK: i32 35, label %return
322 ; CHECK: i32 24, label %return
323 ; CHECK: i32 23, label %return
324 ; CHECK: i32 12, label %return
325 ; CHECK: i32 4, label %return
329 define void @test12() nounwind {
334 %A = icmp eq i32 undef, undef
335 br i1 %A, label %bb55.us.us, label %malformed
338 %B = icmp ugt i32 undef, undef
339 br i1 %B, label %bb55.us.us, label %bb49.us.us
350 ; test13 - handle switch formation with ult.
351 define void @test13(i32 %x) nounwind ssp noredzone {
353 %cmp = icmp ult i32 %x, 2
354 br i1 %cmp, label %if.then, label %lor.lhs.false3
356 lor.lhs.false3: ; preds = %lor.lhs.false
357 %cmp5 = icmp eq i32 %x, 3
358 br i1 %cmp5, label %if.then, label %lor.lhs.false6
360 lor.lhs.false6: ; preds = %lor.lhs.false3
361 %cmp8 = icmp eq i32 %x, 4
362 br i1 %cmp8, label %if.then, label %lor.lhs.false9
364 lor.lhs.false9: ; preds = %lor.lhs.false6
365 %cmp11 = icmp eq i32 %x, 6
366 br i1 %cmp11, label %if.then, label %if.end
368 if.then: ; preds = %lor.lhs.false9, %lor.lhs.false6, %lor.lhs.false3, %lor.lhs.false, %entry
369 call void @foo1() noredzone
372 if.end: ; preds = %if.then, %lor.lhs.false9
375 ; CHECK: switch i32 %x, label %if.end [
376 ; CHECK: i32 6, label %if.then
377 ; CHECK: i32 4, label %if.then
378 ; CHECK: i32 3, label %if.then
379 ; CHECK: i32 1, label %if.then
380 ; CHECK: i32 0, label %if.then
384 ; test14 - handle switch formation with ult.
385 define void @test14(i32 %x) nounwind ssp noredzone {
387 %cmp = icmp ugt i32 %x, 2
388 br i1 %cmp, label %lor.lhs.false3, label %if.then
390 lor.lhs.false3: ; preds = %lor.lhs.false
391 %cmp5 = icmp ne i32 %x, 3
392 br i1 %cmp5, label %lor.lhs.false6, label %if.then
394 lor.lhs.false6: ; preds = %lor.lhs.false3
395 %cmp8 = icmp ne i32 %x, 4
396 br i1 %cmp8, label %lor.lhs.false9, label %if.then
398 lor.lhs.false9: ; preds = %lor.lhs.false6
399 %cmp11 = icmp ne i32 %x, 6
400 br i1 %cmp11, label %if.end, label %if.then
402 if.then: ; preds = %lor.lhs.false9, %lor.lhs.false6, %lor.lhs.false3, %lor.lhs.false, %entry
403 call void @foo1() noredzone
406 if.end: ; preds = %if.then, %lor.lhs.false9
409 ; CHECK: switch i32 %x, label %if.end [
410 ; CHECK: i32 6, label %if.then
411 ; CHECK: i32 4, label %if.then
412 ; CHECK: i32 3, label %if.then
413 ; CHECK: i32 1, label %if.then
414 ; CHECK: i32 0, label %if.then
418 ; Don't crash on ginormous ranges.
419 define void @test15(i128 %x) nounwind {
420 %cmp = icmp ugt i128 %x, 2
421 br i1 %cmp, label %if.end, label %lor.false
424 %cmp2 = icmp ne i128 %x, 100000000000000000000
425 br i1 %cmp2, label %if.end, label %if.then
428 call void @foo1() noredzone
441 define zeroext i1 @test16(i32 %x) nounwind {
444 ; CHECK: %x.off = add i32 %x, -1
445 ; CHECK: %switch = icmp ult i32 %x.off, 3
446 %cmp.i = icmp eq i32 %x, 1
447 br i1 %cmp.i, label %lor.end, label %lor.lhs.false
450 %cmp.i2 = icmp eq i32 %x, 2
451 br i1 %cmp.i2, label %lor.end, label %lor.rhs
454 %cmp.i1 = icmp eq i32 %x, 3
458 %0 = phi i1 [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp.i1, %lor.rhs ]
462 ; Check that we don't turn an icmp into a switch where it's not useful.
463 define void @test17(i32 %x, i32 %y) {
464 %cmp = icmp ult i32 %x, 3
465 %switch = icmp ult i32 %y, 2
466 %or.cond775 = or i1 %cmp, %switch
467 br i1 %or.cond775, label %lor.lhs.false8, label %return
470 tail call void @foo1()
477 ; CHECK-NOT: switch.early.test
478 ; CHECK-NOT: switch i32