1; RUN: opt < %s -simplifycfg -S | FileCheck %s
2
3declare void @foo1()
4
5declare void @foo2()
6
7define 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
12T:              ; preds = %0
13        call void @foo1( )
14        ret void
15F:              ; preds = %0
16        call void @foo2( )
17        ret void
18; CHECK: @test1
19; CHECK:  switch i32 %V, label %F [
20; CHECK:    i32 17, label %T
21; CHECK:    i32 4, label %T
22; CHECK:  ]
23}
24
25define 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
30T:              ; preds = %0
31        call void @foo1( )
32        ret void
33F:              ; preds = %0
34        call void @foo2( )
35        ret void
36; CHECK: @test2
37; CHECK:  switch i32 %V, label %T [
38; CHECK:    i32 17, label %F
39; CHECK:    i32 4, label %F
40; CHECK:  ]
41}
42
43define void @test3(i32 %V) {
44        %C1 = icmp eq i32 %V, 4         ; <i1> [#uses=1]
45        br i1 %C1, label %T, label %N
46N:              ; preds = %0
47        %C2 = icmp eq i32 %V, 17                ; <i1> [#uses=1]
48        br i1 %C2, label %T, label %F
49T:              ; preds = %N, %0
50        call void @foo1( )
51        ret void
52F:              ; preds = %N
53        call void @foo2( )
54        ret void
55
56; CHECK: @test3
57; CHECK: switch i32 %V, label %F [
58; CHECK:     i32 4, label %T
59; CHECK:     i32 17, label %T
60; CHECK:   ]
61}
62
63
64
65define i32 @test4(i8 zeroext %c) nounwind ssp noredzone {
66entry:
67  %cmp = icmp eq i8 %c, 62
68  br i1 %cmp, label %lor.end, label %lor.lhs.false
69
70lor.lhs.false:                                    ; preds = %entry
71  %cmp4 = icmp eq i8 %c, 34
72  br i1 %cmp4, label %lor.end, label %lor.rhs
73
74lor.rhs:                                          ; preds = %lor.lhs.false
75  %cmp8 = icmp eq i8 %c, 92
76  br label %lor.end
77
78lor.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
81  ret i32 %lor.ext
82
83; CHECK: @test4
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
88; CHECK:  ]
89}
90
91define i32 @test5(i8 zeroext %c) nounwind ssp noredzone {
92entry:
93  switch i8 %c, label %lor.rhs [
94    i8 62, label %lor.end
95    i8 34, label %lor.end
96    i8 92, label %lor.end
97  ]
98
99lor.rhs:                                          ; preds = %entry
100  %V = icmp eq i8 %c, 92
101  br label %lor.end
102
103lor.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
106  ret i32 %lor.ext
107; CHECK: @test5
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
112; CHECK:  ]
113}
114
115
116define i1 @test6({ i32, i32 }* %I) {
117entry:
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
122shortcirc_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
125shortcirc_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
128shortcirc_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
131shortcirc_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
134shortcirc_next.4:               ; preds = %shortcirc_next.3
135        %tmp.26 = icmp eq i32 %tmp.2.i, 19              ; <i1> [#uses=1]
136        br label %UnifiedReturnBlock
137shortcirc_done.4:               ; preds = %shortcirc_next.3, %shortcirc_next.2, %shortcirc_next.1, %shortcirc_next.0, %entry
138        br label %UnifiedReturnBlock
139UnifiedReturnBlock:             ; 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
142
143; CHECK: @test6
144; CHECK: %tmp.2.i.off = add i32 %tmp.2.i, -14
145; CHECK: %switch = icmp ult i32 %tmp.2.i.off, 6
146}
147
148define void @test7(i8 zeroext %c, i32 %x) nounwind ssp noredzone {
149entry:
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
156
157if.then:                                          ; preds = %entry
158  tail call void @foo1() nounwind noredzone
159  ret void
160
161if.end:                                           ; preds = %entry
162  ret void
163
164; CHECK: @test7
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
171; CHECK:   ]
172}
173
174define i32 @test8(i8 zeroext %c, i32 %x, i1 %C) nounwind ssp noredzone {
175entry:
176  br i1 %C, label %N, label %if.then
177N:
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
184
185if.then:                                          ; preds = %entry
186  %A = phi i32 [0, %entry], [42, %N]
187  tail call void @foo1() nounwind noredzone
188  ret i32 %A
189
190if.end:                                           ; preds = %entry
191  ret i32 0
192
193; CHECK: @test8
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
198; CHECK:   ]
199; CHECK:   %A = phi i32 [ 0, %entry ], [ 42, %switch.early.test ], [ 42, %N ], [ 42, %switch.early.test ]
200}
201
202;; This is "Example 7" from http://blog.regehr.org/archives/320
203define i32 @test9(i8 zeroext %c) nounwind ssp noredzone {
204entry:
205  %cmp = icmp ult i8 %c, 33
206  br i1 %cmp, label %lor.end, label %lor.lhs.false
207
208lor.lhs.false:                                    ; preds = %entry
209  %cmp4 = icmp eq i8 %c, 46
210  br i1 %cmp4, label %lor.end, label %lor.lhs.false6
211
212lor.lhs.false6:                                   ; preds = %lor.lhs.false
213  %cmp9 = icmp eq i8 %c, 44
214  br i1 %cmp9, label %lor.end, label %lor.lhs.false11
215
216lor.lhs.false11:                                  ; preds = %lor.lhs.false6
217  %cmp14 = icmp eq i8 %c, 58
218  br i1 %cmp14, label %lor.end, label %lor.lhs.false16
219
220lor.lhs.false16:                                  ; preds = %lor.lhs.false11
221  %cmp19 = icmp eq i8 %c, 59
222  br i1 %cmp19, label %lor.end, label %lor.lhs.false21
223
224lor.lhs.false21:                                  ; preds = %lor.lhs.false16
225  %cmp24 = icmp eq i8 %c, 60
226  br i1 %cmp24, label %lor.end, label %lor.lhs.false26
227
228lor.lhs.false26:                                  ; preds = %lor.lhs.false21
229  %cmp29 = icmp eq i8 %c, 62
230  br i1 %cmp29, label %lor.end, label %lor.lhs.false31
231
232lor.lhs.false31:                                  ; preds = %lor.lhs.false26
233  %cmp34 = icmp eq i8 %c, 34
234  br i1 %cmp34, label %lor.end, label %lor.lhs.false36
235
236lor.lhs.false36:                                  ; preds = %lor.lhs.false31
237  %cmp39 = icmp eq i8 %c, 92
238  br i1 %cmp39, label %lor.end, label %lor.rhs
239
240lor.rhs:                                          ; preds = %lor.lhs.false36
241  %cmp43 = icmp eq i8 %c, 39
242  br label %lor.end
243
244lor.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
247  ret i32 %conv46
248
249; CHECK: @test9
250; CHECK:   %cmp = icmp ult i8 %c, 33
251; CHECK:   br i1 %cmp, label %lor.end, label %switch.early.test
252
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
264; CHECK:   ]
265}
266
267define i32 @test10(i32 %mode, i1 %Cond) {
268  %A = icmp ne i32 %mode, 0
269  %B = icmp ne i32 %mode, 51
270  %C = and i1 %A, %B
271  %D = and i1 %C, %Cond
272  br i1 %D, label %T, label %F
273T:
274  ret i32 123
275F:
276  ret i32 324
277
278; CHECK: @test10
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
284; CHECK:  ]
285}
286
287; PR8780
288define i32 @test11(i32 %bar) nounwind {
289entry:
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
306
307if.then:                                          ; preds = %entry
308  br label %return
309
310if.end:                                           ; preds = %entry
311  br label %return
312
313return:                                           ; preds = %if.end, %if.then
314  %retval.0 = phi i32 [ 1, %if.then ], [ 0, %if.end ]
315  ret i32 %retval.0
316
317; CHECK: @test11
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
326; CHECK: ]
327}
328
329define void @test12() nounwind {
330entry:
331  br label %bb49.us.us
332
333bb49.us.us:
334  %A = icmp eq i32 undef, undef
335  br i1 %A, label %bb55.us.us, label %malformed
336
337bb48.us.us:
338  %B = icmp ugt i32 undef, undef
339  br i1 %B, label %bb55.us.us, label %bb49.us.us
340
341bb55.us.us:
342  br label %bb48.us.us
343
344malformed:
345  ret void
346; CHECK: @test12
347
348}
349
350; test13 - handle switch formation with ult.
351define void @test13(i32 %x) nounwind ssp noredzone {
352entry:
353  %cmp = icmp ult i32 %x, 2
354  br i1 %cmp, label %if.then, label %lor.lhs.false3
355
356lor.lhs.false3:                                   ; preds = %lor.lhs.false
357  %cmp5 = icmp eq i32 %x, 3
358  br i1 %cmp5, label %if.then, label %lor.lhs.false6
359
360lor.lhs.false6:                                   ; preds = %lor.lhs.false3
361  %cmp8 = icmp eq i32 %x, 4
362  br i1 %cmp8, label %if.then, label %lor.lhs.false9
363
364lor.lhs.false9:                                   ; preds = %lor.lhs.false6
365  %cmp11 = icmp eq i32 %x, 6
366  br i1 %cmp11, label %if.then, label %if.end
367
368if.then:                                          ; preds = %lor.lhs.false9, %lor.lhs.false6, %lor.lhs.false3, %lor.lhs.false, %entry
369  call void @foo1() noredzone
370  br label %if.end
371
372if.end:                                           ; preds = %if.then, %lor.lhs.false9
373  ret void
374; CHECK: @test13
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
381; CHECK:   ]
382}
383
384; test14 - handle switch formation with ult.
385define void @test14(i32 %x) nounwind ssp noredzone {
386entry:
387  %cmp = icmp ugt i32 %x, 2
388  br i1 %cmp, label %lor.lhs.false3, label %if.then
389
390lor.lhs.false3:                                   ; preds = %lor.lhs.false
391  %cmp5 = icmp ne i32 %x, 3
392  br i1 %cmp5, label %lor.lhs.false6, label %if.then
393
394lor.lhs.false6:                                   ; preds = %lor.lhs.false3
395  %cmp8 = icmp ne i32 %x, 4
396  br i1 %cmp8, label %lor.lhs.false9, label %if.then
397
398lor.lhs.false9:                                   ; preds = %lor.lhs.false6
399  %cmp11 = icmp ne i32 %x, 6
400  br i1 %cmp11, label %if.end, label %if.then
401
402if.then:                                          ; preds = %lor.lhs.false9, %lor.lhs.false6, %lor.lhs.false3, %lor.lhs.false, %entry
403  call void @foo1() noredzone
404  br label %if.end
405
406if.end:                                           ; preds = %if.then, %lor.lhs.false9
407  ret void
408; CHECK: @test14
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
415; CHECK:   ]
416}
417
418; Don't crash on ginormous ranges.
419define void @test15(i128 %x) nounwind {
420  %cmp = icmp ugt i128 %x, 2
421  br i1 %cmp, label %if.end, label %lor.false
422
423lor.false:
424  %cmp2 = icmp ne i128 %x, 100000000000000000000
425  br i1 %cmp2, label %if.end, label %if.then
426
427if.then:
428  call void @foo1() noredzone
429  br label %if.end
430
431if.end:
432  ret void
433
434; CHECK: @test15
435; CHECK-NOT: switch
436; CHECK: ret void
437}
438
439; PR8675
440; rdar://5134905
441define zeroext i1 @test16(i32 %x) nounwind {
442entry:
443; CHECK: @test16
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
448
449lor.lhs.false:
450  %cmp.i2 = icmp eq i32 %x, 2
451  br i1 %cmp.i2, label %lor.end, label %lor.rhs
452
453lor.rhs:
454  %cmp.i1 = icmp eq i32 %x, 3
455  br label %lor.end
456
457lor.end:
458  %0 = phi i1 [ true, %lor.lhs.false ], [ true, %entry ], [ %cmp.i1, %lor.rhs ]
459  ret i1 %0
460}
461
462; Check that we don't turn an icmp into a switch where it's not useful.
463define 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
468
469lor.lhs.false8:
470  tail call void @foo1()
471  ret void
472
473return:
474  ret void
475
476; CHECK: @test17
477; CHECK-NOT: switch.early.test
478; CHECK-NOT: switch i32
479; CHECK: ret void
480}
481
482