diff options
Diffstat (limited to 'llvm/test/Transforms/UnifyLoopExits/nested.ll')
| -rw-r--r-- | llvm/test/Transforms/UnifyLoopExits/nested.ll | 142 | 
1 files changed, 142 insertions, 0 deletions
| diff --git a/llvm/test/Transforms/UnifyLoopExits/nested.ll b/llvm/test/Transforms/UnifyLoopExits/nested.ll index 8fae2c4..2ec576a 100644 --- a/llvm/test/Transforms/UnifyLoopExits/nested.ll +++ b/llvm/test/Transforms/UnifyLoopExits/nested.ll @@ -78,3 +78,145 @@ exit:    %exit.phi = phi i32 [%A4.phi, %A5], [%Z, %C]    ret void  } + +define void @nested_callbr(i1 %PredB3, i1 %PredB4, i1 %PredA4, i1 %PredA3, i32 %X, i32 %Y, i32 %Z) { +; CHECK-LABEL: @nested_callbr( +; CHECK-NEXT:  entry: +; CHECK-NEXT:    br label [[A1:%.*]] +; CHECK:       A1: +; CHECK-NEXT:    br label [[B1:%.*]] +; CHECK:       B1: +; CHECK-NEXT:    br label [[B2:%.*]] +; CHECK:       B2: +; CHECK-NEXT:    [[X_INC:%.*]] = add i32 [[X:%.*]], 1 +; CHECK-NEXT:    br label [[B3:%.*]] +; CHECK:       B3: +; CHECK-NEXT:    callbr void asm "", "r,!i"(i1 [[PREDB3:%.*]]) +; CHECK-NEXT:            to label [[B4:%.*]] [label %B3.target.A3] +; CHECK:       B4: +; CHECK-NEXT:    callbr void asm "", "r,!i"(i1 [[PREDB4:%.*]]) +; CHECK-NEXT:            to label [[B1]] [label %B4.target.A2] +; CHECK:       A2: +; CHECK-NEXT:    br label [[A4:%.*]] +; CHECK:       A3: +; CHECK-NEXT:    br label [[A4]] +; CHECK:       A4: +; CHECK-NEXT:    [[A4_PHI:%.*]] = phi i32 [ [[Y:%.*]], [[A3:%.*]] ], [ [[X_INC_MOVED:%.*]], [[A2:%.*]] ] +; CHECK-NEXT:    callbr void asm "", "r,!i"(i1 [[PREDA4:%.*]]) +; CHECK-NEXT:            to label [[A4_TARGET_C:%.*]] [label %A5] +; CHECK:       A5: +; CHECK-NEXT:    callbr void asm "", "r,!i"(i1 [[PREDA3:%.*]]) +; CHECK-NEXT:            to label [[A5_TARGET_EXIT:%.*]] [label %A1] +; CHECK:       C: +; CHECK-NEXT:    br label [[EXIT:%.*]] +; CHECK:       exit: +; CHECK-NEXT:    [[EXIT_PHI:%.*]] = phi i32 [ [[Z:%.*]], [[C:%.*]] ], [ [[EXIT_PHI_MOVED:%.*]], [[LOOP_EXIT_GUARD:%.*]] ] +; CHECK-NEXT:    ret void +; CHECK:       A4.target.C: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD]] +; CHECK:       A5.target.exit: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD]] +; CHECK:       loop.exit.guard: +; CHECK-NEXT:    [[EXIT_PHI_MOVED]] = phi i32 [ poison, [[A4_TARGET_C]] ], [ [[A4_PHI]], [[A5_TARGET_EXIT]] ] +; CHECK-NEXT:    [[GUARD_C:%.*]] = phi i1 [ true, [[A4_TARGET_C]] ], [ false, [[A5_TARGET_EXIT]] ] +; CHECK-NEXT:    br i1 [[GUARD_C]], label [[C]], label [[EXIT]] +; CHECK:       B3.target.A3: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD1:%.*]] +; CHECK:       B4.target.A2: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD1]] +; CHECK:       loop.exit.guard1: +; CHECK-NEXT:    [[X_INC_MOVED]] = phi i32 [ [[X_INC]], [[B3_TARGET_A3:%.*]] ], [ [[X_INC]], [[B4_TARGET_A2:%.*]] ] +; CHECK-NEXT:    [[GUARD_A3:%.*]] = phi i1 [ true, [[B3_TARGET_A3]] ], [ false, [[B4_TARGET_A2]] ] +; CHECK-NEXT:    br i1 [[GUARD_A3]], label [[A3]], label [[A2]] +; +entry: +  br label %A1 + +A1: +  br label %B1 + +B1: +  br label %B2 + +B2: +  %X.inc = add i32 %X, 1 +  br label %B3 + +B3: +  callbr void asm "", "r,!i"(i1 %PredB3) to label %B4 [label %A3] + +B4: +  callbr void asm "", "r,!i"(i1 %PredB4) to label %B1 [label %A2] + +A2: +  br label %A4 + +A3: +  br label %A4 + +A4: +  %A4.phi = phi i32 [%Y, %A3], [%X.inc, %A2] +  callbr void asm "", "r,!i"(i1 %PredA4) to label %C [label %A5] + +A5: +  callbr void asm "", "r,!i"(i1 %PredA3) to label %exit [label %A1] + +C: +  br label %exit + +exit: +  %exit.phi = phi i32 [%A4.phi, %A5], [%Z, %C] +  ret void +} + +; Here, the newly created target loop that connects b to r1 needs to be part of +; the parent loop (the outer loop b participates in). Otherwise, it will be +; regarded as an additional loop entry point to this outer loop. +define void @nested_callbr_multiple_exits() { +; CHECK-LABEL: @nested_callbr_multiple_exits( +; CHECK-NEXT:    br label [[A:%.*]] +; CHECK:       a: +; CHECK-NEXT:    callbr void asm "", ""() +; CHECK-NEXT:            to label [[B:%.*]] [] +; CHECK:       b: +; CHECK-NEXT:    callbr void asm "", "!i"() +; CHECK-NEXT:            to label [[C:%.*]] [label %b.target.b.target.r1] +; CHECK:       c: +; CHECK-NEXT:    callbr void asm "", "!i"() +; CHECK-NEXT:            to label [[C_TARGET_E:%.*]] [label %b] +; CHECK:       e: +; CHECK-NEXT:    callbr void asm "", "!i"() +; CHECK-NEXT:            to label [[A]] [label %e.target.r2] +; CHECK:       r1: +; CHECK-NEXT:    ret void +; CHECK:       r2: +; CHECK-NEXT:    ret void +; CHECK:       b.target.r1: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD:%.*]] +; CHECK:       e.target.r2: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD]] +; CHECK:       loop.exit.guard: +; CHECK-NEXT:    [[GUARD_R1:%.*]] = phi i1 [ true, [[B_TARGET_R1:%.*]] ], [ false, [[E_TARGET_R2:%.*]] ] +; CHECK-NEXT:    br i1 [[GUARD_R1]], label [[R1:%.*]], label [[R2:%.*]] +; CHECK:       b.target.b.target.r1: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD1:%.*]] +; CHECK:       c.target.e: +; CHECK-NEXT:    br label [[LOOP_EXIT_GUARD1]] +; CHECK:       loop.exit.guard1: +; CHECK-NEXT:    [[GUARD_B_TARGET_R1:%.*]] = phi i1 [ true, [[B_TARGET_B_TARGET_R1:%.*]] ], [ false, [[C_TARGET_E]] ] +; CHECK-NEXT:    br i1 [[GUARD_B_TARGET_R1]], label [[B_TARGET_R1]], label [[E:%.*]] +; +  br label %a +a: +  callbr void asm "", ""() to label %b [] +b: +  callbr void asm "", "!i"() to label %c [label %r1] +c: +  callbr void asm "", "!i"() to label %e [label %b] +e: +  callbr void asm "", "!i"() to label %a [label %r2] +r1: +  ret void +r2: +  ret void +} | 
