• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
2; RUN: opt < %s -basicaa -newgvn -enable-phi-of-ops=true -S | FileCheck %s
3target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128"
4
5define i32 @test1(i32, i8**) {
6; CHECK-LABEL: @test1(
7; CHECK-NEXT:    [[TMP3:%.*]] = icmp ne i32 [[TMP0:%.*]], 0
8; CHECK-NEXT:    br i1 [[TMP3]], label [[TMP4:%.*]], label [[TMP5:%.*]]
9; CHECK:         br label [[TMP6:%.*]]
10; CHECK:         br label [[TMP6]]
11; CHECK:         [[PHIOFOPS:%.*]] = phi i32 [ 105, [[TMP5]] ], [ 75, [[TMP4]] ]
12; CHECK-NEXT:    [[DOT0:%.*]] = phi i32 [ 5, [[TMP4]] ], [ 7, [[TMP5]] ]
13; CHECK-NEXT:    ret i32 [[PHIOFOPS]]
14;
15  %3 = icmp ne i32 %0, 0
16  br i1 %3, label %4, label %5
17
18; <label>:4:                                      ; preds = %2
19  br label %6
20
21; <label>:5:                                      ; preds = %2
22  br label %6
23
24; <label>:6:                                      ; preds = %5, %4
25  %.0 = phi i32 [ 5, %4 ], [ 7, %5 ]
26  %7 = mul nsw i32 %.0, 15
27  ret i32 %7
28}
29;; Dependent phi of ops
30define i32 @test1b(i32, i8**) {
31; CHECK-LABEL: @test1b(
32; CHECK-NEXT:    [[TMP3:%.*]] = icmp ne i32 [[TMP0:%.*]], 0
33; CHECK-NEXT:    br i1 [[TMP3]], label [[TMP4:%.*]], label [[TMP5:%.*]]
34; CHECK:         br label [[TMP6:%.*]]
35; CHECK:         br label [[TMP6]]
36; CHECK:         [[PHIOFOPS1:%.*]] = phi i32 [ 105, [[TMP5]] ], [ 75, [[TMP4]] ]
37; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i32 [ 1575, [[TMP5]] ], [ 1125, [[TMP4]] ]
38; CHECK-NEXT:    [[DOT0:%.*]] = phi i32 [ 5, [[TMP4]] ], [ 7, [[TMP5]] ]
39; CHECK-NEXT:    ret i32 [[PHIOFOPS]]
40;
41  %3 = icmp ne i32 %0, 0
42  br i1 %3, label %4, label %5
43
44; <label>:4:                                      ; preds = %2
45  br label %6
46
47; <label>:5:                                      ; preds = %2
48  br label %6
49
50; <label>:6:                                      ; preds = %5, %4
51  %.0 = phi i32 [ 5, %4 ], [ 7, %5 ]
52  %7 = mul nsw i32 %.0, 15
53  %8 = mul nsw i32 %7, 15
54  ret i32 %8
55}
56
57define i32 @test2(i32) {
58; CHECK-LABEL: @test2(
59; CHECK-NEXT:    [[TMP2:%.*]] = icmp ne i32 [[TMP0:%.*]], 0
60; CHECK-NEXT:    br i1 [[TMP2]], label [[TMP3:%.*]], label [[TMP4:%.*]]
61; CHECK:         br label [[TMP5:%.*]]
62; CHECK:         br label [[TMP5]]
63; CHECK:         [[DOT01:%.*]] = phi i32 [ 3, [[TMP3]] ], [ 2, [[TMP4]] ]
64; CHECK-NEXT:    [[DOT0:%.*]] = phi i32 [ 2, [[TMP3]] ], [ 3, [[TMP4]] ]
65; CHECK-NEXT:    ret i32 5
66;
67  %2 = icmp ne i32 %0, 0
68  br i1 %2, label %3, label %4
69
70; <label>:3:                                      ; preds = %1
71  br label %5
72
73; <label>:4:                                      ; preds = %1
74  br label %5
75
76; <label>:5:                                      ; preds = %4, %3
77  %.01 = phi i32 [ 3, %3 ], [ 2, %4 ]
78  %.0 = phi i32 [ 2, %3 ], [ 3, %4 ]
79  %6 = add nsw i32 %.01, %.0
80  ret i32 %6
81}
82define i32 @test3(i1 %which) {
83; CHECK-LABEL: @test3(
84; CHECK-NEXT:  entry:
85; CHECK-NEXT:    br i1 [[WHICH:%.*]], label [[FINAL:%.*]], label [[DELAY:%.*]]
86; CHECK:       delay:
87; CHECK-NEXT:    br label [[FINAL]]
88; CHECK:       final:
89; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i32 [ -877, [[ENTRY:%.*]] ], [ 113, [[DELAY]] ]
90; CHECK-NEXT:    [[A:%.*]] = phi i32 [ 1000, [[ENTRY]] ], [ 10, [[DELAY]] ]
91; CHECK-NEXT:    ret i32 [[PHIOFOPS]]
92;
93
94entry:
95  br i1 %which, label %final, label %delay
96
97delay:
98  br label %final
99
100final:
101  %A = phi i32 [ 1000, %entry ], [ 10, %delay ]
102  %value = sub i32 123, %A
103  ret i32 %value
104}
105
106define <2 x i32> @test3vec(i1 %which) {
107; CHECK-LABEL: @test3vec(
108; CHECK-NEXT:  entry:
109; CHECK-NEXT:    br i1 [[WHICH:%.*]], label [[FINAL:%.*]], label [[DELAY:%.*]]
110; CHECK:       delay:
111; CHECK-NEXT:    br label [[FINAL]]
112; CHECK:       final:
113; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi <2 x i32> [ <i32 -877, i32 -877>, [[ENTRY:%.*]] ], [ <i32 113, i32 113>, [[DELAY]] ]
114; CHECK-NEXT:    [[A:%.*]] = phi <2 x i32> [ <i32 1000, i32 1000>, [[ENTRY]] ], [ <i32 10, i32 10>, [[DELAY]] ]
115; CHECK-NEXT:    ret <2 x i32> [[PHIOFOPS]]
116;
117
118entry:
119  br i1 %which, label %final, label %delay
120
121delay:
122  br label %final
123
124final:
125  %A = phi <2 x i32> [ <i32 1000, i32 1000>, %entry ], [ <i32 10, i32 10>, %delay ]
126  %value = sub <2 x i32> <i32 123, i32 123>, %A
127  ret <2 x i32> %value
128}
129
130define <2 x i32> @test3vec2(i1 %which) {
131; CHECK-LABEL: @test3vec2(
132; CHECK-NEXT:  entry:
133; CHECK-NEXT:    br i1 [[WHICH:%.*]], label [[FINAL:%.*]], label [[DELAY:%.*]]
134; CHECK:       delay:
135; CHECK-NEXT:    br label [[FINAL]]
136; CHECK:       final:
137; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi <2 x i32> [ <i32 -877, i32 -2167>, [[ENTRY:%.*]] ], [ <i32 113, i32 303>, [[DELAY]] ]
138; CHECK-NEXT:    [[A:%.*]] = phi <2 x i32> [ <i32 1000, i32 2500>, [[ENTRY]] ], [ <i32 10, i32 30>, [[DELAY]] ]
139; CHECK-NEXT:    ret <2 x i32> [[PHIOFOPS]]
140;
141
142entry:
143  br i1 %which, label %final, label %delay
144
145delay:
146  br label %final
147
148final:
149  %A = phi <2 x i32> [ <i32 1000, i32 2500>, %entry ], [ <i32 10, i32 30>, %delay ]
150  %value = sub <2 x i32> <i32 123, i32 333>, %A
151  ret <2 x i32> %value
152}
153
154;; This example is a bit contrived because we can't create fake memoryuses, so we use two loads in the if blocks
155define i32 @test4(i32, i8**, i32* noalias, i32* noalias) {
156; CHECK-LABEL: @test4(
157; CHECK-NEXT:    store i32 5, i32* [[TMP2:%.*]], align 4
158; CHECK-NEXT:    store i32 7, i32* [[TMP3:%.*]], align 4
159; CHECK-NEXT:    [[TMP5:%.*]] = icmp ne i32 [[TMP0:%.*]], 0
160; CHECK-NEXT:    br i1 [[TMP5]], label [[TMP6:%.*]], label [[TMP7:%.*]]
161; CHECK:         br label [[TMP8:%.*]]
162; CHECK:         br label [[TMP8]]
163; CHECK:         [[DOT01:%.*]] = phi i32 [ 5, [[TMP6]] ], [ 7, [[TMP7]] ]
164; CHECK-NEXT:    [[DOT0:%.*]] = phi i32* [ [[TMP2]], [[TMP6]] ], [ [[TMP3]], [[TMP7]] ]
165; CHECK-NEXT:    [[TMP9:%.*]] = load i32, i32* [[DOT0]], align 4
166; CHECK-NEXT:    [[TMP10:%.*]] = mul nsw i32 [[TMP9]], 15
167; CHECK-NEXT:    [[TMP11:%.*]] = mul nsw i32 [[TMP10]], [[DOT01]]
168; CHECK-NEXT:    ret i32 [[TMP11]]
169;
170  store i32 5, i32* %2, align 4
171  store i32 7, i32* %3, align 4
172  %5 = icmp ne i32 %0, 0
173  br i1 %5, label %6, label %8
174
175; <label>:6:                                      ; preds = %4
176  %7 = load i32, i32* %2, align 4
177  br label %10
178
179; <label>:8:                                      ; preds = %4
180  %9 = load i32, i32* %3, align 4
181  br label %10
182
183; <label>:10:                                     ; preds = %8, %6
184  %.01 = phi i32 [ %7, %6 ], [ %9, %8 ]
185  %.0 = phi i32* [ %2, %6 ], [ %3, %8 ]
186  %11 = load i32, i32* %.0, align 4
187  %12 = mul nsw i32 %11, 15
188  %13 = mul nsw i32 %12, %.01
189  ret i32 %13
190}
191
192@global = common global [100 x i64] zeroinitializer, align 16
193@global.1 = common global [100 x i64] zeroinitializer, align 16
194define i64 @test5(i64 %arg) {
195; CHECK-LABEL: @test5(
196; CHECK-NEXT:  bb:
197; CHECK-NEXT:    [[TMP:%.*]] = alloca i64, align 8
198; CHECK-NEXT:    [[TMP1:%.*]] = icmp eq i64 [[ARG:%.*]], 0
199; CHECK-NEXT:    br i1 [[TMP1]], label [[BB28:%.*]], label [[BB2:%.*]]
200; CHECK:       bb2:
201; CHECK-NEXT:    br label [[BB7:%.*]]
202; CHECK:       bb4:
203; CHECK-NEXT:    br label [[BB5:%.*]]
204; CHECK:       bb5:
205; CHECK-NEXT:    [[TMP6:%.*]] = icmp eq i64 [[TMP9:%.*]], 0
206; CHECK-NEXT:    br i1 [[TMP6]], label [[BB27:%.*]], label [[BB7]]
207; CHECK:       bb7:
208; CHECK-NEXT:    [[TMP8:%.*]] = phi i64 [ [[ARG]], [[BB2]] ], [ [[TMP9]], [[BB5]] ]
209; CHECK-NEXT:    [[TMP9]] = add nsw i64 [[TMP8]], -1
210; CHECK-NEXT:    [[TMP10:%.*]] = load i64, i64* getelementptr inbounds ([100 x i64], [100 x i64]* @global, i64 0, i64 0), align 16
211; CHECK-NEXT:    [[TMP11:%.*]] = load i64, i64* getelementptr inbounds ([100 x i64], [100 x i64]* @global.1, i64 0, i64 0), align 16
212; CHECK-NEXT:    [[TMP12:%.*]] = mul nsw i64 [[TMP11]], [[TMP10]]
213; CHECK-NEXT:    [[TMP13:%.*]] = icmp eq i64 [[TMP12]], 0
214; CHECK-NEXT:    br i1 [[TMP13]], label [[BB5]], label [[BB14:%.*]]
215; CHECK:       bb14:
216; CHECK-NEXT:    br label [[BB15:%.*]]
217; CHECK:       bb15:
218; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i64 [ [[TMP12]], [[BB14]] ], [ [[TMP25:%.*]], [[BB15]] ]
219; CHECK-NEXT:    [[TMP16:%.*]] = phi i64 [ [[TMP24:%.*]], [[BB15]] ], [ [[TMP11]], [[BB14]] ]
220; CHECK-NEXT:    [[TMP17:%.*]] = phi i64 [ [[TMP22:%.*]], [[BB15]] ], [ [[TMP10]], [[BB14]] ]
221; CHECK-NEXT:    [[TMP18:%.*]] = phi i64 [ [[TMP20:%.*]], [[BB15]] ], [ 0, [[BB14]] ]
222; CHECK-NEXT:    store i64 [[PHIOFOPS]], i64* [[TMP]], align 8
223; CHECK-NEXT:    [[TMP20]] = add nuw nsw i64 [[TMP18]], 1
224; CHECK-NEXT:    [[TMP21:%.*]] = getelementptr inbounds [100 x i64], [100 x i64]* @global, i64 0, i64 [[TMP20]]
225; CHECK-NEXT:    [[TMP22]] = load i64, i64* [[TMP21]], align 8
226; CHECK-NEXT:    [[TMP23:%.*]] = getelementptr inbounds [100 x i64], [100 x i64]* @global.1, i64 0, i64 [[TMP20]]
227; CHECK-NEXT:    [[TMP24]] = load i64, i64* [[TMP23]], align 8
228; CHECK-NEXT:    [[TMP25]] = mul nsw i64 [[TMP24]], [[TMP22]]
229; CHECK-NEXT:    [[TMP26:%.*]] = icmp eq i64 [[TMP20]], [[TMP25]]
230; CHECK-NEXT:    br i1 [[TMP26]], label [[BB4:%.*]], label [[BB15]]
231; CHECK:       bb27:
232; CHECK-NEXT:    br label [[BB28]]
233; CHECK:       bb28:
234; CHECK-NEXT:    ret i64 0
235;
236bb:
237  %tmp = alloca i64, align 8
238  %tmp1 = icmp eq i64 %arg, 0
239  br i1 %tmp1, label %bb28, label %bb2
240
241bb2:                                              ; preds = %bb
242  %tmp3 = bitcast i64* %tmp to i8*
243  br label %bb7
244
245bb4:                                              ; preds = %bb15
246  br label %bb5
247
248bb5:                                              ; preds = %bb7, %bb4
249  %tmp6 = icmp eq i64 %tmp9, 0
250  br i1 %tmp6, label %bb27, label %bb7
251
252bb7:                                              ; preds = %bb5, %bb2
253  %tmp8 = phi i64 [ %arg, %bb2 ], [ %tmp9, %bb5 ]
254  %tmp9 = add nsw i64 %tmp8, -1
255  %tmp10 = load i64, i64* getelementptr inbounds ([100 x i64], [100 x i64]* @global, i64 0, i64 0), align 16
256  %tmp11 = load i64, i64* getelementptr inbounds ([100 x i64], [100 x i64]* @global.1, i64 0, i64 0), align 16
257  %tmp12 = mul nsw i64 %tmp11, %tmp10
258  %tmp13 = icmp eq i64 %tmp12, 0
259  br i1 %tmp13, label %bb5, label %bb14
260
261bb14:                                             ; preds = %bb7
262  br label %bb15
263
264bb15:                                             ; preds = %bb15, %bb14
265  %tmp16 = phi i64 [ %tmp24, %bb15 ], [ %tmp11, %bb14 ]
266  %tmp17 = phi i64 [ %tmp22, %bb15 ], [ %tmp10, %bb14 ]
267  %tmp18 = phi i64 [ %tmp20, %bb15 ], [ 0, %bb14 ]
268;; This multiply is an op of phis which is really equivalent to phi(tmp25, tmp12)
269  %tmp19 = mul nsw i64 %tmp16, %tmp17
270  store i64 %tmp19, i64* %tmp, align 8
271  %tmp20 = add nuw nsw i64 %tmp18, 1
272  %tmp21 = getelementptr inbounds [100 x i64], [100 x i64]* @global, i64 0, i64 %tmp20
273  %tmp22 = load i64, i64* %tmp21, align 8
274  %tmp23 = getelementptr inbounds [100 x i64], [100 x i64]* @global.1, i64 0, i64 %tmp20
275  %tmp24 = load i64, i64* %tmp23, align 8
276  %tmp25 = mul nsw i64 %tmp24, %tmp22
277  %tmp26 = icmp eq i64 %tmp20, %tmp25
278  br i1 %tmp26, label %bb4, label %bb15
279
280bb27:                                             ; preds = %bb5
281  br label %bb28
282
283bb28:                                             ; preds = %bb27, %bb
284  ret i64 0
285}
286
287;; These icmps are all equivalent to phis of constants
288define i8 @test6(i8* %addr) {
289; CHECK-LABEL: @test6(
290; CHECK-NEXT:  entry-block:
291; CHECK-NEXT:    br label %main-loop
292; CHECK:       main-loop:
293; CHECK-NEXT:    [[PHIOFOPS1:%.*]] = phi i1 [ true, %entry-block ], [ false, [[CORE:%.*]] ]
294; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i1 [ false, %entry-block ], [ true, [[CORE]] ]
295; CHECK-NEXT:    [[PHI:%.*]] = phi i8 [ 0, %entry-block ], [ 1, [[CORE]] ]
296; CHECK-NEXT:    store volatile i8 0, i8* [[ADDR:%.*]]
297; CHECK-NEXT:    br i1 [[PHIOFOPS1]], label %busy-wait-phi-0, label [[EXIT:%.*]]
298; CHECK:       busy-wait-phi-0:
299; CHECK-NEXT:    [[LOAD:%.*]] = load volatile i8, i8* [[ADDR]]
300; CHECK-NEXT:    [[ICMP:%.*]] = icmp eq i8 [[LOAD]], 0
301; CHECK-NEXT:    br i1 [[ICMP]], label %busy-wait-phi-0, label [[CORE]]
302; CHECK:       core:
303; CHECK-NEXT:    br i1 [[PHIOFOPS]], label [[TRAP:%.*]], label %main-loop
304; CHECK:       trap:
305; CHECK-NEXT:    ret i8 1
306; CHECK:       exit:
307; CHECK-NEXT:    ret i8 0
308;
309entry-block:
310  br label %main-loop
311
312main-loop:
313  %phi = phi i8 [ 0, %entry-block ], [ 1, %core ]
314  %switch_0 = icmp eq i8 %phi, 0
315  store volatile i8 0, i8* %addr
316  br i1 %switch_0, label %busy-wait-phi-0, label %exit
317
318busy-wait-phi-0:
319  %load = load volatile i8, i8* %addr
320  %icmp = icmp eq i8 %load, 0
321  br i1 %icmp, label %busy-wait-phi-0, label %core
322
323core:
324  %switch_1 = icmp eq i8 %phi, 1
325  br i1 %switch_1, label %trap, label %main-loop
326
327trap:
328  ret i8 1
329
330exit:
331  ret i8 0
332}
333
334; Test that we don't infinite loop simplifying
335; an undefined value that can go both ways.
336define void @test7() {
337; CHECK-LABEL: @test7(
338; CHECK-NEXT:  bb:
339; CHECK-NEXT:    br label [[BB1:%.*]]
340; CHECK:       bb1:
341; CHECK-NEXT:    br label [[BB1]]
342;
343bb:
344  br label %bb1
345
346bb1:                                              ; preds = %bb1, %bb
347  %tmp = phi i32 [ undef, %bb ], [ %tmp3, %bb1 ]
348  %tmp2 = icmp eq i32 %tmp, 0
349  %tmp3 = select i1 %tmp2, i32 1, i32 %tmp
350  br label %bb1
351}
352
353
354
355; Test that we get a consistent answer about what the
356; value of this undefined select is.
357define void @test8() {
358; CHECK-LABEL: @test8(
359; CHECK-NEXT:  bb:
360; CHECK-NEXT:    br label [[BB1:%.*]]
361; CHECK:       bb1:
362; CHECK-NEXT:    br label [[BB1]]
363;
364bb:
365  %tmp = select i1 undef, i8 0, i8 1
366  br label %bb1
367
368bb1:                                              ; preds = %bb1, %bb
369  %tmp2 = phi i8 [ %tmp4, %bb1 ], [ %tmp, %bb ]
370  %tmp3 = icmp eq i8 %tmp2, 0
371  %tmp4 = select i1 %tmp3, i8 1, i8 %tmp2
372  br label %bb1
373}
374
375
376;; Make sure we handle the case where we later come up with an expression that we need
377;; for a phi of ops.
378define void @test9() {
379; CHECK-LABEL: @test9(
380; CHECK-NEXT:  bb:
381; CHECK-NEXT:    br label [[BB1:%.*]]
382; CHECK:       bb1:
383; CHECK-NEXT:    br i1 undef, label [[BB1]], label [[BB2:%.*]]
384; CHECK:       bb2:
385; CHECK-NEXT:    br label [[BB6:%.*]]
386; CHECK:       bb6:
387; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i32 [ -13, [[BB2]] ], [ [[TMP11:%.*]], [[BB6]] ]
388; CHECK-NEXT:    [[TMP7:%.*]] = phi i32 [ 1, [[BB2]] ], [ [[TMP8:%.*]], [[BB6]] ]
389; CHECK-NEXT:    [[TMP8]] = add nuw nsw i32 [[TMP7]], 1
390; CHECK-NEXT:    [[TMP11]] = add i32 -14, [[TMP8]]
391; CHECK-NEXT:    br label [[BB6]]
392;
393bb:
394  br label %bb1
395
396bb1:                                              ; preds = %bb1, %bb
397  br i1 undef, label %bb1, label %bb2
398
399bb2:                                              ; preds = %bb1
400  %tmp = select i1 true, i32 -14, i32 -10
401  %tmp3 = add i32 %tmp, 0
402  %tmp4 = select i1 true, i32 -14, i32 -10
403  %tmp5 = add i32 %tmp4, 0
404  br label %bb6
405
406bb6:                                              ; preds = %bb6, %bb2
407  %tmp7 = phi i32 [ 1, %bb2 ], [ %tmp13, %bb6 ]
408  %tmp8 = add nuw nsw i32 %tmp7, 1
409  %tmp9 = add i32 %tmp3, %tmp7
410  %tmp10 = select i1 false, i32 undef, i32 %tmp9
411  %tmp11 = add i32 %tmp5, %tmp8
412  %tmp12 = select i1 undef, i32 undef, i32 %tmp11
413  %tmp13 = add nuw nsw i32 %tmp7, 1
414  br label %bb6
415}
416
417;; Ensure that we revisit predicateinfo operands at the right points in time.
418define void @test10() {
419; CHECK-LABEL: @test10(
420; CHECK-NEXT:  b:
421; CHECK-NEXT:    br label [[G:%.*]]
422; CHECK:       g:
423; CHECK-NEXT:    [[N:%.*]] = phi i32* [ [[H:%.*]], [[I:%.*]] ], [ null, [[B:%.*]] ]
424; CHECK-NEXT:    [[H]] = getelementptr i32, i32* [[N]], i64 1
425; CHECK-NEXT:    [[J:%.*]] = icmp eq i32* [[H]], inttoptr (i64 32 to i32*)
426; CHECK-NEXT:    br i1 [[J]], label [[C:%.*]], label [[I]]
427; CHECK:       i:
428; CHECK-NEXT:    br i1 undef, label [[K:%.*]], label [[G]]
429; CHECK:       k:
430; CHECK-NEXT:    br i1 false, label [[C]], label [[O:%.*]]
431; CHECK:       o:
432; CHECK-NEXT:    br label [[C]]
433; CHECK:       c:
434; CHECK-NEXT:    ret void
435;
436b:
437  %m = getelementptr i32, i32* null, i64 8
438  br label %g
439
440g:                                                ; preds = %i, %b
441  %n = phi i32* [ %h, %i ], [ null, %b ]
442  %h = getelementptr i32, i32* %n, i64 1
443  %j = icmp eq i32* %h, %m
444  br i1 %j, label %c, label %i
445
446i:                                                ; preds = %g
447  br i1 undef, label %k, label %g
448
449k:                                                ; preds = %i
450  %l = icmp eq i32* %n, %m
451  br i1 %l, label %c, label %o
452
453o:                                                ; preds = %k
454  br label %c
455
456c:                                                ; preds = %o, %k, %g
457  %0 = phi i32* [ undef, %o ], [ %m, %k ], [ %m, %g ]
458  ret void
459}
460
461;; Ensure we handle VariableExpression properly.
462define void @test11() {
463; CHECK-LABEL: @test11(
464; CHECK-NEXT:  bb:
465; CHECK-NEXT:    br i1 undef, label [[BB1:%.*]], label [[BB2:%.*]]
466; CHECK:       bb1:
467; CHECK-NEXT:    br label [[BB2]]
468; CHECK:       bb2:
469; CHECK-NEXT:    [[TMP:%.*]] = phi i1 [ false, [[BB1]] ], [ true, [[BB:%.*]] ]
470; CHECK-NEXT:    [[TMP3:%.*]] = call i32* @wombat()
471; CHECK-NEXT:    [[TMP4:%.*]] = icmp ne i32* [[TMP3]], null
472; CHECK-NEXT:    [[TMP5:%.*]] = and i1 [[TMP]], [[TMP4]]
473; CHECK-NEXT:    br i1 [[TMP5]], label [[BB6:%.*]], label [[BB7:%.*]]
474; CHECK:       bb6:
475; CHECK-NEXT:    unreachable
476; CHECK:       bb7:
477; CHECK-NEXT:    ret void
478;
479bb:
480  br i1 undef, label %bb1, label %bb2
481
482bb1:                                              ; preds = %bb
483  br label %bb2
484
485bb2:                                              ; preds = %bb1, %bb
486  %tmp = phi i1 [ false, %bb1 ], [ true, %bb ]
487  %tmp3 = call i32* @wombat()
488  %tmp4 = icmp ne i32* %tmp3, null
489  %tmp5 = and i1 %tmp, %tmp4
490  br i1 %tmp5, label %bb6, label %bb7
491
492bb6:                                              ; preds = %bb2
493  unreachable
494
495bb7:                                              ; preds = %bb2
496  ret void
497}
498
499declare i32* @wombat()
500
501;; Ensure that when reachability affects a phi of ops, we recompute
502;; it.  Here, the phi node is marked for recomputation when bb7->bb3
503;; becomes live, but the value does not change. if we do not directly
504;; recompute the phi of ops instruction (tmp5), the value number will
505;; change in the verifier, as it goes from a constant value to a
506;; phi of [true, false]
507
508define void @test12() {
509; CHECK-LABEL: @test12(
510; CHECK-NEXT:  bb:
511; CHECK-NEXT:    [[TMP:%.*]] = load i32, i32* null
512; CHECK-NEXT:    [[TMP1:%.*]] = icmp sgt i32 [[TMP]], 0
513; CHECK-NEXT:    br i1 [[TMP1]], label [[BB2:%.*]], label [[BB8:%.*]]
514; CHECK:       bb2:
515; CHECK-NEXT:    br label [[BB3:%.*]]
516; CHECK:       bb3:
517; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i1 [ true, [[BB2]] ], [ false, [[BB7:%.*]] ]
518; CHECK-NEXT:    br i1 [[PHIOFOPS]], label [[BB6:%.*]], label [[BB7]]
519; CHECK:       bb6:
520; CHECK-NEXT:    br label [[BB7]]
521; CHECK:       bb7:
522; CHECK-NEXT:    br label [[BB3]]
523; CHECK:       bb8:
524; CHECK-NEXT:    ret void
525;
526bb:
527  %tmp = load i32, i32* null
528  %tmp1 = icmp sgt i32 %tmp, 0
529  br i1 %tmp1, label %bb2, label %bb8
530
531bb2:                                              ; preds = %bb
532  br label %bb3
533
534bb3:                                              ; preds = %bb7, %bb2
535  %tmp4 = phi i32 [ %tmp, %bb2 ], [ undef, %bb7 ]
536  %tmp5 = icmp sgt i32 %tmp4, 0
537  br i1 %tmp5, label %bb6, label %bb7
538
539bb6:                                              ; preds = %bb3
540  br label %bb7
541
542bb7:                                              ; preds = %bb6, %bb3
543  br label %bb3
544
545bb8:                                              ; preds = %bb
546  ret void
547}
548
549;; Make sure we reprocess phi of ops involving loads when loads change class.
550;; This is PR 34473
551define void @test13() {
552; CHECK-LABEL: @test13(
553; CHECK-NEXT:  bb:
554; CHECK-NEXT:    br label [[BB1:%.*]]
555; CHECK:       bb1:
556; CHECK-NEXT:    [[TMP:%.*]] = load i8, i8* null
557; CHECK-NEXT:    br label [[BB3:%.*]]
558; CHECK:       bb3:
559; CHECK-NEXT:    [[PHIOFOPS:%.*]] = phi i8 [ [[TMP]], [[BB1]] ], [ [[TMP10:%.*]], [[BB3]] ]
560; CHECK-NEXT:    [[TMP4:%.*]] = phi i8* [ null, [[BB1]] ], [ [[TMP6:%.*]], [[BB3]] ]
561; CHECK-NEXT:    [[TMP5:%.*]] = phi i32 [ undef, [[BB1]] ], [ [[TMP9:%.*]], [[BB3]] ]
562; CHECK-NEXT:    [[TMP6]] = getelementptr i8, i8* [[TMP4]], i64 1
563; CHECK-NEXT:    [[TMP8:%.*]] = sext i8 [[PHIOFOPS]] to i32
564; CHECK-NEXT:    [[TMP9]] = mul i32 [[TMP5]], [[TMP8]]
565; CHECK-NEXT:    [[TMP10]] = load i8, i8* [[TMP6]]
566; CHECK-NEXT:    [[TMP11:%.*]] = icmp eq i8 [[TMP10]], 0
567; CHECK-NEXT:    br i1 [[TMP11]], label [[BB12:%.*]], label [[BB3]]
568; CHECK:       bb12:
569; CHECK-NEXT:    [[TMP14:%.*]] = icmp eq i32 [[TMP9]], 0
570; CHECK-NEXT:    br i1 [[TMP14]], label [[BB1]], label [[BB15:%.*]]
571; CHECK:       bb15:
572; CHECK-NEXT:    call void (...) @bar()
573; CHECK-NEXT:    br label [[BB1]]
574;
575bb:
576  br label %bb1
577
578bb1:                                              ; preds = %bb15, %bb12, %bb
579  %tmp = load i8, i8* null
580  %tmp2 = icmp eq i8 %tmp, 8
581  br label %bb3
582
583bb3:                                              ; preds = %bb3, %bb1
584  %tmp4 = phi i8* [ null, %bb1 ], [ %tmp6, %bb3 ]
585  %tmp5 = phi i32 [ undef, %bb1 ], [ %tmp9, %bb3 ]
586  %tmp6 = getelementptr i8, i8* %tmp4, i64 1
587  %tmp7 = load i8, i8* %tmp4
588  %tmp8 = sext i8 %tmp7 to i32
589  %tmp9 = mul i32 %tmp5, %tmp8
590  %tmp10 = load i8, i8* %tmp6
591  %tmp11 = icmp eq i8 %tmp10, 0
592  br i1 %tmp11, label %bb12, label %bb3
593
594bb12:                                             ; preds = %bb3
595  %tmp13 = phi i32 [ %tmp9, %bb3 ]
596  %tmp14 = icmp eq i32 %tmp13, 0
597  br i1 %tmp14, label %bb1, label %bb15
598
599bb15:                                             ; preds = %bb12
600  call void (...) @bar()
601  br label %bb1
602}
603
604declare void @bar(...)
605
606