[clang][modules] Don't prevent translation of FW_Private includes when explicitly...
[llvm-project.git] / llvm / test / Transforms / CorrelatedValuePropagation / range.ll
blob5c10fdcd2b492179697e2189b4c8ce39f2b9d581
1 ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2 ; RUN: opt -passes=correlated-propagation -S < %s | FileCheck %s
4 declare i32 @foo()
6 define i32 @test1(i32 %a) nounwind {
7 ; CHECK-LABEL: @test1(
8 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
9 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A_OFF]], 8
10 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
11 ; CHECK:       then:
12 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
13 ; CHECK:       else:
14 ; CHECK-NEXT:    ret i32 1
15 ; CHECK:       end:
16 ; CHECK-NEXT:    ret i32 2
18   %a.off = add i32 %a, -8
19   %cmp = icmp ult i32 %a.off, 8
20   br i1 %cmp, label %then, label %else
22 then:
23   %dead = icmp eq i32 %a, 7
24   br i1 %dead, label %end, label %else
26 else:
27   ret i32 1
29 end:
30   ret i32 2
33 define i32 @test2(i32 %a) nounwind {
34 ; CHECK-LABEL: @test2(
35 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
36 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A_OFF]], 8
37 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
38 ; CHECK:       then:
39 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
40 ; CHECK:       else:
41 ; CHECK-NEXT:    ret i32 1
42 ; CHECK:       end:
43 ; CHECK-NEXT:    ret i32 2
45   %a.off = add i32 %a, -8
46   %cmp = icmp ult i32 %a.off, 8
47   br i1 %cmp, label %then, label %else
49 then:
50   %dead = icmp ugt i32 %a, 15
51   br i1 %dead, label %end, label %else
53 else:
54   ret i32 1
56 end:
57   ret i32 2
60 define i32 @test3(i32 %c) nounwind {
61 ; CHECK-LABEL: @test3(
62 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt i32 [[C:%.*]], 2
63 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF_THEN:%.*]], label [[IF_END:%.*]]
64 ; CHECK:       if.then:
65 ; CHECK-NEXT:    ret i32 1
66 ; CHECK:       if.end:
67 ; CHECK-NEXT:    [[CMP1:%.*]] = icmp ult i32 [[C]], 3
68 ; CHECK-NEXT:    br i1 [[CMP1]], label [[IF_THEN2:%.*]], label [[IF_END8:%.*]]
69 ; CHECK:       if.then2:
70 ; CHECK-NEXT:    br i1 true, label [[IF_THEN4:%.*]], label [[IF_END6:%.*]]
71 ; CHECK:       if.end6:
72 ; CHECK-NEXT:    ret i32 2
73 ; CHECK:       if.then4:
74 ; CHECK-NEXT:    ret i32 3
75 ; CHECK:       if.end8:
76 ; CHECK-NEXT:    ret i32 4
78   %cmp = icmp slt i32 %c, 2
79   br i1 %cmp, label %if.then, label %if.end
81 if.then:
82   ret i32 1
84 if.end:
85   %cmp1 = icmp slt i32 %c, 3
86   br i1 %cmp1, label %if.then2, label %if.end8
88 if.then2:
89   %cmp2 = icmp eq i32 %c, 2
90   br i1 %cmp2, label %if.then4, label %if.end6
92 if.end6:
93   ret i32 2
95 if.then4:
96   ret i32 3
98 if.end8:
99   ret i32 4
102 define i32 @test4(i32 %c) nounwind {
103 ; CHECK-LABEL: @test4(
104 ; CHECK-NEXT:    switch i32 [[C:%.*]], label [[SW_DEFAULT:%.*]] [
105 ; CHECK-NEXT:    i32 1, label [[SW_BB:%.*]]
106 ; CHECK-NEXT:    i32 2, label [[SW_BB]]
107 ; CHECK-NEXT:    i32 4, label [[SW_BB]]
108 ; CHECK-NEXT:    ]
109 ; CHECK:       sw.bb:
110 ; CHECK-NEXT:    br i1 true, label [[IF_THEN:%.*]], label [[IF_END:%.*]]
111 ; CHECK:       if.then:
112 ; CHECK-NEXT:    br label [[RETURN:%.*]]
113 ; CHECK:       if.end:
114 ; CHECK-NEXT:    br label [[RETURN]]
115 ; CHECK:       sw.default:
116 ; CHECK-NEXT:    br label [[RETURN]]
117 ; CHECK:       return:
118 ; CHECK-NEXT:    [[RETVAL_0:%.*]] = phi i32 [ 42, [[SW_DEFAULT]] ], [ 4, [[IF_THEN]] ], [ 9, [[IF_END]] ]
119 ; CHECK-NEXT:    ret i32 [[RETVAL_0]]
121   switch i32 %c, label %sw.default [
122   i32 1, label %sw.bb
123   i32 2, label %sw.bb
124   i32 4, label %sw.bb
125   ]
127 sw.bb:
128   %cmp = icmp sge i32 %c, 1
129   br i1 %cmp, label %if.then, label %if.end
131 if.then:
132   br label %return
134 if.end:
135   br label %return
137 sw.default:
138   br label %return
140 return:
141   %retval.0 = phi i32 [ 42, %sw.default ], [ 4, %if.then ], [ 9, %if.end ]
142   ret i32 %retval.0
145 define i1 @test5(i32 %c) nounwind {
146 ; CHECK-LABEL: @test5(
147 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt i32 [[C:%.*]], 5
148 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF_THEN:%.*]], label [[IF_END:%.*]]
149 ; CHECK:       if.then:
150 ; CHECK-NEXT:    [[CMP1:%.*]] = icmp eq i32 [[C]], 4
151 ; CHECK-NEXT:    br i1 [[CMP1]], label [[IF_END]], label [[IF_END8:%.*]]
152 ; CHECK:       if.end:
153 ; CHECK-NEXT:    ret i1 true
154 ; CHECK:       if.end8:
155 ; CHECK-NEXT:    [[CMP2:%.*]] = icmp eq i32 [[C]], 3
156 ; CHECK-NEXT:    [[OR:%.*]] = or i1 false, false
157 ; CHECK-NEXT:    ret i1 [[CMP2]]
159   %cmp = icmp slt i32 %c, 5
160   br i1 %cmp, label %if.then, label %if.end
162 if.then:
163   %cmp1 = icmp eq i32 %c, 4
164   br i1 %cmp1, label %if.end, label %if.end8
166 if.end:
167   ret i1 true
169 if.end8:
170   %cmp2 = icmp eq i32 %c, 3
171   %cmp3 = icmp eq i32 %c, 4
172   %cmp4 = icmp eq i32 %c, 6
173   %or = or i1 %cmp3, %cmp4
174   ret i1 %cmp2
177 define i1 @test6(i32 %c) nounwind {
178 ; CHECK-LABEL: @test6(
179 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ule i32 [[C:%.*]], 7
180 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF_THEN:%.*]], label [[IF_END:%.*]]
181 ; CHECK:       if.then:
182 ; CHECK-NEXT:    [[COND:%.*]] = icmp eq i32 [[C]], 6
183 ; CHECK-NEXT:    br i1 [[COND]], label [[SW_BB:%.*]], label [[IF_END]]
184 ; CHECK:       if.end:
185 ; CHECK-NEXT:    ret i1 true
186 ; CHECK:       sw.bb:
187 ; CHECK-NEXT:    ret i1 true
189   %cmp = icmp ule i32 %c, 7
190   br i1 %cmp, label %if.then, label %if.end
192 if.then:
193   switch i32 %c, label %if.end [
194   i32 6, label %sw.bb
195   i32 8, label %sw.bb
196   ]
198 if.end:
199   ret i1 true
201 sw.bb:
202   %cmp2 = icmp eq i32 %c, 6
203   ret i1 %cmp2
206 define i1 @test7(i32 %c) nounwind {
207 ; CHECK-LABEL: @test7(
208 ; CHECK-NEXT:  entry:
209 ; CHECK-NEXT:    switch i32 [[C:%.*]], label [[SW_DEFAULT:%.*]] [
210 ; CHECK-NEXT:    i32 6, label [[SW_BB:%.*]]
211 ; CHECK-NEXT:    i32 7, label [[SW_BB]]
212 ; CHECK-NEXT:    ]
213 ; CHECK:       sw.bb:
214 ; CHECK-NEXT:    ret i1 true
215 ; CHECK:       sw.default:
216 ; CHECK-NEXT:    [[CMP5:%.*]] = icmp eq i32 [[C]], 5
217 ; CHECK-NEXT:    [[CMP8:%.*]] = icmp eq i32 [[C]], 8
218 ; CHECK-NEXT:    [[OR:%.*]] = or i1 [[CMP5]], false
219 ; CHECK-NEXT:    [[OR2:%.*]] = or i1 false, [[CMP8]]
220 ; CHECK-NEXT:    ret i1 false
222 entry:
223   switch i32 %c, label %sw.default [
224   i32 6, label %sw.bb
225   i32 7, label %sw.bb
226   ]
228 sw.bb:
229   ret i1 true
231 sw.default:
232   %cmp5 = icmp eq i32 %c, 5
233   %cmp6 = icmp eq i32 %c, 6
234   %cmp7 = icmp eq i32 %c, 7
235   %cmp8 = icmp eq i32 %c, 8
236   %or = or i1 %cmp5, %cmp6
237   %or2 = or i1 %cmp7, %cmp8
238   ret i1 false
241 define i1 @test8(ptr %p) {
242 ; CHECK-LABEL: @test8(
243 ; CHECK-NEXT:    [[A:%.*]] = load i64, ptr [[P:%.*]], align 4, !range [[RNG0:![0-9]+]]
244 ; CHECK-NEXT:    ret i1 false
246   %a = load i64, i64* %p, !range !{i64 4, i64 255}
247   %res = icmp eq i64 %a, 0
248   ret i1 %res
251 define i1 @test9(ptr %p) {
252 ; CHECK-LABEL: @test9(
253 ; CHECK-NEXT:    [[A:%.*]] = load i64, ptr [[P:%.*]], align 4, !range [[RNG1:![0-9]+]]
254 ; CHECK-NEXT:    ret i1 true
256   %a = load i64, i64* %p, !range !{i64 0, i64 1}
257   %res = icmp eq i64 %a, 0
258   ret i1 %res
261 define i1 @test10(ptr %p) {
262 ; CHECK-LABEL: @test10(
263 ; CHECK-NEXT:    [[A:%.*]] = load i64, ptr [[P:%.*]], align 4, !range [[RNG2:![0-9]+]]
264 ; CHECK-NEXT:    ret i1 false
266   %a = load i64, i64* %p, !range !{i64 4, i64 8, i64 15, i64 20}
267   %res = icmp eq i64 %a, 0
268   ret i1 %res
271 @g = external global i32
273 define i1 @test11() {
274 ; CHECK-LABEL: @test11(
275 ; CHECK-NEXT:    [[POSITIVE:%.*]] = load i32, ptr @g, align 4, !range [[RNG3:![0-9]+]]
276 ; CHECK-NEXT:    [[ADD:%.*]] = add nuw nsw i32 [[POSITIVE]], 1
277 ; CHECK-NEXT:    br label [[NEXT:%.*]]
278 ; CHECK:       next:
279 ; CHECK-NEXT:    ret i1 true
281   %positive = load i32, i32* @g, !range !{i32 1, i32 2048}
282   %add = add i32 %positive, 1
283   %test = icmp sgt i32 %add, 0
284   br label %next
286 next:
287   ret i1 %test
290 define i32 @test12(i32 %a, i32 %b) {
291 ; CHECK-LABEL: @test12(
292 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A:%.*]], [[B:%.*]]
293 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
294 ; CHECK:       then:
295 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
296 ; CHECK:       else:
297 ; CHECK-NEXT:    ret i32 1
298 ; CHECK:       end:
299 ; CHECK-NEXT:    ret i32 2
301   %cmp = icmp ult i32 %a, %b
302   br i1 %cmp, label %then, label %else
304 then:
305   %dead = icmp eq i32 %a, -1
306   br i1 %dead, label %end, label %else
308 else:
309   ret i32 1
311 end:
312   ret i32 2
315 define i32 @test12_swap(i32 %a, i32 %b) {
316 ; CHECK-LABEL: @test12_swap(
317 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ugt i32 [[B:%.*]], [[A:%.*]]
318 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
319 ; CHECK:       then:
320 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
321 ; CHECK:       else:
322 ; CHECK-NEXT:    ret i32 1
323 ; CHECK:       end:
324 ; CHECK-NEXT:    ret i32 2
326   %cmp = icmp ugt i32 %b, %a
327   br i1 %cmp, label %then, label %else
329 then:
330   %dead = icmp eq i32 %a, -1
331   br i1 %dead, label %end, label %else
333 else:
334   ret i32 1
336 end:
337   ret i32 2
340 ; The same as @test12 but the second check is on the false path
342 define i32 @test12_neg(i32 %a, i32 %b) {
343 ; CHECK-LABEL: @test12_neg(
344 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A:%.*]], [[B:%.*]]
345 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
346 ; CHECK:       else:
347 ; CHECK-NEXT:    [[ALIVE:%.*]] = icmp eq i32 [[A]], -1
348 ; CHECK-NEXT:    br i1 [[ALIVE]], label [[END:%.*]], label [[THEN]]
349 ; CHECK:       then:
350 ; CHECK-NEXT:    ret i32 1
351 ; CHECK:       end:
352 ; CHECK-NEXT:    ret i32 2
354   %cmp = icmp ult i32 %a, %b
355   br i1 %cmp, label %then, label %else
357 else:
358   %alive = icmp eq i32 %a, -1
359   br i1 %alive, label %end, label %then
361 then:
362   ret i32 1
364 end:
365   ret i32 2
368 ; The same as @test12 but with signed comparison
370 define i32 @test12_signed(i32 %a, i32 %b) {
371 ; CHECK-LABEL: @test12_signed(
372 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt i32 [[A:%.*]], [[B:%.*]]
373 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
374 ; CHECK:       then:
375 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
376 ; CHECK:       else:
377 ; CHECK-NEXT:    ret i32 1
378 ; CHECK:       end:
379 ; CHECK-NEXT:    ret i32 2
381   %cmp = icmp slt i32 %a, %b
382   br i1 %cmp, label %then, label %else
384 then:
385   %dead = icmp eq i32 %a, 2147483647
386   br i1 %dead, label %end, label %else
388 else:
389   ret i32 1
391 end:
392   ret i32 2
395 define i32 @test13(i32 %a, i32 %b) {
396 ; CHECK-LABEL: @test13(
397 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
398 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A_OFF]], [[B:%.*]]
399 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
400 ; CHECK:       then:
401 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
402 ; CHECK:       else:
403 ; CHECK-NEXT:    ret i32 1
404 ; CHECK:       end:
405 ; CHECK-NEXT:    ret i32 2
407   %a.off = add i32 %a, -8
408   %cmp = icmp ult i32 %a.off, %b
409   br i1 %cmp, label %then, label %else
411 then:
412   %dead = icmp eq i32 %a, 7
413   br i1 %dead, label %end, label %else
415 else:
416   ret i32 1
418 end:
419   ret i32 2
422 define i32 @test13_swap(i32 %a, i32 %b) {
423 ; CHECK-LABEL: @test13_swap(
424 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
425 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ugt i32 [[B:%.*]], [[A_OFF]]
426 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
427 ; CHECK:       then:
428 ; CHECK-NEXT:    br i1 false, label [[END:%.*]], label [[ELSE]]
429 ; CHECK:       else:
430 ; CHECK-NEXT:    ret i32 1
431 ; CHECK:       end:
432 ; CHECK-NEXT:    ret i32 2
434   %a.off = add i32 %a, -8
435   %cmp = icmp ugt i32 %b, %a.off
436   br i1 %cmp, label %then, label %else
438 then:
439   %dead = icmp eq i32 %a, 7
440   br i1 %dead, label %end, label %else
442 else:
443   ret i32 1
445 end:
446   ret i32 2
449 define i1 @test14_slt(i32 %a) {
450 ; CHECK-LABEL: @test14_slt(
451 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
452 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt i32 [[A_OFF]], 8
453 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
454 ; CHECK:       then:
455 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, false
456 ; CHECK-NEXT:    ret i1 false
457 ; CHECK:       else:
458 ; CHECK-NEXT:    ret i1 false
460   %a.off = add i32 %a, -8
461   %cmp = icmp slt i32 %a.off, 8
462   br i1 %cmp, label %then, label %else
464 then:
465   %dead.1 = icmp eq i32 %a, -2147483641
466   %dead.2 = icmp eq i32 %a, 16
467   %result = or i1 %dead.1, %dead.2
468   ret i1 %result
470 else:
471   ret i1 false
474 define i1 @test14_sle(i32 %a) {
475 ; CHECK-LABEL: @test14_sle(
476 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
477 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sle i32 [[A_OFF]], 8
478 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
479 ; CHECK:       then:
480 ; CHECK-NEXT:    [[ALIVE:%.*]] = icmp eq i32 [[A]], 16
481 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, [[ALIVE]]
482 ; CHECK-NEXT:    ret i1 [[RESULT]]
483 ; CHECK:       else:
484 ; CHECK-NEXT:    ret i1 false
486   %a.off = add i32 %a, -8
487   %cmp = icmp sle i32 %a.off, 8
488   br i1 %cmp, label %then, label %else
490 then:
491   %dead = icmp eq i32 %a, -2147483641
492   %alive = icmp eq i32 %a, 16
493   %result = or i1 %dead, %alive
494   ret i1 %result
496 else:
497   ret i1 false
500 define i1 @test14_sgt(i32 %a) {
501 ; CHECK-LABEL: @test14_sgt(
502 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
503 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sgt i32 [[A_OFF]], 8
504 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
505 ; CHECK:       then:
506 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, false
507 ; CHECK-NEXT:    ret i1 false
508 ; CHECK:       else:
509 ; CHECK-NEXT:    ret i1 false
511   %a.off = add i32 %a, -8
512   %cmp = icmp sgt i32 %a.off, 8
513   br i1 %cmp, label %then, label %else
515 then:
516   %dead.1 = icmp eq i32 %a, -2147483640
517   %dead.2 = icmp eq i32 %a, 16
518   %result = or i1 %dead.1, %dead.2
519   ret i1 %result
521 else:
522   ret i1 false
525 define i1 @test14_sge(i32 %a) {
526 ; CHECK-LABEL: @test14_sge(
527 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
528 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sge i32 [[A_OFF]], 8
529 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
530 ; CHECK:       then:
531 ; CHECK-NEXT:    [[ALIVE:%.*]] = icmp eq i32 [[A]], 16
532 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, [[ALIVE]]
533 ; CHECK-NEXT:    ret i1 [[RESULT]]
534 ; CHECK:       else:
535 ; CHECK-NEXT:    ret i1 false
537   %a.off = add i32 %a, -8
538   %cmp = icmp sge i32 %a.off, 8
539   br i1 %cmp, label %then, label %else
541 then:
542   %dead = icmp eq i32 %a, -2147483640
543   %alive = icmp eq i32 %a, 16
544   %result = or i1 %dead, %alive
545   ret i1 %result
547 else:
548   ret i1 false
551 define i1 @test14_ule(i32 %a) {
552 ; CHECK-LABEL: @test14_ule(
553 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
554 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ule i32 [[A_OFF]], 8
555 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
556 ; CHECK:       then:
557 ; CHECK-NEXT:    [[ALIVE:%.*]] = icmp eq i32 [[A]], 16
558 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, [[ALIVE]]
559 ; CHECK-NEXT:    ret i1 [[RESULT]]
560 ; CHECK:       else:
561 ; CHECK-NEXT:    ret i1 false
563   %a.off = add i32 %a, -8
564   %cmp = icmp ule i32 %a.off, 8
565   br i1 %cmp, label %then, label %else
567 then:
568   %dead = icmp eq i32 %a, 7
569   %alive = icmp eq i32 %a, 16
570   %result = or i1 %dead, %alive
571   ret i1 %result
573 else:
574   ret i1 false
577 define i1 @test14_ugt(i32 %a) {
578 ; CHECK-LABEL: @test14_ugt(
579 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
580 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ugt i32 [[A_OFF]], 8
581 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
582 ; CHECK:       then:
583 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, false
584 ; CHECK-NEXT:    ret i1 false
585 ; CHECK:       else:
586 ; CHECK-NEXT:    ret i1 false
588   %a.off = add i32 %a, -8
589   %cmp = icmp ugt i32 %a.off, 8
590   br i1 %cmp, label %then, label %else
592 then:
593   %dead.1 = icmp eq i32 %a, 8
594   %dead.2 = icmp eq i32 %a, 16
595   %result = or i1 %dead.1, %dead.2
596   ret i1 %result
598 else:
599   ret i1 false
602 define i1 @test14_uge(i32 %a) {
603 ; CHECK-LABEL: @test14_uge(
604 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
605 ; CHECK-NEXT:    [[CMP:%.*]] = icmp uge i32 [[A_OFF]], 8
606 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
607 ; CHECK:       then:
608 ; CHECK-NEXT:    [[ALIVE:%.*]] = icmp eq i32 [[A]], 16
609 ; CHECK-NEXT:    [[RESULT:%.*]] = or i1 false, [[ALIVE]]
610 ; CHECK-NEXT:    ret i1 [[RESULT]]
611 ; CHECK:       else:
612 ; CHECK-NEXT:    ret i1 false
614   %a.off = add i32 %a, -8
615   %cmp = icmp uge i32 %a.off, 8
616   br i1 %cmp, label %then, label %else
618 then:
619   %dead = icmp eq i32 %a, 8
620   %alive = icmp eq i32 %a, 16
621   %result = or i1 %dead, %alive
622   ret i1 %result
624 else:
625   ret i1 false
628 define i1 @test14_ugt_and(i32 %a) {
629 ; CHECK-LABEL: @test14_ugt_and(
630 ; CHECK-NEXT:    [[A_OFF:%.*]] = add i32 [[A:%.*]], -8
631 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ugt i32 [[A_OFF]], 8
632 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
633 ; CHECK:       then:
634 ; CHECK-NEXT:    [[RESULT:%.*]] = and i1 false, false
635 ; CHECK-NEXT:    ret i1 false
636 ; CHECK:       else:
637 ; CHECK-NEXT:    ret i1 false
639   %a.off = add i32 %a, -8
640   %cmp = icmp ugt i32 %a.off, 8
641   br i1 %cmp, label %then, label %else
643 then:
644   %dead.1 = icmp eq i32 %a, 8
645   %dead.2 = icmp eq i32 %a, 16
646   %result = and i1 %dead.1, %dead.2
647   ret i1 %result
649 else:
650   ret i1 false
653 @limit = external global i32
654 define i1 @test15(i32 %a) {
655 ; CHECK-LABEL: @test15(
656 ; CHECK-NEXT:    [[LIMIT:%.*]] = load i32, ptr @limit, align 4, !range [[RNG4:![0-9]+]]
657 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i32 [[A:%.*]], [[LIMIT]]
658 ; CHECK-NEXT:    br i1 [[CMP]], label [[THEN:%.*]], label [[ELSE:%.*]]
659 ; CHECK:       then:
660 ; CHECK-NEXT:    ret i1 false
661 ; CHECK:       else:
662 ; CHECK-NEXT:    ret i1 false
664   %limit = load i32, i32* @limit, !range !{i32 0, i32 256}
665   %cmp = icmp ult i32 %a, %limit
666   br i1 %cmp, label %then, label %else
668 then:
669   %result = icmp eq i32 %a, 255
670   ret i1 %result
672 else:
673   ret i1 false
676 define i32 @test16(i8 %a) {
677 ; CHECK-LABEL: @test16(
678 ; CHECK-NEXT:  entry:
679 ; CHECK-NEXT:    [[B:%.*]] = zext i8 [[A:%.*]] to i32
680 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
681 ; CHECK:       dispatch:
682 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq i8 [[A]], 93
683 ; CHECK-NEXT:    br i1 [[CMP]], label [[TARGET93:%.*]], label [[DISPATCH]]
684 ; CHECK:       target93:
685 ; CHECK-NEXT:    ret i32 93
687 entry:
688   %b = zext i8 %a to i32
689   br label %dispatch
691 dispatch:
692   %cmp = icmp eq i8 %a, 93
693   br i1 %cmp, label %target93, label %dispatch
695 target93:
696   ret i32 %b
699 define i32 @test16_i1(i1 %a) {
700 ; CHECK-LABEL: @test16_i1(
701 ; CHECK-NEXT:  entry:
702 ; CHECK-NEXT:    [[B:%.*]] = zext i1 [[A:%.*]] to i32
703 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
704 ; CHECK:       dispatch:
705 ; CHECK-NEXT:    br i1 [[A]], label [[TRUE:%.*]], label [[DISPATCH]]
706 ; CHECK:       true:
707 ; CHECK-NEXT:    ret i32 1
709 entry:
710   %b = zext i1 %a to i32
711   br label %dispatch
713 dispatch:
714   br i1 %a, label %true, label %dispatch
716 true:
717   ret i32 %b
720 define i8 @test17(i8 %a) {
721 ; CHECK-LABEL: @test17(
722 ; CHECK-NEXT:  entry:
723 ; CHECK-NEXT:    [[C:%.*]] = add i8 [[A:%.*]], 3
724 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
725 ; CHECK:       dispatch:
726 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq i8 [[A]], 93
727 ; CHECK-NEXT:    br i1 [[CMP]], label [[TARGET93:%.*]], label [[DISPATCH]]
728 ; CHECK:       target93:
729 ; CHECK-NEXT:    ret i8 96
731 entry:
732   %c = add i8 %a, 3
733   br label %dispatch
735 dispatch:
736   %cmp = icmp eq i8 %a, 93
737   br i1 %cmp, label %target93, label %dispatch
739 target93:
740   ret i8 %c
743 define i8 @test17_2(i8 %a) {
744 ; CHECK-LABEL: @test17_2(
745 ; CHECK-NEXT:  entry:
746 ; CHECK-NEXT:    [[C:%.*]] = add i8 [[A:%.*]], [[A]]
747 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
748 ; CHECK:       dispatch:
749 ; CHECK-NEXT:    [[CMP:%.*]] = icmp eq i8 [[A]], 93
750 ; CHECK-NEXT:    br i1 [[CMP]], label [[TARGET93:%.*]], label [[DISPATCH]]
751 ; CHECK:       target93:
752 ; CHECK-NEXT:    ret i8 -70
754 entry:
755   %c = add i8 %a, %a
756   br label %dispatch
758 dispatch:
759   %cmp = icmp eq i8 %a, 93
760   br i1 %cmp, label %target93, label %dispatch
762 target93:
763   ret i8 %c
766 define i1 @test17_i1(i1 %a) {
767 ; CHECK-LABEL: @test17_i1(
768 ; CHECK-NEXT:  entry:
769 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
770 ; CHECK:       dispatch:
771 ; CHECK-NEXT:    br i1 [[A:%.*]], label [[TRUE:%.*]], label [[DISPATCH]]
772 ; CHECK:       true:
773 ; CHECK-NEXT:    ret i1 true
775 entry:
776   %c = and i1 %a, true
777   br label %dispatch
779 dispatch:
780   br i1 %a, label %true, label %dispatch
782 true:
783   ret i1 %c
786 define i32 @test18(i8 %a) {
787 ; CHECK-LABEL: @test18(
788 ; CHECK-NEXT:  entry:
789 ; CHECK-NEXT:    [[B:%.*]] = zext i8 [[A:%.*]] to i32
790 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
791 ; CHECK:       dispatch:
792 ; CHECK-NEXT:    switch i8 [[A]], label [[DISPATCH]] [
793 ; CHECK-NEXT:    i8 93, label [[TARGET93:%.*]]
794 ; CHECK-NEXT:    i8 -111, label [[DISPATCH]]
795 ; CHECK-NEXT:    ]
796 ; CHECK:       target93:
797 ; CHECK-NEXT:    ret i32 93
799 entry:
800   %b = zext i8 %a to i32
801   br label %dispatch
803 dispatch:
804   switch i8 %a, label %dispatch [
805   i8 93, label %target93
806   i8 -111, label %dispatch
807   ]
809 target93:
810   ret i32 %b
813 define i8 @test19(i8 %a) {
814 ; CHECK-LABEL: @test19(
815 ; CHECK-NEXT:  entry:
816 ; CHECK-NEXT:    [[C:%.*]] = add i8 [[A:%.*]], 3
817 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
818 ; CHECK:       dispatch:
819 ; CHECK-NEXT:    switch i8 [[A]], label [[DISPATCH]] [
820 ; CHECK-NEXT:    i8 93, label [[TARGET93:%.*]]
821 ; CHECK-NEXT:    i8 -111, label [[DISPATCH]]
822 ; CHECK-NEXT:    ]
823 ; CHECK:       target93:
824 ; CHECK-NEXT:    ret i8 96
826 entry:
827   %c = add i8 %a, 3
828   br label %dispatch
830 dispatch:
831   switch i8 %a, label %dispatch [
832   i8 93, label %target93
833   i8 -111, label %dispatch
834   ]
836 target93:
837   ret i8 %c
840 ; Negative test. Shouldn't be incorrectly optimized to "ret i1 false".
842 define i1 @test20(i64 %a) {
843 ; CHECK-LABEL: @test20(
844 ; CHECK-NEXT:  entry:
845 ; CHECK-NEXT:    [[B:%.*]] = and i64 [[A:%.*]], 7
846 ; CHECK-NEXT:    br label [[DISPATCH:%.*]]
847 ; CHECK:       dispatch:
848 ; CHECK-NEXT:    switch i64 [[A]], label [[DEFAULT:%.*]] [
849 ; CHECK-NEXT:    i64 0, label [[EXIT2:%.*]]
850 ; CHECK-NEXT:    i64 -2147483647, label [[EXIT2]]
851 ; CHECK-NEXT:    ]
852 ; CHECK:       default:
853 ; CHECK-NEXT:    [[C:%.*]] = icmp eq i64 [[B]], 0
854 ; CHECK-NEXT:    br label [[EXIT:%.*]]
855 ; CHECK:       exit:
856 ; CHECK-NEXT:    ret i1 [[C]]
857 ; CHECK:       exit2:
858 ; CHECK-NEXT:    ret i1 false
860 entry:
861   %b = and i64 %a, 7
862   br label %dispatch
864 dispatch:
865   switch i64 %a, label %default [
866   i64 0, label %exit2
867   i64 -2147483647, label %exit2
868   ]
870 default:
871   %c = icmp eq i64 %b, 0
872   br label %exit
874 exit:
875   ret i1 %c
877 exit2:
878   ret i1 false
881 define i1 @slt(i8 %a, i8 %b) {
882 ; CHECK-LABEL: @slt(
883 ; CHECK-NEXT:  entry:
884 ; CHECK-NEXT:    [[CMP:%.*]] = icmp slt i8 [[A:%.*]], [[B:%.*]]
885 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
886 ; CHECK-NEXT:    ret i1 true
888 entry:
889   %cmp = icmp slt i8 %a, %b
890   call void @llvm.assume(i1 %cmp)
891   %res = icmp slt i8 %a, 127
892   ret i1 %res
895 define i1 @sgt(i8 %a, i8 %b) {
896 ; CHECK-LABEL: @sgt(
897 ; CHECK-NEXT:  entry:
898 ; CHECK-NEXT:    [[CMP:%.*]] = icmp sgt i8 [[A:%.*]], [[B:%.*]]
899 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
900 ; CHECK-NEXT:    ret i1 true
902 entry:
903   %cmp = icmp sgt i8 %a, %b
904   call void @llvm.assume(i1 %cmp)
905   %res = icmp sgt i8 %a, -128
906   ret i1 %res
909 define i1 @ult(i8 %a, i8 %b) {
910 ; CHECK-LABEL: @ult(
911 ; CHECK-NEXT:  entry:
912 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i8 [[A:%.*]], [[B:%.*]]
913 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
914 ; CHECK-NEXT:    ret i1 true
916 entry:
917   %cmp = icmp ult i8 %a, %b
918   call void @llvm.assume(i1 %cmp)
919   %res = icmp ult i8 %a, 255
920   ret i1 %res
923 define i1 @ugt(i8 %a, i8 %b) {
924 ; CHECK-LABEL: @ugt(
925 ; CHECK-NEXT:  entry:
926 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ugt i8 [[A:%.*]], [[B:%.*]]
927 ; CHECK-NEXT:    call void @llvm.assume(i1 [[CMP]])
928 ; CHECK-NEXT:    ret i1 true
930 entry:
931   %cmp = icmp ugt i8 %a, %b
932   call void @llvm.assume(i1 %cmp)
933   %res = icmp ugt i8 %a, 0
934   ret i1 %res
937 define i1 @ctlz_with_range_metadata(i16 %x) {
938 ; CHECK-LABEL: @ctlz_with_range_metadata(
939 ; CHECK-NEXT:    [[CTLZ:%.*]] = call i16 @llvm.ctlz.i16(i16 [[X:%.*]], i1 false), !range [[RNG5:![0-9]+]]
940 ; CHECK-NEXT:    [[TRUNC:%.*]] = trunc i16 [[CTLZ]] to i8
941 ; CHECK-NEXT:    ret i1 true
943   %ctlz = call i16 @llvm.ctlz.i16(i16 %x, i1 false), !range !{i16 0, i16 8}
944   %trunc = trunc i16 %ctlz to i8
945   %res = icmp ult i8 %trunc, 8
946   ret i1 %res
949 define i1 @abs_with_range_metadata(i16 %x) {
950 ; CHECK-LABEL: @abs_with_range_metadata(
951 ; CHECK-NEXT:    [[ABS:%.*]] = call i16 @llvm.abs.i16(i16 [[X:%.*]], i1 false), !range [[RNG5]]
952 ; CHECK-NEXT:    [[TRUNC:%.*]] = trunc i16 [[ABS]] to i8
953 ; CHECK-NEXT:    ret i1 true
955   %abs = call i16 @llvm.abs.i16(i16 %x, i1 false), !range !{i16 0, i16 8}
956   %trunc = trunc i16 %abs to i8
957   %res = icmp ult i8 %trunc, 8
958   ret i1 %res
961 define i1 @ctlz_fold(i16 %x) {
962 ; CHECK-LABEL: @ctlz_fold(
963 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
964 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
965 ; CHECK:       if:
966 ; CHECK-NEXT:    [[CTLZ:%.*]] = call i16 @llvm.ctlz.i16(i16 [[X]], i1 false)
967 ; CHECK-NEXT:    ret i1 true
968 ; CHECK:       else:
969 ; CHECK-NEXT:    [[CTLZ2:%.*]] = call i16 @llvm.ctlz.i16(i16 [[X]], i1 false)
970 ; CHECK-NEXT:    ret i1 true
972   %cmp = icmp ult i16 %x, 256
973   br i1 %cmp, label %if, label %else
976   %ctlz = call i16 @llvm.ctlz.i16(i16 %x, i1 false)
977   %res = icmp uge i16 %ctlz, 8
978   ret i1 %res
980 else:
981   %ctlz2 = call i16 @llvm.ctlz.i16(i16 %x, i1 false)
982   %res2 = icmp ult i16 %ctlz2, 8
983   ret i1 %res2
986 define i1 @ctlz_nofold(i16 %x) {
987 ; CHECK-LABEL: @ctlz_nofold(
988 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
989 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
990 ; CHECK:       if:
991 ; CHECK-NEXT:    [[CTLZ:%.*]] = call i16 @llvm.ctlz.i16(i16 [[X]], i1 false)
992 ; CHECK-NEXT:    [[RES:%.*]] = icmp uge i16 [[CTLZ]], 9
993 ; CHECK-NEXT:    ret i1 [[RES]]
994 ; CHECK:       else:
995 ; CHECK-NEXT:    [[CTLZ2:%.*]] = call i16 @llvm.ctlz.i16(i16 [[X]], i1 false)
996 ; CHECK-NEXT:    [[RES2:%.*]] = icmp ult i16 [[CTLZ2]], 7
997 ; CHECK-NEXT:    ret i1 [[RES2]]
999   %cmp = icmp ult i16 %x, 256
1000   br i1 %cmp, label %if, label %else
1003   %ctlz = call i16 @llvm.ctlz.i16(i16 %x, i1 false)
1004   %res = icmp uge i16 %ctlz, 9
1005   ret i1 %res
1007 else:
1008   %ctlz2 = call i16 @llvm.ctlz.i16(i16 %x, i1 false)
1009   %res2 = icmp ult i16 %ctlz2, 7
1010   ret i1 %res2
1013 define i1 @cttz_fold(i16 %x) {
1014 ; CHECK-LABEL: @cttz_fold(
1015 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
1016 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
1017 ; CHECK:       if:
1018 ; CHECK-NEXT:    [[CTTZ:%.*]] = call i16 @llvm.cttz.i16(i16 [[X]], i1 true)
1019 ; CHECK-NEXT:    [[RES:%.*]] = icmp uge i16 [[CTTZ]], 8
1020 ; CHECK-NEXT:    ret i1 [[RES]]
1021 ; CHECK:       else:
1022 ; CHECK-NEXT:    ret i1 false
1024   %cmp = icmp ult i16 %x, 256
1025   br i1 %cmp, label %if, label %else
1028   %cttz = call i16 @llvm.cttz.i16(i16 %x, i1 true)
1029   %res = icmp uge i16 %cttz, 8
1030   ret i1 %res
1032 else:
1033   ret i1 false
1036 define i1 @cttz_nofold1(i16 %x) {
1037 ; CHECK-LABEL: @cttz_nofold1(
1038 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
1039 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
1040 ; CHECK:       if:
1041 ; CHECK-NEXT:    [[CTTZ:%.*]] = call i16 @llvm.cttz.i16(i16 [[X]], i1 true)
1042 ; CHECK-NEXT:    [[RES:%.*]] = icmp uge i16 [[CTTZ]], 7
1043 ; CHECK-NEXT:    ret i1 [[RES]]
1044 ; CHECK:       else:
1045 ; CHECK-NEXT:    ret i1 false
1047   %cmp = icmp ult i16 %x, 256
1048   br i1 %cmp, label %if, label %else
1051   %cttz = call i16 @llvm.cttz.i16(i16 %x, i1 true)
1052   %res = icmp uge i16 %cttz, 7
1053   ret i1 %res
1055 else:
1056   ret i1 false
1059 define i1 @cttz_nofold2(i16 %x) {
1060 ; CHECK-LABEL: @cttz_nofold2(
1061 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
1062 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
1063 ; CHECK:       if:
1064 ; CHECK-NEXT:    [[CTTZ:%.*]] = call i16 @llvm.cttz.i16(i16 [[X]], i1 false)
1065 ; CHECK-NEXT:    [[RES:%.*]] = icmp uge i16 [[CTTZ]], 8
1066 ; CHECK-NEXT:    ret i1 [[RES]]
1067 ; CHECK:       else:
1068 ; CHECK-NEXT:    ret i1 false
1070   %cmp = icmp ult i16 %x, 256
1071   br i1 %cmp, label %if, label %else
1074   %cttz = call i16 @llvm.cttz.i16(i16 %x, i1 false)
1075   %res = icmp uge i16 %cttz, 8
1076   ret i1 %res
1078 else:
1079   ret i1 false
1082 define i1 @ctpop_fold(i16 %x) {
1083 ; CHECK-LABEL: @ctpop_fold(
1084 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
1085 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
1086 ; CHECK:       if:
1087 ; CHECK-NEXT:    [[CTPOP:%.*]] = call i16 @llvm.ctpop.i16(i16 [[X]])
1088 ; CHECK-NEXT:    [[RES:%.*]] = icmp ule i16 [[CTPOP]], 8
1089 ; CHECK-NEXT:    ret i1 [[RES]]
1090 ; CHECK:       else:
1091 ; CHECK-NEXT:    ret i1 true
1093   %cmp = icmp ult i16 %x, 256
1094   br i1 %cmp, label %if, label %else
1097   %ctpop = call i16 @llvm.ctpop.i16(i16 %x)
1098   %res = icmp ule i16 %ctpop, 8
1099   ret i1 %res
1101 else:
1102   ret i1 true
1105 define i1 @ctpop_nofold(i16 %x) {
1106 ; CHECK-LABEL: @ctpop_nofold(
1107 ; CHECK-NEXT:    [[CMP:%.*]] = icmp ult i16 [[X:%.*]], 256
1108 ; CHECK-NEXT:    br i1 [[CMP]], label [[IF:%.*]], label [[ELSE:%.*]]
1109 ; CHECK:       if:
1110 ; CHECK-NEXT:    [[CTPOP:%.*]] = call i16 @llvm.ctpop.i16(i16 [[X]])
1111 ; CHECK-NEXT:    [[RES:%.*]] = icmp ule i16 [[CTPOP]], 7
1112 ; CHECK-NEXT:    ret i1 [[RES]]
1113 ; CHECK:       else:
1114 ; CHECK-NEXT:    ret i1 true
1116   %cmp = icmp ult i16 %x, 256
1117   br i1 %cmp, label %if, label %else
1120   %ctpop = call i16 @llvm.ctpop.i16(i16 %x)
1121   %res = icmp ule i16 %ctpop, 7
1122   ret i1 %res
1124 else:
1125   ret i1 true
1128 declare i16 @llvm.ctlz.i16(i16, i1)
1129 declare i16 @llvm.cttz.i16(i16, i1)
1130 declare i16 @llvm.ctpop.i16(i16)
1131 declare i16 @llvm.abs.i16(i16, i1)
1132 declare void @llvm.assume(i1)