Bump version to 19.1.0-rc3
[llvm-project.git] / llvm / test / Transforms / SimplifyCFG / switch-range-to-icmp.ll
blob4136f33983a2bf3b6c8fd2f4062c1672e70f48d1
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt %s -passes=simplifycfg -simplifycfg-require-and-preserve-domtree=1 -switch-range-to-icmp -S | FileCheck %s
4 declare i32 @f(i32)
6 define i32 @basic(i32 %x) {
7 ; CHECK-LABEL: @basic(
8 ; CHECK-NEXT:  entry:
9 ; CHECK-NEXT:    [[X_OFF:%.*]] = add i32 [[X:%.*]], -5
10 ; CHECK-NEXT:    [[SWITCH:%.*]] = icmp ult i32 [[X_OFF]], 3
11 ; CHECK-NEXT:    br i1 [[SWITCH]], label [[A:%.*]], label [[DEFAULT:%.*]]
12 ; CHECK:       common.ret:
13 ; CHECK-NEXT:    [[COMMON_RET_OP:%.*]] = phi i32 [ [[TMP0:%.*]], [[DEFAULT]] ], [ [[TMP1:%.*]], [[A]] ]
14 ; CHECK-NEXT:    ret i32 [[COMMON_RET_OP]]
15 ; CHECK:       default:
16 ; CHECK-NEXT:    [[TMP0]] = call i32 @f(i32 0)
17 ; CHECK-NEXT:    br label [[COMMON_RET:%.*]]
18 ; CHECK:       a:
19 ; CHECK-NEXT:    [[TMP1]] = call i32 @f(i32 1)
20 ; CHECK-NEXT:    br label [[COMMON_RET]]
23 entry:
24   switch i32 %x, label %default [
25   i32 5, label %a
26   i32 6, label %a
27   i32 7, label %a
28   ]
29 default:
30   %0 = call i32 @f(i32 0)
31   ret i32 %0
33   %1 = call i32 @f(i32 1)
34   ret i32 %1
38 define i32 @unreachable(i32 %x) {
39 ; CHECK-LABEL: @unreachable(
40 ; CHECK-NEXT:  entry:
41 ; CHECK-NEXT:    [[X_OFF:%.*]] = add i32 [[X:%.*]], -5
42 ; CHECK-NEXT:    [[SWITCH:%.*]] = icmp ult i32 [[X_OFF]], 3
43 ; CHECK-NEXT:    br i1 [[SWITCH]], label [[A:%.*]], label [[B:%.*]]
44 ; CHECK:       common.ret:
45 ; CHECK-NEXT:    [[COMMON_RET_OP:%.*]] = phi i32 [ [[TMP0:%.*]], [[A]] ], [ [[TMP1:%.*]], [[B]] ]
46 ; CHECK-NEXT:    ret i32 [[COMMON_RET_OP]]
47 ; CHECK:       a:
48 ; CHECK-NEXT:    [[TMP0]] = call i32 @f(i32 0)
49 ; CHECK-NEXT:    br label [[COMMON_RET:%.*]]
50 ; CHECK:       b:
51 ; CHECK-NEXT:    [[TMP1]] = call i32 @f(i32 1)
52 ; CHECK-NEXT:    br label [[COMMON_RET]]
55 entry:
56   switch i32 %x, label %unreachable [
57   i32 5, label %a
58   i32 6, label %a
59   i32 7, label %a
60   i32 10, label %b
61   i32 20, label %b
62   i32 30, label %b
63   i32 40, label %b
64   ]
65 unreachable:
66   unreachable
68   %0 = call i32 @f(i32 0)
69   ret i32 %0
71   %1 = call i32 @f(i32 1)
72   ret i32 %1
76 define i32 @unreachable2(i32 %x) {
77 ; CHECK-LABEL: @unreachable2(
78 ; CHECK-NEXT:  entry:
79 ; CHECK-NEXT:    [[X_OFF:%.*]] = add i32 [[X:%.*]], -5
80 ; CHECK-NEXT:    [[SWITCH:%.*]] = icmp ult i32 [[X_OFF]], 3
81 ; CHECK-NEXT:    br i1 [[SWITCH]], label [[A:%.*]], label [[B:%.*]]
82 ; CHECK:       common.ret:
83 ; CHECK-NEXT:    [[COMMON_RET_OP:%.*]] = phi i32 [ [[TMP0:%.*]], [[A]] ], [ [[TMP1:%.*]], [[B]] ]
84 ; CHECK-NEXT:    ret i32 [[COMMON_RET_OP]]
85 ; CHECK:       a:
86 ; CHECK-NEXT:    [[TMP0]] = call i32 @f(i32 0)
87 ; CHECK-NEXT:    br label [[COMMON_RET:%.*]]
88 ; CHECK:       b:
89 ; CHECK-NEXT:    [[TMP1]] = call i32 @f(i32 1)
90 ; CHECK-NEXT:    br label [[COMMON_RET]]
93 entry:
94   ; Note: folding the most popular case destination into the default
95   ; would prevent switch-to-icmp here.
96   switch i32 %x, label %unreachable [
97   i32 5, label %a
98   i32 6, label %a
99   i32 7, label %a
100   i32 10, label %b
101   i32 20, label %b
102   ]
103 unreachable:
104   unreachable
106   %0 = call i32 @f(i32 0)
107   ret i32 %0
109   %1 = call i32 @f(i32 1)
110   ret i32 %1
113 ; This would crash because we did not clean up the
114 ; default block of the switch before removing the switch.
116 define void @PR42737(ptr %a, i1 %c) {
117 ; CHECK-LABEL: @PR42737(
118 ; CHECK-NEXT:  entry:
119 ; CHECK-NEXT:    unreachable
121 entry:
122   br i1 %c, label %switch, label %else
124 else:
125   store i32 2, ptr %a
126   br label %switch
128 switch:
129   %cleanup.dest1 = phi i32 [ 0, %else ], [ 3, %entry ]
130   switch i32 %cleanup.dest1, label %unreach1 [
131   i32 0, label %cleanup1
132   i32 3, label %cleanup2
133   ]
135 cleanup1:
136   br label %unreach2
138 cleanup2:
139   br label %unreach2
141 unreach1:
142   %phi2 = phi i32 [ %cleanup.dest1, %switch ]
143   unreachable
145 unreach2:
146   unreachable
150 define void @pr53208_single_reachable_dest(i8 %sw, ptr %p0) {
151 ; CHECK-LABEL: @pr53208_single_reachable_dest(
152 ; CHECK-NEXT:  group2:
153 ; CHECK-NEXT:    call void @bar(ptr [[P0:%.*]])
154 ; CHECK-NEXT:    ret void
156   switch i8 %sw, label %group3 [
157   i8 0, label %group1
158   i8 1, label %group1
159   i8 2, label %group1
160   i8 3, label %group1
161   i8 11, label %group1
162   i8 12, label %group1
163   i8 13, label %group1
164   i8 7, label %group1
165   i8 17, label %group1
166   i8 14, label %group1
167   i8 15, label %group1
168   i8 4, label %group2
169   i8 5, label %group2
170   i8 6, label %group2
171   i8 8, label %group2
172   i8 9, label %group2
173   i8 10, label %group2
174   ]
176 group1:
177   br label %exit
179 group2:
180   br label %exit
182 group3:
183   br label %exit
185 exit:
186   %phi = phi ptr [ null, %group3 ], [ %p0, %group2 ], [ null, %group1 ]
187   call void @bar(ptr %phi)
188   ret void
191 declare void @bar(ptr nonnull dereferenceable(4))