; NOTE: Assertions have been autogenerated by utils/update_test_checks.py ; RUN: opt -S %s -passes='loop-mssa(licm)' -verify-memoryssa | FileCheck %s ; turn to %iv u umax(inv_1, inv_2) and hoist it out of loop. define i32 @test_ugt(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_ugt( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp ugt i32 [[IV]], [[INVARIANT_UMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ugt i32 %iv, %inv_1 %cmp_2 = icmp ugt i32 %iv, %inv_2 %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv >=u umax(inv_1, inv_2) and hoist it out of loop. define i32 @test_uge(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_uge( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp uge i32 [[IV]], [[INVARIANT_UMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp uge i32 %iv, %inv_1 %cmp_2 = icmp uge i32 %iv, %inv_2 %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv >s smax(inv_1, inv_2) and hoist it out of loop. define i32 @test_sgt(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sgt( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMAX:%.*]] = call i32 @llvm.smax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sgt i32 [[IV]], [[INVARIANT_SMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp sgt i32 %iv, %inv_1 %cmp_2 = icmp sgt i32 %iv, %inv_2 %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv >=s smax(inv_1, inv_2) and hoist it out of loop. define i32 @test_sge(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sge( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMAX:%.*]] = call i32 @llvm.smax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sge i32 [[IV]], [[INVARIANT_SMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp sge i32 %iv, %inv_1 %cmp_2 = icmp sge i32 %iv, %inv_2 %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_ult_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_ult_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp ult i32 [[IV]], [[INVARIANT_UMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ult i32 %iv, %inv_1 %cmp_2 = icmp ult i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_ule_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_ule_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp ule i32 [[IV]], [[INVARIANT_UMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ule i32 %iv, %inv_1 %cmp_2 = icmp ule i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_slt_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_slt_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMAX:%.*]] = call i32 @llvm.smax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp slt i32 [[IV]], [[INVARIANT_SMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp slt i32 %iv, %inv_1 %cmp_2 = icmp slt i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_sle_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sle_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMAX:%.*]] = call i32 @llvm.smax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sle i32 [[IV]], [[INVARIANT_SMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp sle i32 %iv, %inv_1 %cmp_2 = icmp sle i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_ugt_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_ugt_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMIN:%.*]] = call i32 @llvm.umin.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp ugt i32 [[IV]], [[INVARIANT_UMIN]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ugt i32 %iv, %inv_1 %cmp_2 = icmp ugt i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_uge_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_uge_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMIN:%.*]] = call i32 @llvm.umin.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp uge i32 [[IV]], [[INVARIANT_UMIN]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp uge i32 %iv, %inv_1 %cmp_2 = icmp uge i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_sgt_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sgt_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMIN:%.*]] = call i32 @llvm.smin.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sgt i32 [[IV]], [[INVARIANT_SMIN]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp sgt i32 %iv, %inv_1 %cmp_2 = icmp sgt i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; Turn OR to AND and handle accordingly. define i32 @test_sge_inv(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sge_inv( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMIN:%.*]] = call i32 @llvm.smin.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sge i32 [[IV]], [[INVARIANT_SMIN]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp sge i32 %iv, %inv_1 %cmp_2 = icmp sge i32 %iv, %inv_2 %loop_cond = or i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv u umax(inv_1, inv_2) and hoist it out of loop. define i32 @test_ugt_swapped(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_ugt_swapped( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp ugt i32 [[IV]], [[INVARIANT_UMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ult i32 %inv_1, %iv %cmp_2 = icmp ult i32 %inv_2, %iv %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv >=u umax(inv_1, inv_2) and hoist it out of loop. define i32 @test_uge_swapped(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_uge_swapped( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_UMAX:%.*]] = call i32 @llvm.umax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp uge i32 [[IV]], [[INVARIANT_UMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ule i32 %inv_1, %iv %cmp_2 = icmp ule i32 %inv_2, %iv %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv >s smax(inv_1, inv_2) and hoist it out of loop. define i32 @test_sgt_swapped(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sgt_swapped( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMAX:%.*]] = call i32 @llvm.smax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sgt i32 [[IV]], [[INVARIANT_SMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp slt i32 %inv_1, %iv %cmp_2 = icmp slt i32 %inv_2, %iv %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv >=s smax(inv_1, inv_2) and hoist it out of loop. define i32 @test_sge_swapped(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_sge_swapped( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INVARIANT_SMAX:%.*]] = call i32 @llvm.smax.i32(i32 [[INV_1:%.*]], i32 [[INV_2:%.*]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp sge i32 [[IV]], [[INVARIANT_SMAX]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp sle i32 %inv_1, %iv %cmp_2 = icmp sle i32 %inv_2, %iv %loop_cond = and i1 %cmp_1, %cmp_2 %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; inv_2 needs freeze because hoisted umax would create a new use that didn't exist before. ; Counter-example: %cmp_1 = false, %inv_2 = poison. define i32 @test_logical_and_ult_needs_freeze(i32 %start, i32 %inv_1, i32 %inv_2) { ; CHECK-LABEL: @test_logical_and_ult_needs_freeze( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[INV_2_FR:%.*]] = freeze i32 [[INV_2:%.*]] ; CHECK-NEXT: [[INVARIANT_UMIN:%.*]] = call i32 @llvm.umin.i32(i32 [[INV_1:%.*]], i32 [[INV_2_FR]]) ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ [[START:%.*]], [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP]] ] ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp ult i32 [[IV]], [[INVARIANT_UMIN]] ; CHECK-NEXT: [[IV_NEXT]] = add i32 [[IV]], 1 ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_LCSSA:%.*]] = phi i32 [ [[IV]], [[LOOP]] ] ; CHECK-NEXT: ret i32 [[IV_LCSSA]] ; entry: br label %loop loop: %iv = phi i32 [%start, %entry], [%iv.next, %loop] %cmp_1 = icmp ult i32 %iv, %inv_1 %cmp_2 = icmp ult i32 %iv, %inv_2 %loop_cond = select i1 %cmp_1, i1 %cmp_2, i1 false %iv.next = add i32 %iv, 1 br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv } ; turn to %iv