1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt < %s -passes=simplifycfg -hoist-common-insts=true -S | FileCheck %s
3 ; RUN: opt < %s -passes=simplifycfg -simplifycfg-require-and-preserve-domtree=1 -hoist-common-insts=true -S | FileCheck %s
5 define void @foo(i1 %C, ptr %P) {
7 ; CHECK-NEXT: common.ret:
8 ; CHECK-NEXT: store i32 7, ptr [[P:%.*]], align 4
11 br i1 %C, label %T, label %F
20 define void @foo_switch(i64 %C, ptr %P) {
21 ; CHECK-LABEL: @foo_switch(
22 ; CHECK-NEXT: common.ret:
23 ; CHECK-NEXT: store i32 7, ptr [[P:%.*]], align 4
24 ; CHECK-NEXT: ret void
26 switch i64 %C, label %bb0 [
41 define float @PR39535min(float %x) {
42 ; CHECK-LABEL: @PR39535min(
44 ; CHECK-NEXT: [[TOBOOL:%.*]] = fcmp une float [[X:%.*]], 0.000000e+00
45 ; CHECK-NEXT: [[DOTX:%.*]] = select fast i1 [[TOBOOL]], float 0.000000e+00, float [[X]]
46 ; CHECK-NEXT: ret float [[DOTX]]
49 %tobool = fcmp une float %x, 0.0
50 br i1 %tobool, label %cond.true, label %cond.false
59 %cond = phi fast float [ 0.0, %cond.true ], [ %x, %cond.false ]
63 define float @PR39535min_switch(i64 %i, float %x) {
64 ; CHECK-LABEL: @PR39535min_switch(
66 ; CHECK-NEXT: switch i64 [[I:%.*]], label [[END:%.*]] [
67 ; CHECK-NEXT: i64 1, label [[BB1:%.*]]
68 ; CHECK-NEXT: i64 2, label [[BB1]]
71 ; CHECK-NEXT: br label [[END]]
73 ; CHECK-NEXT: [[COND:%.*]] = phi fast float [ [[X:%.*]], [[BB1]] ], [ 0.000000e+00, [[ENTRY:%.*]] ]
74 ; CHECK-NEXT: ret float [[COND]]
77 switch i64 %i, label %bb0 [
92 %cond = phi fast float [ 0.0, %bb0 ], [ %x, %bb1 ], [ %x, %bb2 ]
96 define i32 @hoist_zext_flags_preserve(i1 %C, i8 %x) {
97 ; CHECK-LABEL: @hoist_zext_flags_preserve(
98 ; CHECK-NEXT: common.ret:
99 ; CHECK-NEXT: [[Z1:%.*]] = zext nneg i8 [[X:%.*]] to i32
100 ; CHECK-NEXT: ret i32 [[Z1]]
102 br i1 %C, label %T, label %F
104 %z1 = zext nneg i8 %x to i32
107 %z2 = zext nneg i8 %x to i32
111 define i32 @hoist_zext_flags_drop(i1 %C, i8 %x) {
112 ; CHECK-LABEL: @hoist_zext_flags_drop(
113 ; CHECK-NEXT: common.ret:
114 ; CHECK-NEXT: [[Z1:%.*]] = zext i8 [[X:%.*]] to i32
115 ; CHECK-NEXT: ret i32 [[Z1]]
117 br i1 %C, label %T, label %F
119 %z1 = zext nneg i8 %x to i32
122 %z2 = zext i8 %x to i32
127 define float @hoist_uitofp_flags_preserve(i1 %C, i8 %x) {
128 ; CHECK-LABEL: @hoist_uitofp_flags_preserve(
129 ; CHECK-NEXT: common.ret:
130 ; CHECK-NEXT: [[Z1:%.*]] = uitofp nneg i8 [[X:%.*]] to float
131 ; CHECK-NEXT: ret float [[Z1]]
133 br i1 %C, label %T, label %F
135 %z1 = uitofp nneg i8 %x to float
138 %z2 = uitofp nneg i8 %x to float
142 define float @hoist_uitofp_flags_drop(i1 %C, i8 %x) {
143 ; CHECK-LABEL: @hoist_uitofp_flags_drop(
144 ; CHECK-NEXT: common.ret:
145 ; CHECK-NEXT: [[Z1:%.*]] = uitofp i8 [[X:%.*]] to float
146 ; CHECK-NEXT: ret float [[Z1]]
148 br i1 %C, label %T, label %F
150 %z1 = uitofp nneg i8 %x to float
153 %z2 = uitofp i8 %x to float
157 define i32 @hoist_or_flags_preserve(i1 %C, i32 %x, i32 %y) {
158 ; CHECK-LABEL: @hoist_or_flags_preserve(
159 ; CHECK-NEXT: common.ret:
160 ; CHECK-NEXT: [[Z1:%.*]] = or disjoint i32 [[X:%.*]], [[Y:%.*]]
161 ; CHECK-NEXT: ret i32 [[Z1]]
163 br i1 %C, label %T, label %F
165 %z1 = or disjoint i32 %x, %y
168 %z2 = or disjoint i32 %x, %y
172 define i32 @hoist_or_flags_drop(i1 %C, i32 %x, i32 %y) {
173 ; CHECK-LABEL: @hoist_or_flags_drop(
174 ; CHECK-NEXT: common.ret:
175 ; CHECK-NEXT: [[Z1:%.*]] = or i32 [[X:%.*]], [[Y:%.*]]
176 ; CHECK-NEXT: ret i32 [[Z1]]
178 br i1 %C, label %T, label %F
183 %z2 = or disjoint i32 %x, %y
187 define i16 @hoist_trunc_flags_preserve(i1 %C, i32 %x) {
188 ; CHECK-LABEL: @hoist_trunc_flags_preserve(
189 ; CHECK-NEXT: common.ret:
190 ; CHECK-NEXT: [[Z1:%.*]] = trunc nuw nsw i32 [[X:%.*]] to i16
191 ; CHECK-NEXT: ret i16 [[Z1]]
193 br i1 %C, label %T, label %F
195 %z1 = trunc nsw nuw i32 %x to i16
198 %z2 = trunc nsw nuw i32 %x to i16
202 define i16 @hoist_trunc_flags_drop(i1 %C, i32 %x) {
203 ; CHECK-LABEL: @hoist_trunc_flags_drop(
204 ; CHECK-NEXT: common.ret:
205 ; CHECK-NEXT: [[Z1:%.*]] = trunc i32 [[X:%.*]] to i16
206 ; CHECK-NEXT: ret i16 [[Z1]]
208 br i1 %C, label %T, label %F
210 %z1 = trunc i32 %x to i16
213 %z2 = trunc nsw nuw i32 %x to i16
217 define ptr @hoist_gep_flags_both_nuw(i1 %C, ptr %p) {
218 ; CHECK-LABEL: @hoist_gep_flags_both_nuw(
219 ; CHECK-NEXT: common.ret:
220 ; CHECK-NEXT: [[GEP1:%.*]] = getelementptr nuw i8, ptr [[P:%.*]], i64 1
221 ; CHECK-NEXT: ret ptr [[GEP1]]
223 br i1 %C, label %T, label %F
225 %gep1 = getelementptr nuw i8, ptr %p, i64 1
228 %gep2 = getelementptr nuw i8, ptr %p, i64 1
232 define ptr @hoist_gep_flags_both_nusw(i1 %C, ptr %p) {
233 ; CHECK-LABEL: @hoist_gep_flags_both_nusw(
234 ; CHECK-NEXT: common.ret:
235 ; CHECK-NEXT: [[GEP1:%.*]] = getelementptr nusw i8, ptr [[P:%.*]], i64 1
236 ; CHECK-NEXT: ret ptr [[GEP1]]
238 br i1 %C, label %T, label %F
240 %gep1 = getelementptr nusw i8, ptr %p, i64 1
243 %gep2 = getelementptr nusw i8, ptr %p, i64 1
247 define ptr @hoist_gep_flags_intersect1(i1 %C, ptr %p) {
248 ; CHECK-LABEL: @hoist_gep_flags_intersect1(
249 ; CHECK-NEXT: common.ret:
250 ; CHECK-NEXT: [[GEP1:%.*]] = getelementptr nusw i8, ptr [[P:%.*]], i64 1
251 ; CHECK-NEXT: ret ptr [[GEP1]]
253 br i1 %C, label %T, label %F
255 %gep1 = getelementptr inbounds nuw i8, ptr %p, i64 1
258 %gep2 = getelementptr nusw i8, ptr %p, i64 1
262 define ptr @hoist_gep_flags_intersect2(i1 %C, ptr %p) {
263 ; CHECK-LABEL: @hoist_gep_flags_intersect2(
264 ; CHECK-NEXT: common.ret:
265 ; CHECK-NEXT: [[GEP1:%.*]] = getelementptr i8, ptr [[P:%.*]], i64 1
266 ; CHECK-NEXT: ret ptr [[GEP1]]
268 br i1 %C, label %T, label %F
270 %gep1 = getelementptr inbounds i8, ptr %p, i64 1
273 %gep2 = getelementptr nuw i8, ptr %p, i64 1
277 define i1 @hoist_icmp_flags_preserve(i1 %C, i32 %x, i32 %y) {
278 ; CHECK-LABEL: @hoist_icmp_flags_preserve(
279 ; CHECK-NEXT: common.ret:
280 ; CHECK-NEXT: [[Z1:%.*]] = icmp samesign ult i32 [[X:%.*]], [[Y:%.*]]
281 ; CHECK-NEXT: ret i1 [[Z1]]
283 br i1 %C, label %T, label %F
285 %z1 = icmp samesign ult i32 %x, %y
288 %z2 = icmp samesign ult i32 %x, %y
292 define i1 @hoist_icmp_flags_drop(i1 %C, i32 %x, i32 %y) {
293 ; CHECK-LABEL: @hoist_icmp_flags_drop(
294 ; CHECK-NEXT: common.ret:
295 ; CHECK-NEXT: [[Z1:%.*]] = icmp ult i32 [[X:%.*]], [[Y:%.*]]
296 ; CHECK-NEXT: ret i1 [[Z1]]
298 br i1 %C, label %T, label %F
300 %z1 = icmp ult i32 %x, %y
303 %z2 = icmp samesign ult i32 %x, %y