; NOTE: Assertions have been autogenerated by utils/update_test_checks.py ; RUN: opt < %s -passes='lower-switch,unify-loop-exits' -S | FileCheck %s define void @loop_1(i1 %PredEntry, i1 %PredB, i1 %PredC, i1 %PredD) { ; CHECK-LABEL: @loop_1( ; CHECK-NEXT: entry: ; CHECK-NEXT: br i1 [[PREDENTRY:%.*]], label [[A:%.*]], label [[G:%.*]] ; CHECK: A: ; CHECK-NEXT: br label [[B:%.*]] ; CHECK: B: ; CHECK-NEXT: br i1 [[PREDB:%.*]], label [[C:%.*]], label [[LOOP_EXIT_GUARD:%.*]] ; CHECK: C: ; CHECK-NEXT: br i1 [[PREDC:%.*]], label [[D:%.*]], label [[LOOP_EXIT_GUARD]] ; CHECK: D: ; CHECK-NEXT: br i1 [[PREDD:%.*]], label [[A]], label [[LOOP_EXIT_GUARD]] ; CHECK: E: ; CHECK-NEXT: br label [[EXIT:%.*]] ; CHECK: F: ; CHECK-NEXT: br label [[EXIT]] ; CHECK: G: ; CHECK-NEXT: br label [[Y:%.*]] ; CHECK: exit: ; CHECK-NEXT: ret void ; CHECK: loop.exit.guard: ; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[B]] ], [ false, [[C]] ], [ false, [[D]] ] ; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[Y]] ; entry: br i1 %PredEntry, label %A, label %G A: br label %B B: br i1 %PredB, label %C, label %E C: br i1 %PredC, label %D, label %F D: br i1 %PredD, label %A, label %F E: br label %exit F: br label %exit G: br label %F exit: ret void } define void @loop_1_callbr(i1 %PredEntry, i1 %PredB, i1 %PredC, i1 %PredD) { ; CHECK-LABEL: @loop_1_callbr( ; CHECK-NEXT: entry: ; CHECK-NEXT: br i1 [[PREDENTRY:%.*]], label [[A:%.*]], label [[G:%.*]] ; CHECK: A: ; CHECK-NEXT: br label [[B:%.*]] ; CHECK: B: ; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDB:%.*]]) ; CHECK-NEXT: to label [[C:%.*]] [label [[B_TARGET_E:%.*]]] ; CHECK: C: ; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDC:%.*]]) ; CHECK-NEXT: to label [[D:%.*]] [label [[C_TARGET_F:%.*]]] ; CHECK: D: ; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDD:%.*]]) ; CHECK-NEXT: to label [[A]] [label [[D_TARGET_F:%.*]]] ; CHECK: E: ; CHECK-NEXT: br label [[EXIT:%.*]] ; CHECK: F: ; CHECK-NEXT: br label [[EXIT]] ; CHECK: G: ; CHECK-NEXT: br label [[Y:%.*]] ; CHECK: exit: ; CHECK-NEXT: ret void ; CHECK: B.target.E: ; CHECK-NEXT: br label [[LOOP_EXIT_GUARD:%.*]] ; CHECK: C.target.F: ; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]] ; CHECK: D.target.F: ; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]] ; CHECK: loop.exit.guard: ; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[B_TARGET_E]] ], [ false, [[C_TARGET_F]] ], [ false, [[D_TARGET_F]] ] ; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[Y]] ; entry: br i1 %PredEntry, label %A, label %G A: br label %B B: callbr void asm "", "r,!i"(i1 %PredB) to label %C [label %E] C: callbr void asm "", "r,!i"(i1 %PredC) to label %D [label %F] D: callbr void asm "", "r,!i"(i1 %PredD) to label %A [label %F] E: br label %exit F: br label %exit G: br label %F exit: ret void } define void @loop_2(i1 %PredA, i1 %PredB, i1 %PredC) { ; CHECK-LABEL: @loop_2( ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[A:%.*]] ; CHECK: A: ; CHECK-NEXT: br i1 [[PREDA:%.*]], label [[B:%.*]], label [[LOOP_EXIT_GUARD:%.*]] ; CHECK: B: ; CHECK-NEXT: br i1 [[PREDB:%.*]], label [[C:%.*]], label [[LOOP_EXIT_GUARD]] ; CHECK: C: ; CHECK-NEXT: br i1 [[PREDC:%.*]], label [[D:%.*]], label [[LOOP_EXIT_GUARD]] ; CHECK: D: ; CHECK-NEXT: br label [[A]] ; CHECK: X: ; CHECK-NEXT: br label [[EXIT:%.*]] ; CHECK: Y: ; CHECK-NEXT: br label [[EXIT]] ; CHECK: Z: ; CHECK-NEXT: br label [[EXIT]] ; CHECK: exit: ; CHECK-NEXT: ret void ; CHECK: loop.exit.guard: ; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[A]] ], [ false, [[B]] ], [ false, [[C]] ] ; CHECK-NEXT: [[GUARD_Y:%.*]] = phi i1 [ false, [[A]] ], [ true, [[B]] ], [ false, [[C]] ] ; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[LOOP_EXIT_GUARD1:%.*]] ; CHECK: loop.exit.guard1: ; CHECK-NEXT: br i1 [[GUARD_Y]], label [[Y:%.*]], label [[Z:%.*]] ; entry: br label %A A: br i1 %PredA, label %B, label %X B: br i1 %PredB, label %C, label %Y C: br i1 %PredC, label %D, label %Z D: br label %A X: br label %exit Y: br label %exit Z: br label %exit exit: ret void } define void @loop_2_callbr(i1 %PredA, i1 %PredB, i1 %PredC) { ; CHECK-LABEL: @loop_2_callbr( ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[A:%.*]] ; CHECK: A: ; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDA:%.*]]) ; CHECK-NEXT: to label [[B:%.*]] [label [[A_TARGET_X:%.*]]] ; CHECK: B: ; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDB:%.*]]) ; CHECK-NEXT: to label [[C:%.*]] [label [[B_TARGET_Y:%.*]]] ; CHECK: C: ; CHECK-NEXT: callbr void asm "", "r,!i"(i1 [[PREDC:%.*]]) ; CHECK-NEXT: to label [[D:%.*]] [label [[C_TARGET_Z:%.*]]] ; CHECK: D: ; CHECK-NEXT: br label [[A]] ; CHECK: X: ; CHECK-NEXT: br label [[EXIT:%.*]] ; CHECK: Y: ; CHECK-NEXT: br label [[EXIT]] ; CHECK: Z: ; CHECK-NEXT: br label [[EXIT]] ; CHECK: exit: ; CHECK-NEXT: ret void ; CHECK: A.target.X: ; CHECK-NEXT: br label [[LOOP_EXIT_GUARD:%.*]] ; CHECK: B.target.Y: ; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]] ; CHECK: C.target.Z: ; CHECK-NEXT: br label [[LOOP_EXIT_GUARD]] ; CHECK: loop.exit.guard: ; CHECK-NEXT: [[GUARD_X:%.*]] = phi i1 [ true, [[A_TARGET_X]] ], [ false, [[B_TARGET_Y]] ], [ false, [[C_TARGET_Z]] ] ; CHECK-NEXT: [[GUARD_Y:%.*]] = phi i1 [ false, [[A_TARGET_X]] ], [ true, [[B_TARGET_Y]] ], [ false, [[C_TARGET_Z]] ] ; CHECK-NEXT: br i1 [[GUARD_X]], label [[X:%.*]], label [[LOOP_EXIT_GUARD1:%.*]] ; CHECK: loop.exit.guard1: ; CHECK-NEXT: br i1 [[GUARD_Y]], label [[Y:%.*]], label [[Z:%.*]] ; entry: br label %A A: callbr void asm "", "r,!i"(i1 %PredA) to label %B [label %X] B: callbr void asm "", "r,!i"(i1 %PredB) to label %C [label %Y] C: callbr void asm "", "r,!i"(i1 %PredC) to label %D [label %Z] D: br label %A X: br label %exit Y: br label %exit Z: br label %exit exit: ret void } ; Test that UnifyLoopExits handles callbr with duplicate successors correctly. ; The exit block appears twice as a successor of the callbr instruction. define void @callbr_duplicate_successors() { ; CHECK-LABEL: @callbr_duplicate_successors( ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: callbr void asm sideeffect "", "!i,!i"() ; CHECK-NEXT: to label [[LOOP_TARGET_EXIT:%.*]] [label [[LOOP]], label [[LOOP_TARGET_EXIT1:%.*]]] ; CHECK: exit: ; CHECK-NEXT: ret void ; CHECK: loop.target.exit: ; CHECK-NEXT: br label [[EXIT:%.*]] ; CHECK: loop.target.exit1: ; CHECK-NEXT: br label [[EXIT]] ; entry: br label %loop loop: callbr void asm sideeffect "", "!i,!i"() to label %exit [label %loop, label %exit] exit: ret void }