[DAGCombiner] Add target hook function to decide folding (mul (add x, c1), c2)
[llvm-project.git] / llvm / test / Transforms / InstCombine / icmp-add.ll
blob1750b5685c50ded5388b4884a59ea35db07bab46
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt < %s -instcombine -S | FileCheck %s
4 declare void @use(i32)
6 ; PR1949
8 define i1 @test1(i32 %a) {
9 ; CHECK-LABEL: @test1(
10 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt i32 [[A:%.*]], -5
11 ; CHECK-NEXT:    ret i1 [[C]]
13   %b = add i32 %a, 4
14   %c = icmp ult i32 %b, 4
15   ret i1 %c
18 define <2 x i1> @test1vec(<2 x i32> %a) {
19 ; CHECK-LABEL: @test1vec(
20 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt <2 x i32> [[A:%.*]], <i32 -5, i32 -5>
21 ; CHECK-NEXT:    ret <2 x i1> [[C]]
23   %b = add <2 x i32> %a, <i32 4, i32 4>
24   %c = icmp ult <2 x i32> %b, <i32 4, i32 4>
25   ret <2 x i1> %c
28 define i1 @test2(i32 %a) {
29 ; CHECK-LABEL: @test2(
30 ; CHECK-NEXT:    [[C:%.*]] = icmp ult i32 [[A:%.*]], 4
31 ; CHECK-NEXT:    ret i1 [[C]]
33   %b = sub i32 %a, 4
34   %c = icmp ugt i32 %b, -5
35   ret i1 %c
38 define <2 x i1> @test2vec(<2 x i32> %a) {
39 ; CHECK-LABEL: @test2vec(
40 ; CHECK-NEXT:    [[C:%.*]] = icmp ult <2 x i32> [[A:%.*]], <i32 4, i32 4>
41 ; CHECK-NEXT:    ret <2 x i1> [[C]]
43   %b = sub <2 x i32> %a, <i32 4, i32 4>
44   %c = icmp ugt <2 x i32> %b, <i32 -5, i32 -5>
45   ret <2 x i1> %c
48 define i1 @test3(i32 %a) {
49 ; CHECK-LABEL: @test3(
50 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt i32 [[A:%.*]], 2147483643
51 ; CHECK-NEXT:    ret i1 [[C]]
53   %b = add i32 %a, 4
54   %c = icmp slt i32 %b, 2147483652
55   ret i1 %c
58 define <2 x i1> @test3vec(<2 x i32> %a) {
59 ; CHECK-LABEL: @test3vec(
60 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt <2 x i32> [[A:%.*]], <i32 2147483643, i32 2147483643>
61 ; CHECK-NEXT:    ret <2 x i1> [[C]]
63   %b = add <2 x i32> %a, <i32 4, i32 4>
64   %c = icmp slt <2 x i32> %b, <i32 2147483652, i32 2147483652>
65   ret <2 x i1> %c
68 define i1 @test4(i32 %a) {
69 ; CHECK-LABEL: @test4(
70 ; CHECK-NEXT:    [[C:%.*]] = icmp slt i32 [[A:%.*]], -4
71 ; CHECK-NEXT:    ret i1 [[C]]
73   %b = add i32 %a, 2147483652
74   %c = icmp sge i32 %b, 4
75   ret i1 %c
78 define { i32, i1 } @test4multiuse(i32 %a) {
79 ; CHECK-LABEL: @test4multiuse(
80 ; CHECK-NEXT:    [[B:%.*]] = add nsw i32 [[A:%.*]], -2147483644
81 ; CHECK-NEXT:    [[C:%.*]] = icmp slt i32 [[A]], 2147483640
82 ; CHECK-NEXT:    [[TMP:%.*]] = insertvalue { i32, i1 } undef, i32 [[B]], 0
83 ; CHECK-NEXT:    [[RES:%.*]] = insertvalue { i32, i1 } [[TMP]], i1 [[C]], 1
84 ; CHECK-NEXT:    ret { i32, i1 } [[RES]]
87   %b = add nsw i32 %a, -2147483644
88   %c = icmp slt i32 %b, -4
90   %tmp = insertvalue { i32, i1 } undef, i32 %b, 0
91   %res = insertvalue { i32, i1 } %tmp, i1 %c, 1
93   ret { i32, i1 } %res
96 define <2 x i1> @test4vec(<2 x i32> %a) {
97 ; CHECK-LABEL: @test4vec(
98 ; CHECK-NEXT:    [[C:%.*]] = icmp slt <2 x i32> [[A:%.*]], <i32 -4, i32 -4>
99 ; CHECK-NEXT:    ret <2 x i1> [[C]]
101   %b = add <2 x i32> %a, <i32 2147483652, i32 2147483652>
102   %c = icmp sge <2 x i32> %b, <i32 4, i32 4>
103   ret <2 x i1> %c
106 ; icmp Pred (add nsw X, C2), C --> icmp Pred X, (C - C2), when C - C2 does not overflow.
107 ; This becomes equality because it's at the limit.
109 define i1 @nsw_slt1(i8 %a) {
110 ; CHECK-LABEL: @nsw_slt1(
111 ; CHECK-NEXT:    [[C:%.*]] = icmp eq i8 [[A:%.*]], -128
112 ; CHECK-NEXT:    ret i1 [[C]]
114   %b = add nsw i8 %a, 100
115   %c = icmp slt i8 %b, -27
116   ret i1 %c
119 define <2 x i1> @nsw_slt1_splat_vec(<2 x i8> %a) {
120 ; CHECK-LABEL: @nsw_slt1_splat_vec(
121 ; CHECK-NEXT:    [[C:%.*]] = icmp eq <2 x i8> [[A:%.*]], <i8 -128, i8 -128>
122 ; CHECK-NEXT:    ret <2 x i1> [[C]]
124   %b = add nsw <2 x i8> %a, <i8 100, i8 100>
125   %c = icmp slt <2 x i8> %b, <i8 -27, i8 -27>
126   ret <2 x i1> %c
129 ; icmp Pred (add nsw X, C2), C --> icmp Pred X, (C - C2), when C - C2 does not overflow.
130 ; This becomes equality because it's at the limit.
132 define i1 @nsw_slt2(i8 %a) {
133 ; CHECK-LABEL: @nsw_slt2(
134 ; CHECK-NEXT:    [[C:%.*]] = icmp ne i8 [[A:%.*]], 127
135 ; CHECK-NEXT:    ret i1 [[C]]
137   %b = add nsw i8 %a, -100
138   %c = icmp slt i8 %b, 27
139   ret i1 %c
142 define <2 x i1> @nsw_slt2_splat_vec(<2 x i8> %a) {
143 ; CHECK-LABEL: @nsw_slt2_splat_vec(
144 ; CHECK-NEXT:    [[C:%.*]] = icmp ne <2 x i8> [[A:%.*]], <i8 127, i8 127>
145 ; CHECK-NEXT:    ret <2 x i1> [[C]]
147   %b = add nsw <2 x i8> %a, <i8 -100, i8 -100>
148   %c = icmp slt <2 x i8> %b, <i8 27, i8 27>
149   ret <2 x i1> %c
152 ; icmp Pred (add nsw X, C2), C --> icmp Pred X, (C - C2), when C - C2 does not overflow.
153 ; Less than the limit, so the predicate doesn't change.
155 define i1 @nsw_slt3(i8 %a) {
156 ; CHECK-LABEL: @nsw_slt3(
157 ; CHECK-NEXT:    [[C:%.*]] = icmp slt i8 [[A:%.*]], -126
158 ; CHECK-NEXT:    ret i1 [[C]]
160   %b = add nsw i8 %a, 100
161   %c = icmp slt i8 %b, -26
162   ret i1 %c
165 ; icmp Pred (add nsw X, C2), C --> icmp Pred X, (C - C2), when C - C2 does not overflow.
166 ; Less than the limit, so the predicate doesn't change.
168 define i1 @nsw_slt4(i8 %a) {
169 ; CHECK-LABEL: @nsw_slt4(
170 ; CHECK-NEXT:    [[C:%.*]] = icmp slt i8 [[A:%.*]], 126
171 ; CHECK-NEXT:    ret i1 [[C]]
173   %b = add nsw i8 %a, -100
174   %c = icmp slt i8 %b, 26
175   ret i1 %c
178 ; icmp Pred (add nsw X, C2), C --> icmp Pred X, (C - C2), when C - C2 does not overflow.
179 ; Try sgt to make sure that works too.
181 define i1 @nsw_sgt1(i8 %a) {
182 ; CHECK-LABEL: @nsw_sgt1(
183 ; CHECK-NEXT:    [[C:%.*]] = icmp eq i8 [[A:%.*]], 127
184 ; CHECK-NEXT:    ret i1 [[C]]
186   %b = add nsw i8 %a, -100
187   %c = icmp sgt i8 %b, 26
188   ret i1 %c
191 define <2 x i1> @nsw_sgt1_splat_vec(<2 x i8> %a) {
192 ; CHECK-LABEL: @nsw_sgt1_splat_vec(
193 ; CHECK-NEXT:    [[C:%.*]] = icmp eq <2 x i8> [[A:%.*]], <i8 127, i8 127>
194 ; CHECK-NEXT:    ret <2 x i1> [[C]]
196   %b = add nsw <2 x i8> %a, <i8 -100, i8 -100>
197   %c = icmp sgt <2 x i8> %b, <i8 26, i8 26>
198   ret <2 x i1> %c
201 define i1 @nsw_sgt2(i8 %a) {
202 ; CHECK-LABEL: @nsw_sgt2(
203 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt i8 [[A:%.*]], -126
204 ; CHECK-NEXT:    ret i1 [[C]]
206   %b = add nsw i8 %a, 100
207   %c = icmp sgt i8 %b, -26
208   ret i1 %c
211 define <2 x i1> @nsw_sgt2_splat_vec(<2 x i8> %a) {
212 ; CHECK-LABEL: @nsw_sgt2_splat_vec(
213 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt <2 x i8> [[A:%.*]], <i8 -126, i8 -126>
214 ; CHECK-NEXT:    ret <2 x i1> [[C]]
216   %b = add nsw <2 x i8> %a, <i8 100, i8 100>
217   %c = icmp sgt <2 x i8> %b, <i8 -26, i8 -26>
218   ret <2 x i1> %c
221 ; icmp Pred (add nsw X, C2), C --> icmp Pred X, (C - C2), when C - C2 does not overflow.
222 ; Comparison with 0 doesn't need special-casing.
224 define i1 @slt_zero_add_nsw(i32 %a) {
225 ; CHECK-LABEL: @slt_zero_add_nsw(
226 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt i32 [[A:%.*]], -1
227 ; CHECK-NEXT:    ret i1 [[CMP]]
229   %add = add nsw i32 %a, 1
230   %cmp = icmp slt i32 %add, 0
231   ret i1 %cmp
234 ; The same fold should work with vectors.
236 define <2 x i1> @slt_zero_add_nsw_splat_vec(<2 x i8> %a) {
237 ; CHECK-LABEL: @slt_zero_add_nsw_splat_vec(
238 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt <2 x i8> [[A:%.*]], <i8 -1, i8 -1>
239 ; CHECK-NEXT:    ret <2 x i1> [[CMP]]
241   %add = add nsw <2 x i8> %a, <i8 1, i8 1>
242   %cmp = icmp slt <2 x i8> %add, zeroinitializer
243   ret <2 x i1> %cmp
246 ; Test the edges - instcombine should not interfere with simplification to constants.
247 ; Constant subtraction does not overflow, but this is false.
249 define i1 @nsw_slt3_ov_no(i8 %a) {
250 ; CHECK-LABEL: @nsw_slt3_ov_no(
251 ; CHECK-NEXT:    ret i1 false
253   %b = add nsw i8 %a, 100
254   %c = icmp slt i8 %b, -28
255   ret i1 %c
258 ; Test the edges - instcombine should not interfere with simplification to constants.
259 ; Constant subtraction overflows. This is false.
261 define i1 @nsw_slt4_ov(i8 %a) {
262 ; CHECK-LABEL: @nsw_slt4_ov(
263 ; CHECK-NEXT:    ret i1 false
265   %b = add nsw i8 %a, 100
266   %c = icmp slt i8 %b, -29
267   ret i1 %c
270 ; Test the edges - instcombine should not interfere with simplification to constants.
271 ; Constant subtraction overflows. This is true.
273 define i1 @nsw_slt5_ov(i8 %a) {
274 ; CHECK-LABEL: @nsw_slt5_ov(
275 ; CHECK-NEXT:    ret i1 true
277   %b = add nsw i8 %a, -100
278   %c = icmp slt i8 %b, 28
279   ret i1 %c
282 ; InstCombine should not thwart this opportunity to simplify completely.
284 define i1 @slt_zero_add_nsw_signbit(i8 %x) {
285 ; CHECK-LABEL: @slt_zero_add_nsw_signbit(
286 ; CHECK-NEXT:    ret i1 true
288   %y = add nsw i8 %x, -128
289   %z = icmp slt i8 %y, 0
290   ret i1 %z
293 ; InstCombine should not thwart this opportunity to simplify completely.
295 define i1 @slt_zero_add_nuw_signbit(i8 %x) {
296 ; CHECK-LABEL: @slt_zero_add_nuw_signbit(
297 ; CHECK-NEXT:    ret i1 true
299   %y = add nuw i8 %x, 128
300   %z = icmp slt i8 %y, 0
301   ret i1 %z
304 define i1 @reduce_add_ult(i32 %in) {
305 ; CHECK-LABEL: @reduce_add_ult(
306 ; CHECK-NEXT:    [[A18:%.*]] = icmp ult i32 [[IN:%.*]], 9
307 ; CHECK-NEXT:    ret i1 [[A18]]
309   %a6 = add nuw i32 %in, 3
310   %a18 = icmp ult i32 %a6, 12
311   ret i1 %a18
314 define i1 @reduce_add_ugt(i32 %in) {
315 ; CHECK-LABEL: @reduce_add_ugt(
316 ; CHECK-NEXT:    [[A18:%.*]] = icmp ugt i32 [[IN:%.*]], 9
317 ; CHECK-NEXT:    ret i1 [[A18]]
319   %a6 = add nuw i32 %in, 3
320   %a18 = icmp ugt i32 %a6, 12
321   ret i1 %a18
324 define i1 @reduce_add_ule(i32 %in) {
325 ; CHECK-LABEL: @reduce_add_ule(
326 ; CHECK-NEXT:    [[A18:%.*]] = icmp ult i32 [[IN:%.*]], 10
327 ; CHECK-NEXT:    ret i1 [[A18]]
329   %a6 = add nuw i32 %in, 3
330   %a18 = icmp ule i32 %a6, 12
331   ret i1 %a18
334 define i1 @reduce_add_uge(i32 %in) {
335 ; CHECK-LABEL: @reduce_add_uge(
336 ; CHECK-NEXT:    [[A18:%.*]] = icmp ugt i32 [[IN:%.*]], 8
337 ; CHECK-NEXT:    ret i1 [[A18]]
339   %a6 = add nuw i32 %in, 3
340   %a18 = icmp uge i32 %a6, 12
341   ret i1 %a18
344 define i1 @ult_add_ssubov(i32 %in) {
345 ; CHECK-LABEL: @ult_add_ssubov(
346 ; CHECK-NEXT:    ret i1 false
348   %a6 = add nuw i32 %in, 71
349   %a18 = icmp ult i32 %a6, 3
350   ret i1 %a18
353 define i1 @ult_add_nonuw(i8 %in) {
354 ; CHECK-LABEL: @ult_add_nonuw(
355 ; CHECK-NEXT:    [[A6:%.*]] = add i8 [[IN:%.*]], 71
356 ; CHECK-NEXT:    [[A18:%.*]] = icmp ult i8 [[A6]], 12
357 ; CHECK-NEXT:    ret i1 [[A18]]
359   %a6 = add i8 %in, 71
360   %a18 = icmp ult i8 %a6, 12
361   ret i1 %a18
364 define i1 @uge_add_nonuw(i32 %in) {
365 ; CHECK-LABEL: @uge_add_nonuw(
366 ; CHECK-NEXT:    [[A6:%.*]] = add i32 [[IN:%.*]], 3
367 ; CHECK-NEXT:    [[A18:%.*]] = icmp ugt i32 [[A6]], 11
368 ; CHECK-NEXT:    ret i1 [[A18]]
370   %a6 = add i32 %in, 3
371   %a18 = icmp uge i32 %a6, 12
372   ret i1 %a18
375 ; Test unsigned add overflow patterns. The div ops are only here to
376 ; thwart complexity based canonicalization of the operand order.
378 define i1 @op_ugt_sum_commute1(i8 %p1, i8 %p2) {
379 ; CHECK-LABEL: @op_ugt_sum_commute1(
380 ; CHECK-NEXT:    [[X:%.*]] = sdiv i8 42, [[P1:%.*]]
381 ; CHECK-NEXT:    [[Y:%.*]] = sdiv i8 42, [[P2:%.*]]
382 ; CHECK-NEXT:    [[TMP1:%.*]] = xor i8 [[X]], -1
383 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt i8 [[Y]], [[TMP1]]
384 ; CHECK-NEXT:    ret i1 [[C]]
386   %x = sdiv i8 42, %p1
387   %y = sdiv i8 42, %p2
388   %a = add i8 %x, %y
389   %c = icmp ugt i8 %x, %a
390   ret i1 %c
393 define <2 x i1> @op_ugt_sum_vec_commute2(<2 x i8> %p1, <2 x i8> %p2) {
394 ; CHECK-LABEL: @op_ugt_sum_vec_commute2(
395 ; CHECK-NEXT:    [[X:%.*]] = sdiv <2 x i8> <i8 42, i8 -42>, [[P1:%.*]]
396 ; CHECK-NEXT:    [[Y:%.*]] = sdiv <2 x i8> <i8 42, i8 -42>, [[P2:%.*]]
397 ; CHECK-NEXT:    [[TMP1:%.*]] = xor <2 x i8> [[X]], <i8 -1, i8 -1>
398 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt <2 x i8> [[Y]], [[TMP1]]
399 ; CHECK-NEXT:    ret <2 x i1> [[C]]
401   %x = sdiv <2 x i8> <i8 42, i8 -42>, %p1
402   %y = sdiv <2 x i8> <i8 42, i8 -42>, %p2
403   %a = add <2 x i8> %y, %x
404   %c = icmp ugt <2 x i8> %x, %a
405   ret <2 x i1> %c
408 define i1 @sum_ugt_op_uses(i8 %p1, i8 %p2, i8* %p3) {
409 ; CHECK-LABEL: @sum_ugt_op_uses(
410 ; CHECK-NEXT:    [[X:%.*]] = sdiv i8 42, [[P1:%.*]]
411 ; CHECK-NEXT:    [[Y:%.*]] = sdiv i8 42, [[P2:%.*]]
412 ; CHECK-NEXT:    [[A:%.*]] = add nsw i8 [[X]], [[Y]]
413 ; CHECK-NEXT:    store i8 [[A]], i8* [[P3:%.*]], align 1
414 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt i8 [[X]], [[A]]
415 ; CHECK-NEXT:    ret i1 [[C]]
417   %x = sdiv i8 42, %p1
418   %y = sdiv i8 42, %p2
419   %a = add i8 %x, %y
420   store i8 %a, i8* %p3
421   %c = icmp ugt i8 %x, %a
422   ret i1 %c
425 define <2 x i1> @sum_ult_op_vec_commute1(<2 x i8> %p1, <2 x i8> %p2) {
426 ; CHECK-LABEL: @sum_ult_op_vec_commute1(
427 ; CHECK-NEXT:    [[X:%.*]] = sdiv <2 x i8> <i8 42, i8 -42>, [[P1:%.*]]
428 ; CHECK-NEXT:    [[Y:%.*]] = sdiv <2 x i8> <i8 -42, i8 42>, [[P2:%.*]]
429 ; CHECK-NEXT:    [[TMP1:%.*]] = xor <2 x i8> [[X]], <i8 -1, i8 -1>
430 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt <2 x i8> [[Y]], [[TMP1]]
431 ; CHECK-NEXT:    ret <2 x i1> [[C]]
433   %x = sdiv <2 x i8> <i8 42, i8 -42>, %p1
434   %y = sdiv <2 x i8> <i8 -42, i8 42>, %p2
435   %a = add <2 x i8> %x, %y
436   %c = icmp ult <2 x i8> %a, %x
437   ret <2 x i1> %c
440 define i1 @sum_ult_op_commute2(i8 %p1, i8 %p2) {
441 ; CHECK-LABEL: @sum_ult_op_commute2(
442 ; CHECK-NEXT:    [[X:%.*]] = sdiv i8 42, [[P1:%.*]]
443 ; CHECK-NEXT:    [[Y:%.*]] = sdiv i8 42, [[P2:%.*]]
444 ; CHECK-NEXT:    [[TMP1:%.*]] = xor i8 [[X]], -1
445 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt i8 [[Y]], [[TMP1]]
446 ; CHECK-NEXT:    ret i1 [[C]]
448   %x = sdiv i8 42, %p1
449   %y = sdiv i8 42, %p2
450   %a = add i8 %y, %x
451   %c = icmp ult i8 %a, %x
452   ret i1 %c
455 define i1 @sum_ult_op_uses(i8 %x, i8 %y, i8* %p) {
456 ; CHECK-LABEL: @sum_ult_op_uses(
457 ; CHECK-NEXT:    [[A:%.*]] = add i8 [[Y:%.*]], [[X:%.*]]
458 ; CHECK-NEXT:    store i8 [[A]], i8* [[P:%.*]], align 1
459 ; CHECK-NEXT:    [[C:%.*]] = icmp ult i8 [[A]], [[X]]
460 ; CHECK-NEXT:    ret i1 [[C]]
462   %a = add i8 %y, %x
463   store i8 %a, i8* %p
464   %c = icmp ult i8 %a, %x
465   ret i1 %c
468 ; X + Z >s Y + Z -> X > Y if there is no overflow.
469 define i1 @common_op_nsw(i32 %x, i32 %y, i32 %z) {
470 ; CHECK-LABEL: @common_op_nsw(
471 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt i32 [[X:%.*]], [[Y:%.*]]
472 ; CHECK-NEXT:    ret i1 [[C]]
474   %lhs = add nsw i32 %x, %z
475   %rhs = add nsw i32 %y, %z
476   %c = icmp sgt i32 %lhs, %rhs
477   ret i1 %c
480 define i1 @common_op_nsw_extra_uses(i32 %x, i32 %y, i32 %z) {
481 ; CHECK-LABEL: @common_op_nsw_extra_uses(
482 ; CHECK-NEXT:    [[LHS:%.*]] = add nsw i32 [[X:%.*]], [[Z:%.*]]
483 ; CHECK-NEXT:    call void @use(i32 [[LHS]])
484 ; CHECK-NEXT:    [[RHS:%.*]] = add nsw i32 [[Y:%.*]], [[Z]]
485 ; CHECK-NEXT:    call void @use(i32 [[RHS]])
486 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt i32 [[X]], [[Y]]
487 ; CHECK-NEXT:    ret i1 [[C]]
489   %lhs = add nsw i32 %x, %z
490   call void @use(i32 %lhs)
491   %rhs = add nsw i32 %y, %z
492   call void @use(i32 %rhs)
493   %c = icmp sgt i32 %lhs, %rhs
494   ret i1 %c
497 ; X + Z >u Z + Y -> X > Y if there is no overflow.
498 define i1 @common_op_nuw(i32 %x, i32 %y, i32 %z) {
499 ; CHECK-LABEL: @common_op_nuw(
500 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt i32 [[X:%.*]], [[Y:%.*]]
501 ; CHECK-NEXT:    ret i1 [[C]]
503   %lhs = add nuw i32 %x, %z
504   %rhs = add nuw i32 %z, %y
505   %c = icmp ugt i32 %lhs, %rhs
506   ret i1 %c
509 define i1 @common_op_nuw_extra_uses(i32 %x, i32 %y, i32 %z) {
510 ; CHECK-LABEL: @common_op_nuw_extra_uses(
511 ; CHECK-NEXT:    [[LHS:%.*]] = add nuw i32 [[X:%.*]], [[Z:%.*]]
512 ; CHECK-NEXT:    call void @use(i32 [[LHS]])
513 ; CHECK-NEXT:    [[RHS:%.*]] = add nuw i32 [[Z]], [[Y:%.*]]
514 ; CHECK-NEXT:    call void @use(i32 [[RHS]])
515 ; CHECK-NEXT:    [[C:%.*]] = icmp ugt i32 [[X]], [[Y]]
516 ; CHECK-NEXT:    ret i1 [[C]]
518   %lhs = add nuw i32 %x, %z
519   call void @use(i32 %lhs)
520   %rhs = add nuw i32 %z, %y
521   call void @use(i32 %rhs)
522   %c = icmp ugt i32 %lhs, %rhs
523   ret i1 %c
526 define i1 @common_op_nsw_commute(i32 %x, i32 %y, i32 %z) {
527 ; CHECK-LABEL: @common_op_nsw_commute(
528 ; CHECK-NEXT:    [[C:%.*]] = icmp slt i32 [[X:%.*]], [[Y:%.*]]
529 ; CHECK-NEXT:    ret i1 [[C]]
531   %lhs = add nsw i32 %z, %x
532   %rhs = add nsw i32 %y, %z
533   %c = icmp slt i32 %lhs, %rhs
534   ret i1 %c
537 define i1 @common_op_nuw_commute(i32 %x, i32 %y, i32 %z) {
538 ; CHECK-LABEL: @common_op_nuw_commute(
539 ; CHECK-NEXT:    [[C:%.*]] = icmp ult i32 [[X:%.*]], [[Y:%.*]]
540 ; CHECK-NEXT:    ret i1 [[C]]
542   %lhs = add nuw i32 %z, %x
543   %rhs = add nuw i32 %z, %y
544   %c = icmp ult i32 %lhs, %rhs
545   ret i1 %c
548 ; X + Y > X -> Y > 0 if there is no overflow.
549 define i1 @common_op_test29(i32 %x, i32 %y) {
550 ; CHECK-LABEL: @common_op_test29(
551 ; CHECK-NEXT:    [[C:%.*]] = icmp sgt i32 [[Y:%.*]], 0
552 ; CHECK-NEXT:    ret i1 [[C]]
554   %lhs = add nsw i32 %x, %y
555   %c = icmp sgt i32 %lhs, %x
556   ret i1 %c
559 ; X + Y > X -> Y > 0 if there is no overflow.
560 define i1 @sum_nuw(i32 %x, i32 %y) {
561 ; CHECK-LABEL: @sum_nuw(
562 ; CHECK-NEXT:    [[C:%.*]] = icmp ne i32 [[Y:%.*]], 0
563 ; CHECK-NEXT:    ret i1 [[C]]
565   %lhs = add nuw i32 %x, %y
566   %c = icmp ugt i32 %lhs, %x
567   ret i1 %c
570 ; X > X + Y -> 0 > Y if there is no overflow.
571 define i1 @sum_nsw_commute(i32 %x, i32 %y) {
572 ; CHECK-LABEL: @sum_nsw_commute(
573 ; CHECK-NEXT:    [[C:%.*]] = icmp slt i32 [[Y:%.*]], 0
574 ; CHECK-NEXT:    ret i1 [[C]]
576   %rhs = add nsw i32 %x, %y
577   %c = icmp sgt i32 %x, %rhs
578   ret i1 %c
581 ; X > X + Y -> 0 > Y if there is no overflow.
582 define i1 @sum_nuw_commute(i32 %x, i32 %y) {
583 ; CHECK-LABEL: @sum_nuw_commute(
584 ; CHECK-NEXT:    ret i1 false
586   %rhs = add nuw i32 %x, %y
587   %c = icmp ugt i32 %x, %rhs
588   ret i1 %c
591 ; PR2698 - https://bugs.llvm.org/show_bug.cgi?id=2698
593 declare void @use1(i1)
594 declare void @use8(i8)
596 define void @bzip1(i8 %a, i8 %b, i8 %x) {
597 ; CHECK-LABEL: @bzip1(
598 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq i8 [[A:%.*]], [[B:%.*]]
599 ; CHECK-NEXT:    call void @use1(i1 [[CMP]])
600 ; CHECK-NEXT:    ret void
602   %add1 = add i8 %a, %x
603   %add2 = add i8 %b, %x
604   %cmp = icmp eq i8 %add1, %add2
605   call void @use1(i1 %cmp)
606   ret void
609 define void @bzip2(i8 %a, i8 %b, i8 %x) {
610 ; CHECK-LABEL: @bzip2(
611 ; CHECK-NEXT:    [[ADD1:%.*]] = add i8 [[A:%.*]], [[X:%.*]]
612 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq i8 [[A]], [[B:%.*]]
613 ; CHECK-NEXT:    call void @use1(i1 [[CMP]])
614 ; CHECK-NEXT:    call void @use8(i8 [[ADD1]])
615 ; CHECK-NEXT:    ret void
617   %add1 = add i8 %a, %x
618   %add2 = add i8 %b, %x
619   %cmp = icmp eq i8 %add1, %add2
620   call void @use1(i1 %cmp)
621   call void @use8(i8 %add1)
622   ret void
625 define <2 x i1> @icmp_eq_add_undef(<2 x i32> %a) {
626 ; CHECK-LABEL: @icmp_eq_add_undef(
627 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq <2 x i32> [[A:%.*]], <i32 5, i32 undef>
628 ; CHECK-NEXT:    ret <2 x i1> [[CMP]]
630   %add = add <2 x i32> %a, <i32 5, i32 undef>
631   %cmp = icmp eq <2 x i32> %add, <i32 10, i32 10>
632   ret <2 x i1> %cmp
635 define <2 x i1> @icmp_eq_add_non_splat(<2 x i32> %a) {
636 ; CHECK-LABEL: @icmp_eq_add_non_splat(
637 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq <2 x i32> [[A:%.*]], <i32 5, i32 4>
638 ; CHECK-NEXT:    ret <2 x i1> [[CMP]]
640   %add = add <2 x i32> %a, <i32 5, i32 6>
641   %cmp = icmp eq <2 x i32> %add, <i32 10, i32 10>
642   ret <2 x i1> %cmp
645 define <2 x i1> @icmp_eq_add_undef2(<2 x i32> %a) {
646 ; CHECK-LABEL: @icmp_eq_add_undef2(
647 ; CHECK-NEXT:    [[ADD:%.*]] = add <2 x i32> [[A:%.*]], <i32 5, i32 5>
648 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq <2 x i32> [[ADD]], <i32 10, i32 undef>
649 ; CHECK-NEXT:    ret <2 x i1> [[CMP]]
651   %add = add <2 x i32> %a, <i32 5, i32 5>
652   %cmp = icmp eq <2 x i32> %add, <i32 10, i32 undef>
653   ret <2 x i1> %cmp
656 define <2 x i1> @icmp_eq_add_non_splat2(<2 x i32> %a) {
657 ; CHECK-LABEL: @icmp_eq_add_non_splat2(
658 ; CHECK-NEXT:    [[ADD:%.*]] = add <2 x i32> [[A:%.*]], <i32 5, i32 5>
659 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq <2 x i32> [[ADD]], <i32 10, i32 11>
660 ; CHECK-NEXT:    ret <2 x i1> [[CMP]]
662   %add = add <2 x i32> %a, <i32 5, i32 5>
663   %cmp = icmp eq <2 x i32> %add, <i32 10, i32 11>
664   ret <2 x i1> %cmp
667 define i1 @without_nsw_nuw(i8 %x, i8 %y) {
668 ; CHECK-LABEL: @without_nsw_nuw(
669 ; CHECK-NEXT:    [[TMP1:%.*]] = add i8 [[X:%.*]], 2
670 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[Y:%.*]]
671 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
673   %t1 = add i8 %x, 37
674   %t2 = add i8 %y, 35
675   %tobool = icmp eq i8 %t2, %t1
676   ret i1 %tobool
679 define i1 @with_nsw_nuw(i8 %x, i8 %y) {
680 ; CHECK-LABEL: @with_nsw_nuw(
681 ; CHECK-NEXT:    [[TMP1:%.*]] = add nuw nsw i8 [[X:%.*]], 2
682 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[Y:%.*]]
683 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
685   %t1 = add nsw nuw i8 %x, 37
686   %t2 = add i8 %y, 35
687   %tobool = icmp eq i8 %t2, %t1
688   ret i1 %tobool
691 define i1 @with_nsw_large(i8 %x, i8 %y) {
692 ; CHECK-LABEL: @with_nsw_large(
693 ; CHECK-NEXT:    [[TMP1:%.*]] = add nsw i8 [[X:%.*]], 2
694 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[Y:%.*]]
695 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
697   %t1 = add nsw i8 %x, 37
698   %t2 = add i8 %y, 35
699   %tobool = icmp eq i8 %t2, %t1
700   ret i1 %tobool
703 define i1 @with_nsw_small(i8 %x, i8 %y) {
704 ; CHECK-LABEL: @with_nsw_small(
705 ; CHECK-NEXT:    [[TMP1:%.*]] = add i8 [[Y:%.*]], 2
706 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[X:%.*]]
707 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
709   %t1 = add nsw i8 %x, 35
710   %t2 = add i8 %y, 37
711   %tobool = icmp eq i8 %t2, %t1
712   ret i1 %tobool
715 define i1 @with_nuw_large(i8 %x, i8 %y) {
716 ; CHECK-LABEL: @with_nuw_large(
717 ; CHECK-NEXT:    [[TMP1:%.*]] = add nuw i8 [[X:%.*]], 2
718 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[Y:%.*]]
719 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
721   %t1 = add nuw i8 %x, 37
722   %t2 = add i8 %y, 35
723   %tobool = icmp eq i8 %t2, %t1
724   ret i1 %tobool
727 define i1 @with_nuw_small(i8 %x, i8 %y) {
728 ; CHECK-LABEL: @with_nuw_small(
729 ; CHECK-NEXT:    [[TMP1:%.*]] = add i8 [[Y:%.*]], 2
730 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[X:%.*]]
731 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
733   %t1 = add nuw i8 %x, 35
734   %t2 = add i8 %y, 37
735   %tobool = icmp eq i8 %t2, %t1
736   ret i1 %tobool
739 define i1 @with_nuw_large_negative(i8 %x, i8 %y) {
740 ; CHECK-LABEL: @with_nuw_large_negative(
741 ; CHECK-NEXT:    [[TMP1:%.*]] = add i8 [[X:%.*]], -2
742 ; CHECK-NEXT:    [[TOBOOL:%.*]] = icmp eq i8 [[TMP1]], [[Y:%.*]]
743 ; CHECK-NEXT:    ret i1 [[TOBOOL]]
745   %t1 = add nuw i8 %x, -37
746   %t2 = add i8 %y, -35
747   %tobool = icmp eq i8 %t2, %t1
748   ret i1 %tobool
751 define i1 @ugt_offset(i8 %a) {
752 ; CHECK-LABEL: @ugt_offset(
753 ; CHECK-NEXT:    [[OV:%.*]] = icmp slt i8 [[A:%.*]], -124
754 ; CHECK-NEXT:    ret i1 [[OV]]
756   %t = add i8 %a, 124
757   %ov = icmp ugt i8 %t, 251
758   ret i1 %ov
761 define i1 @ugt_offset_use(i32 %a) {
762 ; CHECK-LABEL: @ugt_offset_use(
763 ; CHECK-NEXT:    [[T:%.*]] = add i32 [[A:%.*]], 42
764 ; CHECK-NEXT:    call void @use(i32 [[T]])
765 ; CHECK-NEXT:    [[OV:%.*]] = icmp slt i32 [[A]], -42
766 ; CHECK-NEXT:    ret i1 [[OV]]
768   %t = add i32 %a, 42
769   call void @use(i32 %t)
770   %ov = icmp ugt i32 %t, 2147483689
771   ret i1 %ov
774 define <2 x i1> @ugt_offset_splat(<2 x i5> %a) {
775 ; CHECK-LABEL: @ugt_offset_splat(
776 ; CHECK-NEXT:    [[OV:%.*]] = icmp slt <2 x i5> [[A:%.*]], <i5 -9, i5 -9>
777 ; CHECK-NEXT:    ret <2 x i1> [[OV]]
779   %t = add <2 x i5> %a, <i5 9, i5 9>
780   %ov = icmp ugt <2 x i5> %t, <i5 24, i5 24>
781   ret <2 x i1> %ov
784 ; negative test - constants must differ by SMAX
786 define i1 @ugt_wrong_offset(i8 %a) {
787 ; CHECK-LABEL: @ugt_wrong_offset(
788 ; CHECK-NEXT:    [[T:%.*]] = add i8 [[A:%.*]], 123
789 ; CHECK-NEXT:    [[OV:%.*]] = icmp ugt i8 [[T]], -5
790 ; CHECK-NEXT:    ret i1 [[OV]]
792   %t = add i8 %a, 123
793   %ov = icmp ugt i8 %t, 251
794   ret i1 %ov
797 define i1 @ugt_offset_nuw(i8 %a) {
798 ; CHECK-LABEL: @ugt_offset_nuw(
799 ; CHECK-NEXT:    [[OV:%.*]] = icmp slt i8 [[A:%.*]], 0
800 ; CHECK-NEXT:    ret i1 [[OV]]
802   %t = add nuw i8 %a, 124
803   %ov = icmp ugt i8 %t, 251
804   ret i1 %ov
807 define i1 @ult_offset(i8 %a) {
808 ; CHECK-LABEL: @ult_offset(
809 ; CHECK-NEXT:    [[OV:%.*]] = icmp sgt i8 [[A:%.*]], 5
810 ; CHECK-NEXT:    ret i1 [[OV]]
812   %t = add i8 %a, 250
813   %ov = icmp ult i8 %t, 122
814   ret i1 %ov
817 define i1 @ult_offset_use(i32 %a) {
818 ; CHECK-LABEL: @ult_offset_use(
819 ; CHECK-NEXT:    [[T:%.*]] = add i32 [[A:%.*]], 42
820 ; CHECK-NEXT:    call void @use(i32 [[T]])
821 ; CHECK-NEXT:    [[OV:%.*]] = icmp sgt i32 [[A]], -43
822 ; CHECK-NEXT:    ret i1 [[OV]]
824   %t = add i32 %a, 42
825   call void @use(i32 %t)
826   %ov = icmp ult i32 %t, 2147483690
827   ret i1 %ov
830 define <2 x i1> @ult_offset_splat(<2 x i5> %a) {
831 ; CHECK-LABEL: @ult_offset_splat(
832 ; CHECK-NEXT:    [[OV:%.*]] = icmp sgt <2 x i5> [[A:%.*]], <i5 -10, i5 -10>
833 ; CHECK-NEXT:    ret <2 x i1> [[OV]]
835   %t = add <2 x i5> %a, <i5 9, i5 9>
836   %ov = icmp ult <2 x i5> %t, <i5 25, i5 25>
837   ret <2 x i1> %ov
840 ; negative test - constants must differ by SMIN
842 define i1 @ult_wrong_offset(i8 %a) {
843 ; CHECK-LABEL: @ult_wrong_offset(
844 ; CHECK-NEXT:    [[T:%.*]] = add i8 [[A:%.*]], -6
845 ; CHECK-NEXT:    [[OV:%.*]] = icmp ult i8 [[T]], 123
846 ; CHECK-NEXT:    ret i1 [[OV]]
848   %t = add i8 %a, 250
849   %ov = icmp ult i8 %t, 123
850   ret i1 %ov
853 define i1 @ult_offset_nuw(i8 %a) {
854 ; CHECK-LABEL: @ult_offset_nuw(
855 ; CHECK-NEXT:    [[OV:%.*]] = icmp sgt i8 [[A:%.*]], -1
856 ; CHECK-NEXT:    ret i1 [[OV]]
858   %t = add nuw i8 %a, 42
859   %ov = icmp ult i8 %t, 170
860   ret i1 %ov
863 define i1 @sgt_offset(i8 %a) {
864 ; CHECK-LABEL: @sgt_offset(
865 ; CHECK-NEXT:    [[OV:%.*]] = icmp ult i8 [[A:%.*]], -122
866 ; CHECK-NEXT:    ret i1 [[OV]]
868   %t = add i8 %a, -6
869   %ov = icmp sgt i8 %t, -7
870   ret i1 %ov
873 define i1 @sgt_offset_use(i32 %a) {
874 ; CHECK-LABEL: @sgt_offset_use(
875 ; CHECK-NEXT:    [[T:%.*]] = add i32 [[A:%.*]], 42
876 ; CHECK-NEXT:    call void @use(i32 [[T]])
877 ; CHECK-NEXT:    [[OV:%.*]] = icmp ult i32 [[A]], 2147483606
878 ; CHECK-NEXT:    ret i1 [[OV]]
880   %t = add i32 %a, 42
881   call void @use(i32 %t)
882   %ov = icmp sgt i32 %t, 41
883   ret i1 %ov
886 define <2 x i1> @sgt_offset_splat(<2 x i5> %a) {
887 ; CHECK-LABEL: @sgt_offset_splat(
888 ; CHECK-NEXT:    [[OV:%.*]] = icmp ult <2 x i5> [[A:%.*]], <i5 7, i5 7>
889 ; CHECK-NEXT:    ret <2 x i1> [[OV]]
891   %t = add <2 x i5> %a, <i5 9, i5 9>
892   %ov = icmp sgt <2 x i5> %t, <i5 8, i5 8>
893   ret <2 x i1> %ov
896 ; negative test - constants must differ by 1
898 define i1 @sgt_wrong_offset(i8 %a) {
899 ; CHECK-LABEL: @sgt_wrong_offset(
900 ; CHECK-NEXT:    [[T:%.*]] = add i8 [[A:%.*]], -7
901 ; CHECK-NEXT:    [[OV:%.*]] = icmp sgt i8 [[T]], -7
902 ; CHECK-NEXT:    ret i1 [[OV]]
904   %t = add i8 %a, -7
905   %ov = icmp sgt i8 %t, -7
906   ret i1 %ov
909 define i1 @sgt_offset_nsw(i8 %a, i8 %c) {
910 ; CHECK-LABEL: @sgt_offset_nsw(
911 ; CHECK-NEXT:    [[OV:%.*]] = icmp sgt i8 [[A:%.*]], -1
912 ; CHECK-NEXT:    ret i1 [[OV]]
914   %t = add nsw i8 %a, 42
915   %ov = icmp sgt i8 %t, 41
916   ret i1 %ov
919 define i1 @slt_offset(i8 %a) {
920 ; CHECK-LABEL: @slt_offset(
921 ; CHECK-NEXT:    [[OV:%.*]] = icmp ugt i8 [[A:%.*]], -123
922 ; CHECK-NEXT:    ret i1 [[OV]]
924   %t = add i8 %a, -6
925   %ov = icmp slt i8 %t, -6
926   ret i1 %ov
929 define i1 @slt_offset_use(i32 %a) {
930 ; CHECK-LABEL: @slt_offset_use(
931 ; CHECK-NEXT:    [[T:%.*]] = add i32 [[A:%.*]], 42
932 ; CHECK-NEXT:    call void @use(i32 [[T]])
933 ; CHECK-NEXT:    [[OV:%.*]] = icmp ugt i32 [[A]], 2147483605
934 ; CHECK-NEXT:    ret i1 [[OV]]
936   %t = add i32 %a, 42
937   call void @use(i32 %t)
938   %ov = icmp slt i32 %t, 42
939   ret i1 %ov
942 define <2 x i1> @slt_offset_splat(<2 x i5> %a) {
943 ; CHECK-LABEL: @slt_offset_splat(
944 ; CHECK-NEXT:    [[OV:%.*]] = icmp ugt <2 x i5> [[A:%.*]], <i5 6, i5 6>
945 ; CHECK-NEXT:    ret <2 x i1> [[OV]]
947   %t = add <2 x i5> %a, <i5 9, i5 9>
948   %ov = icmp slt <2 x i5> %t, <i5 9, i5 9>
949   ret <2 x i1> %ov
952 ; negative test - constants must be equal
954 define i1 @slt_wrong_offset(i8 %a) {
955 ; CHECK-LABEL: @slt_wrong_offset(
956 ; CHECK-NEXT:    [[T:%.*]] = add i8 [[A:%.*]], -6
957 ; CHECK-NEXT:    [[OV:%.*]] = icmp slt i8 [[T]], -7
958 ; CHECK-NEXT:    ret i1 [[OV]]
960   %t = add i8 %a, -6
961   %ov = icmp slt i8 %t, -7
962   ret i1 %ov
965 define i1 @slt_offset_nsw(i8 %a, i8 %c) {
966 ; CHECK-LABEL: @slt_offset_nsw(
967 ; CHECK-NEXT:    [[OV:%.*]] = icmp slt i8 [[A:%.*]], 0
968 ; CHECK-NEXT:    ret i1 [[OV]]
970   %t = add nsw i8 %a, 42
971   %ov = icmp slt i8 %t, 42
972   ret i1 %ov
975 ; In the following 4 tests, we could push the inc/dec
976 ; through the min/max, but we should not break up the
977 ; min/max idiom by using different icmp and select
978 ; operands.
980 define i32 @increment_max(i32 %x) {
981 ; CHECK-LABEL: @increment_max(
982 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp sgt i32 [[X:%.*]], -1
983 ; CHECK-NEXT:    [[TMP2:%.*]] = select i1 [[TMP1]], i32 [[X]], i32 -1
984 ; CHECK-NEXT:    [[S:%.*]] = add nsw i32 [[TMP2]], 1
985 ; CHECK-NEXT:    ret i32 [[S]]
987   %a = add nsw i32 %x, 1
988   %c = icmp sgt i32 %a, 0
989   %s = select i1 %c, i32 %a, i32 0
990   ret i32 %s
993 define i32 @decrement_max(i32 %x) {
994 ; CHECK-LABEL: @decrement_max(
995 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp sgt i32 [[X:%.*]], 1
996 ; CHECK-NEXT:    [[TMP2:%.*]] = select i1 [[TMP1]], i32 [[X]], i32 1
997 ; CHECK-NEXT:    [[S:%.*]] = add nsw i32 [[TMP2]], -1
998 ; CHECK-NEXT:    ret i32 [[S]]
1000   %a = add nsw i32 %x, -1
1001   %c = icmp sgt i32 %a, 0
1002   %s = select i1 %c, i32 %a, i32 0
1003   ret i32 %s
1006 define i32 @increment_min(i32 %x) {
1007 ; CHECK-LABEL: @increment_min(
1008 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp slt i32 [[X:%.*]], -1
1009 ; CHECK-NEXT:    [[TMP2:%.*]] = select i1 [[TMP1]], i32 [[X]], i32 -1
1010 ; CHECK-NEXT:    [[S:%.*]] = add nsw i32 [[TMP2]], 1
1011 ; CHECK-NEXT:    ret i32 [[S]]
1013   %a = add nsw i32 %x, 1
1014   %c = icmp slt i32 %a, 0
1015   %s = select i1 %c, i32 %a, i32 0
1016   ret i32 %s
1019 define i32 @decrement_min(i32 %x) {
1020 ; CHECK-LABEL: @decrement_min(
1021 ; CHECK-NEXT:    [[TMP1:%.*]] = icmp slt i32 [[X:%.*]], 1
1022 ; CHECK-NEXT:    [[TMP2:%.*]] = select i1 [[TMP1]], i32 [[X]], i32 1
1023 ; CHECK-NEXT:    [[S:%.*]] = add nsw i32 [[TMP2]], -1
1024 ; CHECK-NEXT:    ret i32 [[S]]
1026   %a = add nsw i32 %x, -1
1027   %c = icmp slt i32 %a, 0
1028   %s = select i1 %c, i32 %a, i32 0
1029   ret i32 %s