| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py |
| ; RUN: opt < %s -passes=newgvn -enable-phi-of-ops=true -S | FileCheck %s |
| target datalayout = "e-m:o-i64:64-f80:128-n8:16:32:64-S128" |
| |
| define i32 @test1(i32, ptr) { |
| ; CHECK-LABEL: @test1( |
| ; CHECK-NEXT: [[TMP3:%.*]] = icmp ne i32 [[TMP0:%.*]], 0 |
| ; CHECK-NEXT: br i1 [[TMP3]], label [[TMP4:%.*]], label [[TMP5:%.*]] |
| ; CHECK: br label [[TMP6:%.*]] |
| ; CHECK: br label [[TMP6]] |
| ; CHECK: [[PHIOFOPS:%.*]] = phi i32 [ 105, [[TMP5]] ], [ 75, [[TMP4]] ] |
| ; CHECK-NEXT: [[DOT0:%.*]] = phi i32 [ 5, [[TMP4]] ], [ 7, [[TMP5]] ] |
| ; CHECK-NEXT: ret i32 [[PHIOFOPS]] |
| ; |
| %3 = icmp ne i32 %0, 0 |
| br i1 %3, label %4, label %5 |
| |
| ; <label>:4: ; preds = %2 |
| br label %6 |
| |
| ; <label>:5: ; preds = %2 |
| br label %6 |
| |
| ; <label>:6: ; preds = %5, %4 |
| %.0 = phi i32 [ 5, %4 ], [ 7, %5 ] |
| %7 = mul nsw i32 %.0, 15 |
| ret i32 %7 |
| } |
| ;; Dependent phi of ops |
| define i32 @test1b(i32, ptr) { |
| ; CHECK-LABEL: @test1b( |
| ; CHECK-NEXT: [[TMP3:%.*]] = icmp ne i32 [[TMP0:%.*]], 0 |
| ; CHECK-NEXT: br i1 [[TMP3]], label [[TMP4:%.*]], label [[TMP5:%.*]] |
| ; CHECK: br label [[TMP6:%.*]] |
| ; CHECK: br label [[TMP6]] |
| ; CHECK: [[PHIOFOPS1:%.*]] = phi i32 [ 105, [[TMP5]] ], [ 75, [[TMP4]] ] |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi i32 [ 1575, [[TMP5]] ], [ 1125, [[TMP4]] ] |
| ; CHECK-NEXT: [[DOT0:%.*]] = phi i32 [ 5, [[TMP4]] ], [ 7, [[TMP5]] ] |
| ; CHECK-NEXT: ret i32 [[PHIOFOPS]] |
| ; |
| %3 = icmp ne i32 %0, 0 |
| br i1 %3, label %4, label %5 |
| |
| ; <label>:4: ; preds = %2 |
| br label %6 |
| |
| ; <label>:5: ; preds = %2 |
| br label %6 |
| |
| ; <label>:6: ; preds = %5, %4 |
| %.0 = phi i32 [ 5, %4 ], [ 7, %5 ] |
| %7 = mul nsw i32 %.0, 15 |
| %8 = mul nsw i32 %7, 15 |
| ret i32 %8 |
| } |
| |
| define i32 @test2(i32) { |
| ; CHECK-LABEL: @test2( |
| ; CHECK-NEXT: [[TMP2:%.*]] = icmp ne i32 [[TMP0:%.*]], 0 |
| ; CHECK-NEXT: br i1 [[TMP2]], label [[TMP3:%.*]], label [[TMP4:%.*]] |
| ; CHECK: br label [[TMP5:%.*]] |
| ; CHECK: br label [[TMP5]] |
| ; CHECK: [[DOT01:%.*]] = phi i32 [ 3, [[TMP3]] ], [ 2, [[TMP4]] ] |
| ; CHECK-NEXT: [[DOT0:%.*]] = phi i32 [ 2, [[TMP3]] ], [ 3, [[TMP4]] ] |
| ; CHECK-NEXT: ret i32 5 |
| ; |
| %2 = icmp ne i32 %0, 0 |
| br i1 %2, label %3, label %4 |
| |
| ; <label>:3: ; preds = %1 |
| br label %5 |
| |
| ; <label>:4: ; preds = %1 |
| br label %5 |
| |
| ; <label>:5: ; preds = %4, %3 |
| %.01 = phi i32 [ 3, %3 ], [ 2, %4 ] |
| %.0 = phi i32 [ 2, %3 ], [ 3, %4 ] |
| %6 = add nsw i32 %.01, %.0 |
| ret i32 %6 |
| } |
| define i32 @test3(i1 %which) { |
| ; CHECK-LABEL: @test3( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: br i1 [[WHICH:%.*]], label [[FINAL:%.*]], label [[DELAY:%.*]] |
| ; CHECK: delay: |
| ; CHECK-NEXT: br label [[FINAL]] |
| ; CHECK: final: |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi i32 [ -877, [[ENTRY:%.*]] ], [ 113, [[DELAY]] ] |
| ; CHECK-NEXT: [[A:%.*]] = phi i32 [ 1000, [[ENTRY]] ], [ 10, [[DELAY]] ] |
| ; CHECK-NEXT: ret i32 [[PHIOFOPS]] |
| ; |
| |
| entry: |
| br i1 %which, label %final, label %delay |
| |
| delay: |
| br label %final |
| |
| final: |
| %A = phi i32 [ 1000, %entry ], [ 10, %delay ] |
| %value = sub i32 123, %A |
| ret i32 %value |
| } |
| |
| define <2 x i32> @test3vec(i1 %which) { |
| ; CHECK-LABEL: @test3vec( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: br i1 [[WHICH:%.*]], label [[FINAL:%.*]], label [[DELAY:%.*]] |
| ; CHECK: delay: |
| ; CHECK-NEXT: br label [[FINAL]] |
| ; CHECK: final: |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi <2 x i32> [ <i32 -877, i32 -877>, [[ENTRY:%.*]] ], [ <i32 113, i32 113>, [[DELAY]] ] |
| ; CHECK-NEXT: [[A:%.*]] = phi <2 x i32> [ <i32 1000, i32 1000>, [[ENTRY]] ], [ <i32 10, i32 10>, [[DELAY]] ] |
| ; CHECK-NEXT: ret <2 x i32> [[PHIOFOPS]] |
| ; |
| |
| entry: |
| br i1 %which, label %final, label %delay |
| |
| delay: |
| br label %final |
| |
| final: |
| %A = phi <2 x i32> [ <i32 1000, i32 1000>, %entry ], [ <i32 10, i32 10>, %delay ] |
| %value = sub <2 x i32> <i32 123, i32 123>, %A |
| ret <2 x i32> %value |
| } |
| |
| define <2 x i32> @test3vec2(i1 %which) { |
| ; CHECK-LABEL: @test3vec2( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: br i1 [[WHICH:%.*]], label [[FINAL:%.*]], label [[DELAY:%.*]] |
| ; CHECK: delay: |
| ; CHECK-NEXT: br label [[FINAL]] |
| ; CHECK: final: |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi <2 x i32> [ <i32 -877, i32 -2167>, [[ENTRY:%.*]] ], [ <i32 113, i32 303>, [[DELAY]] ] |
| ; CHECK-NEXT: [[A:%.*]] = phi <2 x i32> [ <i32 1000, i32 2500>, [[ENTRY]] ], [ <i32 10, i32 30>, [[DELAY]] ] |
| ; CHECK-NEXT: ret <2 x i32> [[PHIOFOPS]] |
| ; |
| |
| entry: |
| br i1 %which, label %final, label %delay |
| |
| delay: |
| br label %final |
| |
| final: |
| %A = phi <2 x i32> [ <i32 1000, i32 2500>, %entry ], [ <i32 10, i32 30>, %delay ] |
| %value = sub <2 x i32> <i32 123, i32 333>, %A |
| ret <2 x i32> %value |
| } |
| |
| ;; This example is a bit contrived because we can't create fake memoryuses, so we use two loads in the if blocks |
| define i32 @test4(i32, ptr, ptr noalias, ptr noalias) { |
| ; CHECK-LABEL: @test4( |
| ; CHECK-NEXT: store i32 5, ptr [[TMP2:%.*]], align 4 |
| ; CHECK-NEXT: store i32 7, ptr [[TMP3:%.*]], align 4 |
| ; CHECK-NEXT: [[TMP5:%.*]] = icmp ne i32 [[TMP0:%.*]], 0 |
| ; CHECK-NEXT: br i1 [[TMP5]], label [[TMP6:%.*]], label [[TMP7:%.*]] |
| ; CHECK: br label [[TMP8:%.*]] |
| ; CHECK: br label [[TMP8]] |
| ; CHECK: [[DOT01:%.*]] = phi i32 [ 5, [[TMP6]] ], [ 7, [[TMP7]] ] |
| ; CHECK-NEXT: [[DOT0:%.*]] = phi ptr [ [[TMP2]], [[TMP6]] ], [ [[TMP3]], [[TMP7]] ] |
| ; CHECK-NEXT: [[TMP9:%.*]] = load i32, ptr [[DOT0]], align 4 |
| ; CHECK-NEXT: [[TMP10:%.*]] = mul nsw i32 [[TMP9]], 15 |
| ; CHECK-NEXT: [[TMP11:%.*]] = mul nsw i32 [[TMP10]], [[DOT01]] |
| ; CHECK-NEXT: ret i32 [[TMP11]] |
| ; |
| store i32 5, ptr %2, align 4 |
| store i32 7, ptr %3, align 4 |
| %5 = icmp ne i32 %0, 0 |
| br i1 %5, label %6, label %8 |
| |
| ; <label>:6: ; preds = %4 |
| %7 = load i32, ptr %2, align 4 |
| br label %10 |
| |
| ; <label>:8: ; preds = %4 |
| %9 = load i32, ptr %3, align 4 |
| br label %10 |
| |
| ; <label>:10: ; preds = %8, %6 |
| %.01 = phi i32 [ %7, %6 ], [ %9, %8 ] |
| %.0 = phi ptr [ %2, %6 ], [ %3, %8 ] |
| %11 = load i32, ptr %.0, align 4 |
| %12 = mul nsw i32 %11, 15 |
| %13 = mul nsw i32 %12, %.01 |
| ret i32 %13 |
| } |
| |
| @global = common global [100 x i64] zeroinitializer, align 16 |
| @global.1 = common global [100 x i64] zeroinitializer, align 16 |
| define i64 @test5(i64 %arg) { |
| ; CHECK-LABEL: @test5( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: [[TMP:%.*]] = alloca i64, align 8 |
| ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq i64 [[ARG:%.*]], 0 |
| ; CHECK-NEXT: br i1 [[TMP1]], label [[BB28:%.*]], label [[BB2:%.*]] |
| ; CHECK: bb2: |
| ; CHECK-NEXT: br label [[BB7:%.*]] |
| ; CHECK: bb4: |
| ; CHECK-NEXT: br label [[BB5:%.*]] |
| ; CHECK: bb5: |
| ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i64 [[TMP9:%.*]], 0 |
| ; CHECK-NEXT: br i1 [[TMP6]], label [[BB27:%.*]], label [[BB7]] |
| ; CHECK: bb7: |
| ; CHECK-NEXT: [[TMP8:%.*]] = phi i64 [ [[ARG]], [[BB2]] ], [ [[TMP9]], [[BB5]] ] |
| ; CHECK-NEXT: [[TMP9]] = add nsw i64 [[TMP8]], -1 |
| ; CHECK-NEXT: [[TMP10:%.*]] = load i64, ptr @global, align 16 |
| ; CHECK-NEXT: [[TMP11:%.*]] = load i64, ptr @global.1, align 16 |
| ; CHECK-NEXT: [[TMP12:%.*]] = mul nsw i64 [[TMP11]], [[TMP10]] |
| ; CHECK-NEXT: [[TMP13:%.*]] = icmp eq i64 [[TMP12]], 0 |
| ; CHECK-NEXT: br i1 [[TMP13]], label [[BB5]], label [[BB14:%.*]] |
| ; CHECK: bb14: |
| ; CHECK-NEXT: br label [[BB15:%.*]] |
| ; CHECK: bb15: |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi i64 [ [[TMP12]], [[BB14]] ], [ [[TMP25:%.*]], [[BB15]] ] |
| ; CHECK-NEXT: [[TMP16:%.*]] = phi i64 [ [[TMP24:%.*]], [[BB15]] ], [ [[TMP11]], [[BB14]] ] |
| ; CHECK-NEXT: [[TMP17:%.*]] = phi i64 [ [[TMP22:%.*]], [[BB15]] ], [ [[TMP10]], [[BB14]] ] |
| ; CHECK-NEXT: [[TMP18:%.*]] = phi i64 [ [[TMP20:%.*]], [[BB15]] ], [ 0, [[BB14]] ] |
| ; CHECK-NEXT: store i64 [[PHIOFOPS]], ptr [[TMP]], align 8 |
| ; CHECK-NEXT: [[TMP20]] = add nuw nsw i64 [[TMP18]], 1 |
| ; CHECK-NEXT: [[TMP21:%.*]] = getelementptr inbounds [100 x i64], ptr @global, i64 0, i64 [[TMP20]] |
| ; CHECK-NEXT: [[TMP22]] = load i64, ptr [[TMP21]], align 8 |
| ; CHECK-NEXT: [[TMP23:%.*]] = getelementptr inbounds [100 x i64], ptr @global.1, i64 0, i64 [[TMP20]] |
| ; CHECK-NEXT: [[TMP24]] = load i64, ptr [[TMP23]], align 8 |
| ; CHECK-NEXT: [[TMP25]] = mul nsw i64 [[TMP24]], [[TMP22]] |
| ; CHECK-NEXT: [[TMP26:%.*]] = icmp eq i64 [[TMP20]], [[TMP25]] |
| ; CHECK-NEXT: br i1 [[TMP26]], label [[BB4:%.*]], label [[BB15]] |
| ; CHECK: bb27: |
| ; CHECK-NEXT: br label [[BB28]] |
| ; CHECK: bb28: |
| ; CHECK-NEXT: ret i64 0 |
| ; |
| bb: |
| %tmp = alloca i64, align 8 |
| %tmp1 = icmp eq i64 %arg, 0 |
| br i1 %tmp1, label %bb28, label %bb2 |
| |
| bb2: ; preds = %bb |
| br label %bb7 |
| |
| bb4: ; preds = %bb15 |
| br label %bb5 |
| |
| bb5: ; preds = %bb7, %bb4 |
| %tmp6 = icmp eq i64 %tmp9, 0 |
| br i1 %tmp6, label %bb27, label %bb7 |
| |
| bb7: ; preds = %bb5, %bb2 |
| %tmp8 = phi i64 [ %arg, %bb2 ], [ %tmp9, %bb5 ] |
| %tmp9 = add nsw i64 %tmp8, -1 |
| %tmp10 = load i64, ptr @global, align 16 |
| %tmp11 = load i64, ptr @global.1, align 16 |
| %tmp12 = mul nsw i64 %tmp11, %tmp10 |
| %tmp13 = icmp eq i64 %tmp12, 0 |
| br i1 %tmp13, label %bb5, label %bb14 |
| |
| bb14: ; preds = %bb7 |
| br label %bb15 |
| |
| bb15: ; preds = %bb15, %bb14 |
| %tmp16 = phi i64 [ %tmp24, %bb15 ], [ %tmp11, %bb14 ] |
| %tmp17 = phi i64 [ %tmp22, %bb15 ], [ %tmp10, %bb14 ] |
| %tmp18 = phi i64 [ %tmp20, %bb15 ], [ 0, %bb14 ] |
| ;; This multiply is an op of phis which is really equivalent to phi(tmp25, tmp12) |
| %tmp19 = mul nsw i64 %tmp16, %tmp17 |
| store i64 %tmp19, ptr %tmp, align 8 |
| %tmp20 = add nuw nsw i64 %tmp18, 1 |
| %tmp21 = getelementptr inbounds [100 x i64], ptr @global, i64 0, i64 %tmp20 |
| %tmp22 = load i64, ptr %tmp21, align 8 |
| %tmp23 = getelementptr inbounds [100 x i64], ptr @global.1, i64 0, i64 %tmp20 |
| %tmp24 = load i64, ptr %tmp23, align 8 |
| %tmp25 = mul nsw i64 %tmp24, %tmp22 |
| %tmp26 = icmp eq i64 %tmp20, %tmp25 |
| br i1 %tmp26, label %bb4, label %bb15 |
| |
| bb27: ; preds = %bb5 |
| br label %bb28 |
| |
| bb28: ; preds = %bb27, %bb |
| ret i64 0 |
| } |
| |
| ;; These icmps are all equivalent to phis of constants |
| define i8 @test6(ptr %addr) { |
| ; CHECK-LABEL: @test6( |
| ; CHECK-NEXT: entry-block: |
| ; CHECK-NEXT: br label %main-loop |
| ; CHECK: main-loop: |
| ; CHECK-NEXT: [[PHIOFOPS1:%.*]] = phi i1 [ true, %entry-block ], [ false, [[CORE:%.*]] ] |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi i1 [ false, %entry-block ], [ true, [[CORE]] ] |
| ; CHECK-NEXT: [[PHI:%.*]] = phi i8 [ 0, %entry-block ], [ 1, [[CORE]] ] |
| ; CHECK-NEXT: store volatile i8 0, ptr [[ADDR:%.*]] |
| ; CHECK-NEXT: br i1 [[PHIOFOPS1]], label %busy-wait-phi-0, label [[EXIT:%.*]] |
| ; CHECK: busy-wait-phi-0: |
| ; CHECK-NEXT: [[LOAD:%.*]] = load volatile i8, ptr [[ADDR]] |
| ; CHECK-NEXT: [[ICMP:%.*]] = icmp eq i8 [[LOAD]], 0 |
| ; CHECK-NEXT: br i1 [[ICMP]], label %busy-wait-phi-0, label [[CORE]] |
| ; CHECK: core: |
| ; CHECK-NEXT: br i1 [[PHIOFOPS]], label [[TRAP:%.*]], label %main-loop |
| ; CHECK: trap: |
| ; CHECK-NEXT: ret i8 1 |
| ; CHECK: exit: |
| ; CHECK-NEXT: ret i8 0 |
| ; |
| entry-block: |
| br label %main-loop |
| |
| main-loop: |
| %phi = phi i8 [ 0, %entry-block ], [ 1, %core ] |
| %switch_0 = icmp eq i8 %phi, 0 |
| store volatile i8 0, ptr %addr |
| br i1 %switch_0, label %busy-wait-phi-0, label %exit |
| |
| busy-wait-phi-0: |
| %load = load volatile i8, ptr %addr |
| %icmp = icmp eq i8 %load, 0 |
| br i1 %icmp, label %busy-wait-phi-0, label %core |
| |
| core: |
| %switch_1 = icmp eq i8 %phi, 1 |
| br i1 %switch_1, label %trap, label %main-loop |
| |
| trap: |
| ret i8 1 |
| |
| exit: |
| ret i8 0 |
| } |
| |
| ; Test that we don't infinite loop simplifying |
| ; an undefined value that can go both ways. |
| define void @test7() { |
| ; CHECK-LABEL: @test7( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: br label [[BB1:%.*]] |
| ; CHECK: bb1: |
| ; CHECK-NEXT: br label [[BB1]] |
| ; |
| bb: |
| br label %bb1 |
| |
| bb1: ; preds = %bb1, %bb |
| %tmp = phi i32 [ undef, %bb ], [ %tmp3, %bb1 ] |
| %tmp2 = icmp eq i32 %tmp, 0 |
| %tmp3 = select i1 %tmp2, i32 1, i32 %tmp |
| br label %bb1 |
| } |
| |
| |
| |
| ; Test that we get a consistent answer about what the |
| ; value of this undefined select is. |
| define void @test8() { |
| ; CHECK-LABEL: @test8( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: br label [[BB1:%.*]] |
| ; CHECK: bb1: |
| ; CHECK-NEXT: br label [[BB1]] |
| ; |
| bb: |
| %tmp = select i1 undef, i8 0, i8 1 |
| br label %bb1 |
| |
| bb1: ; preds = %bb1, %bb |
| %tmp2 = phi i8 [ %tmp4, %bb1 ], [ %tmp, %bb ] |
| %tmp3 = icmp eq i8 %tmp2, 0 |
| %tmp4 = select i1 %tmp3, i8 1, i8 %tmp2 |
| br label %bb1 |
| } |
| |
| |
| ;; Make sure we handle the case where we later come up with an expression that we need |
| ;; for a phi of ops. |
| define void @test9() { |
| ; CHECK-LABEL: @test9( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: br label [[BB1:%.*]] |
| ; CHECK: bb1: |
| ; CHECK-NEXT: br i1 undef, label [[BB1]], label [[BB2:%.*]] |
| ; CHECK: bb2: |
| ; CHECK-NEXT: br label [[BB6:%.*]] |
| ; CHECK: bb6: |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi i32 [ -13, [[BB2]] ], [ [[TMP11:%.*]], [[BB6]] ] |
| ; CHECK-NEXT: [[TMP7:%.*]] = phi i32 [ 1, [[BB2]] ], [ [[TMP8:%.*]], [[BB6]] ] |
| ; CHECK-NEXT: [[TMP8]] = add nuw nsw i32 [[TMP7]], 1 |
| ; CHECK-NEXT: [[TMP11]] = add i32 -14, [[TMP8]] |
| ; CHECK-NEXT: br label [[BB6]] |
| ; |
| bb: |
| br label %bb1 |
| |
| bb1: ; preds = %bb1, %bb |
| br i1 undef, label %bb1, label %bb2 |
| |
| bb2: ; preds = %bb1 |
| %tmp = select i1 true, i32 -14, i32 -10 |
| %tmp3 = add i32 %tmp, 0 |
| %tmp4 = select i1 true, i32 -14, i32 -10 |
| %tmp5 = add i32 %tmp4, 0 |
| br label %bb6 |
| |
| bb6: ; preds = %bb6, %bb2 |
| %tmp7 = phi i32 [ 1, %bb2 ], [ %tmp13, %bb6 ] |
| %tmp8 = add nuw nsw i32 %tmp7, 1 |
| %tmp9 = add i32 %tmp3, %tmp7 |
| %tmp10 = select i1 false, i32 undef, i32 %tmp9 |
| %tmp11 = add i32 %tmp5, %tmp8 |
| %tmp12 = select i1 undef, i32 undef, i32 %tmp11 |
| %tmp13 = add nuw nsw i32 %tmp7, 1 |
| br label %bb6 |
| } |
| |
| ;; Ensure that we revisit predicateinfo operands at the right points in time. |
| define void @test10() { |
| ; CHECK-LABEL: @test10( |
| ; CHECK-NEXT: b: |
| ; CHECK-NEXT: br label [[G:%.*]] |
| ; CHECK: g: |
| ; CHECK-NEXT: [[N:%.*]] = phi ptr [ [[H:%.*]], [[I:%.*]] ], [ null, [[B:%.*]] ] |
| ; CHECK-NEXT: [[H]] = getelementptr i32, ptr [[N]], i64 1 |
| ; CHECK-NEXT: [[J:%.*]] = icmp eq ptr [[H]], inttoptr (i64 32 to ptr) |
| ; CHECK-NEXT: br i1 [[J]], label [[C:%.*]], label [[I]] |
| ; CHECK: i: |
| ; CHECK-NEXT: br i1 undef, label [[K:%.*]], label [[G]] |
| ; CHECK: k: |
| ; CHECK-NEXT: br i1 false, label [[C]], label [[O:%.*]] |
| ; CHECK: o: |
| ; CHECK-NEXT: br label [[C]] |
| ; CHECK: c: |
| ; CHECK-NEXT: ret void |
| ; |
| b: |
| %m = getelementptr i32, ptr null, i64 8 |
| br label %g |
| |
| g: ; preds = %i, %b |
| %n = phi ptr [ %h, %i ], [ null, %b ] |
| %h = getelementptr i32, ptr %n, i64 1 |
| %j = icmp eq ptr %h, %m |
| br i1 %j, label %c, label %i |
| |
| i: ; preds = %g |
| br i1 undef, label %k, label %g |
| |
| k: ; preds = %i |
| %l = icmp eq ptr %n, %m |
| br i1 %l, label %c, label %o |
| |
| o: ; preds = %k |
| br label %c |
| |
| c: ; preds = %o, %k, %g |
| %0 = phi ptr [ undef, %o ], [ %m, %k ], [ %m, %g ] |
| ret void |
| } |
| |
| ;; Ensure we handle VariableExpression properly. |
| define void @test11() { |
| ; CHECK-LABEL: @test11( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: br i1 undef, label [[BB1:%.*]], label [[BB2:%.*]] |
| ; CHECK: bb1: |
| ; CHECK-NEXT: br label [[BB2]] |
| ; CHECK: bb2: |
| ; CHECK-NEXT: [[TMP:%.*]] = phi i1 [ false, [[BB1]] ], [ true, [[BB:%.*]] ] |
| ; CHECK-NEXT: [[TMP3:%.*]] = call ptr @wombat() |
| ; CHECK-NEXT: [[TMP4:%.*]] = icmp ne ptr [[TMP3]], null |
| ; CHECK-NEXT: [[TMP5:%.*]] = and i1 [[TMP]], [[TMP4]] |
| ; CHECK-NEXT: br i1 [[TMP5]], label [[BB6:%.*]], label [[BB7:%.*]] |
| ; CHECK: bb6: |
| ; CHECK-NEXT: unreachable |
| ; CHECK: bb7: |
| ; CHECK-NEXT: ret void |
| ; |
| bb: |
| br i1 undef, label %bb1, label %bb2 |
| |
| bb1: ; preds = %bb |
| br label %bb2 |
| |
| bb2: ; preds = %bb1, %bb |
| %tmp = phi i1 [ false, %bb1 ], [ true, %bb ] |
| %tmp3 = call ptr @wombat() |
| %tmp4 = icmp ne ptr %tmp3, null |
| %tmp5 = and i1 %tmp, %tmp4 |
| br i1 %tmp5, label %bb6, label %bb7 |
| |
| bb6: ; preds = %bb2 |
| unreachable |
| |
| bb7: ; preds = %bb2 |
| ret void |
| } |
| |
| declare ptr @wombat() |
| |
| ;; Ensure that when reachability affects a phi of ops, we recompute |
| ;; it. Here, the phi node is marked for recomputation when bb7->bb3 |
| ;; becomes live, but the value does not change. if we do not directly |
| ;; recompute the phi of ops instruction (tmp5), the value number will |
| ;; change in the verifier, as it goes from a constant value to a |
| ;; phi of [true, false] |
| |
| define void @test12(ptr %p) { |
| ; CHECK-LABEL: @test12( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: [[TMP:%.*]] = load i32, ptr %p |
| ; CHECK-NEXT: [[TMP1:%.*]] = icmp sgt i32 [[TMP]], 0 |
| ; CHECK-NEXT: br i1 [[TMP1]], label [[BB2:%.*]], label [[BB8:%.*]] |
| ; CHECK: bb2: |
| ; CHECK-NEXT: br label [[BB3:%.*]] |
| ; CHECK: bb3: |
| ; CHECK-NEXT: br i1 true, label [[BB6:%.*]], label [[BB7]] |
| ; CHECK: bb6: |
| ; CHECK-NEXT: br label [[BB7]] |
| ; CHECK: bb7: |
| ; CHECK-NEXT: br label [[BB3]] |
| ; CHECK: bb8: |
| ; CHECK-NEXT: ret void |
| ; |
| bb: |
| %tmp = load i32, ptr %p |
| %tmp1 = icmp sgt i32 %tmp, 0 |
| br i1 %tmp1, label %bb2, label %bb8 |
| |
| bb2: ; preds = %bb |
| br label %bb3 |
| |
| bb3: ; preds = %bb7, %bb2 |
| %tmp4 = phi i32 [ %tmp, %bb2 ], [ poison, %bb7 ] |
| %tmp5 = icmp sgt i32 %tmp4, 0 |
| br i1 %tmp5, label %bb6, label %bb7 |
| |
| bb6: ; preds = %bb3 |
| br label %bb7 |
| |
| bb7: ; preds = %bb6, %bb3 |
| br label %bb3 |
| |
| bb8: ; preds = %bb |
| ret void |
| } |
| |
| ;; Make sure we reprocess phi of ops involving loads when loads change class. |
| ;; This is PR 34473 |
| define void @test13() { |
| ; CHECK-LABEL: @test13( |
| ; CHECK-NEXT: bb: |
| ; CHECK-NEXT: br label [[BB1:%.*]] |
| ; CHECK: bb1: |
| ; CHECK-NEXT: [[TMP:%.*]] = load i8, ptr null |
| ; CHECK-NEXT: br label [[BB3:%.*]] |
| ; CHECK: bb3: |
| ; CHECK-NEXT: [[PHIOFOPS:%.*]] = phi i8 [ [[TMP]], [[BB1]] ], [ [[TMP10:%.*]], [[BB3]] ] |
| ; CHECK-NEXT: [[TMP4:%.*]] = phi ptr [ null, [[BB1]] ], [ [[TMP6:%.*]], [[BB3]] ] |
| ; CHECK-NEXT: [[TMP5:%.*]] = phi i32 [ undef, [[BB1]] ], [ [[TMP9:%.*]], [[BB3]] ] |
| ; CHECK-NEXT: [[TMP6]] = getelementptr i8, ptr [[TMP4]], i64 1 |
| ; CHECK-NEXT: [[TMP8:%.*]] = sext i8 [[PHIOFOPS]] to i32 |
| ; CHECK-NEXT: [[TMP9]] = mul i32 [[TMP5]], [[TMP8]] |
| ; CHECK-NEXT: [[TMP10]] = load i8, ptr [[TMP6]] |
| ; CHECK-NEXT: [[TMP11:%.*]] = icmp eq i8 [[TMP10]], 0 |
| ; CHECK-NEXT: br i1 [[TMP11]], label [[BB12:%.*]], label [[BB3]] |
| ; CHECK: bb12: |
| ; CHECK-NEXT: [[TMP14:%.*]] = icmp eq i32 [[TMP9]], 0 |
| ; CHECK-NEXT: br i1 [[TMP14]], label [[BB1]], label [[BB15:%.*]] |
| ; CHECK: bb15: |
| ; CHECK-NEXT: call void (...) @bar() |
| ; CHECK-NEXT: br label [[BB1]] |
| ; |
| bb: |
| br label %bb1 |
| |
| bb1: ; preds = %bb15, %bb12, %bb |
| %tmp = load i8, ptr null |
| %tmp2 = icmp eq i8 %tmp, 8 |
| br label %bb3 |
| |
| bb3: ; preds = %bb3, %bb1 |
| %tmp4 = phi ptr [ null, %bb1 ], [ %tmp6, %bb3 ] |
| %tmp5 = phi i32 [ undef, %bb1 ], [ %tmp9, %bb3 ] |
| %tmp6 = getelementptr i8, ptr %tmp4, i64 1 |
| %tmp7 = load i8, ptr %tmp4 |
| %tmp8 = sext i8 %tmp7 to i32 |
| %tmp9 = mul i32 %tmp5, %tmp8 |
| %tmp10 = load i8, ptr %tmp6 |
| %tmp11 = icmp eq i8 %tmp10, 0 |
| br i1 %tmp11, label %bb12, label %bb3 |
| |
| bb12: ; preds = %bb3 |
| %tmp13 = phi i32 [ %tmp9, %bb3 ] |
| %tmp14 = icmp eq i32 %tmp13, 0 |
| br i1 %tmp14, label %bb1, label %bb15 |
| |
| bb15: ; preds = %bb12 |
| call void (...) @bar() |
| br label %bb1 |
| } |
| |
| declare void @bar(...) |
| |