; NOTE: Assertions have been autogenerated by utils/update_test_checks.py ; RUN: opt -S -passes=indvars < %s | FileCheck %s ; Check that we replace signed comparisons between non-negative values with ; unsigned comparisons if we can. target datalayout = "n8:16:32:64" define i32 @test_01(i32 %a, i32 %b, ptr %p) { ; CHECK-LABEL: @test_01( ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_ENTRY:%.*]] ; CHECK: loop.entry: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP_BE:%.*]] ] ; CHECK-NEXT: [[CMP1:%.*]] = icmp ult i32 [[IV]], 100 ; CHECK-NEXT: br i1 [[CMP1]], label [[B1:%.*]], label [[B2:%.*]] ; CHECK: b1: ; CHECK-NEXT: store i32 [[IV]], ptr [[P:%.*]], align 4 ; CHECK-NEXT: br label [[MERGE:%.*]] ; CHECK: b2: ; CHECK-NEXT: store i32 [[A:%.*]], ptr [[P]], align 4 ; CHECK-NEXT: br label [[MERGE]] ; CHECK: merge: ; CHECK-NEXT: [[CMP2:%.*]] = icmp ult i32 [[IV]], 100 ; CHECK-NEXT: br i1 [[CMP2]], label [[B3:%.*]], label [[B4:%.*]] ; CHECK: b3: ; CHECK-NEXT: store i32 [[IV]], ptr [[P]], align 4 ; CHECK-NEXT: br label [[LOOP_BE]] ; CHECK: b4: ; CHECK-NEXT: store i32 [[B:%.*]], ptr [[P]], align 4 ; CHECK-NEXT: br label [[LOOP_BE]] ; CHECK: loop.be: ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i32 [[IV]], 1 ; CHECK-NEXT: [[EXITCOND:%.*]] = icmp ne i32 [[IV_NEXT]], 1000 ; CHECK-NEXT: br i1 [[EXITCOND]], label [[LOOP_ENTRY]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: ret i32 999 ; entry: br label %loop.entry loop.entry: %iv = phi i32 [ 0, %entry ], [ %iv.next, %loop.be ] %cmp1 = icmp slt i32 %iv, 100 br i1 %cmp1, label %b1, label %b2 b1: store i32 %iv, ptr %p br label %merge b2: store i32 %a, ptr %p br label %merge merge: %cmp2 = icmp ult i32 %iv, 100 br i1 %cmp2, label %b3, label %b4 b3: store i32 %iv, ptr %p br label %loop.be b4: store i32 %b, ptr %p br label %loop.be loop.be: %iv.next = add i32 %iv, 1 %cmp3 = icmp slt i32 %iv.next, 1000 br i1 %cmp3, label %loop.entry, label %exit exit: ret i32 %iv } define i32 @test_02(i32 %a, i32 %b, ptr %p) { ; CHECK-LABEL: @test_02( ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_ENTRY:%.*]] ; CHECK: loop.entry: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[LOOP_BE:%.*]] ] ; CHECK-NEXT: [[CMP1:%.*]] = icmp ugt i32 100, [[IV]] ; CHECK-NEXT: br i1 [[CMP1]], label [[B1:%.*]], label [[B2:%.*]] ; CHECK: b1: ; CHECK-NEXT: store i32 [[IV]], ptr [[P:%.*]], align 4 ; CHECK-NEXT: br label [[MERGE:%.*]] ; CHECK: b2: ; CHECK-NEXT: store i32 [[A:%.*]], ptr [[P]], align 4 ; CHECK-NEXT: br label [[MERGE]] ; CHECK: merge: ; CHECK-NEXT: [[CMP2:%.*]] = icmp ugt i32 100, [[IV]] ; CHECK-NEXT: br i1 [[CMP2]], label [[B3:%.*]], label [[B4:%.*]] ; CHECK: b3: ; CHECK-NEXT: store i32 [[IV]], ptr [[P]], align 4 ; CHECK-NEXT: br label [[LOOP_BE]] ; CHECK: b4: ; CHECK-NEXT: store i32 [[B:%.*]], ptr [[P]], align 4 ; CHECK-NEXT: br label [[LOOP_BE]] ; CHECK: loop.be: ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i32 [[IV]], 1 ; CHECK-NEXT: [[EXITCOND:%.*]] = icmp ne i32 [[IV_NEXT]], 1000 ; CHECK-NEXT: br i1 [[EXITCOND]], label [[LOOP_ENTRY]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: ret i32 999 ; entry: br label %loop.entry loop.entry: %iv = phi i32 [ 0, %entry ], [ %iv.next, %loop.be ] %cmp1 = icmp sgt i32 100, %iv br i1 %cmp1, label %b1, label %b2 b1: store i32 %iv, ptr %p br label %merge b2: store i32 %a, ptr %p br label %merge merge: %cmp2 = icmp ugt i32 100, %iv br i1 %cmp2, label %b3, label %b4 b3: store i32 %iv, ptr %p br label %loop.be b4: store i32 %b, ptr %p br label %loop.be loop.be: %iv.next = add i32 %iv, 1 %cmp3 = icmp sgt i32 1000, %iv.next br i1 %cmp3, label %loop.entry, label %exit exit: ret i32 %iv } define i32 @test_03(ptr %p, ptr %capacity_p, ptr %num_elements_p) { ; CHECK-LABEL: @test_03( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[CAPACITY:%.*]] = load i32, ptr [[CAPACITY_P:%.*]], align 4, !range [[RNG0:![0-9]+]] ; CHECK-NEXT: [[NUM_ELEMENTS:%.*]] = load i32, ptr [[NUM_ELEMENTS_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[BACKEDGE:%.*]] ] ; CHECK-NEXT: [[BYTES_TO_WRITE:%.*]] = sub nuw nsw i32 [[CAPACITY]], [[IV]] ; CHECK-NEXT: [[CAPACITY_CHECK:%.*]] = icmp slt i32 [[BYTES_TO_WRITE]], 4 ; CHECK-NEXT: br i1 [[CAPACITY_CHECK]], label [[OUT_OF_BOUNDS:%.*]], label [[BACKEDGE]] ; CHECK: backedge: ; CHECK-NEXT: [[EL_PTR:%.*]] = getelementptr i32, ptr [[P:%.*]], i32 [[IV]] ; CHECK-NEXT: store i32 1, ptr [[EL_PTR]], align 4 ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i32 [[IV]], 4 ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp slt i32 [[IV_NEXT]], [[NUM_ELEMENTS]] ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_NEXT_LCSSA:%.*]] = phi i32 [ [[IV_NEXT]], [[BACKEDGE]] ] ; CHECK-NEXT: ret i32 [[IV_NEXT_LCSSA]] ; CHECK: out_of_bounds: ; CHECK-NEXT: ret i32 -1 ; entry: %capacity = load i32, ptr %capacity_p, !range !0 %num_elements = load i32, ptr %num_elements_p, !range !0 br label %loop loop: %iv = phi i32 [0, %entry], [%iv.next, %backedge] %bytes_to_write = sub i32 %capacity, %iv %capacity_check = icmp slt i32 %bytes_to_write, 4 br i1 %capacity_check, label %out_of_bounds, label %backedge backedge: %el.ptr = getelementptr i32, ptr %p, i32 %iv store i32 1, ptr %el.ptr %iv.next = add nuw nsw i32 %iv, 4 %loop_cond = icmp slt i32 %iv.next, %num_elements br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv.next out_of_bounds: ret i32 -1 } define i32 @test_04(ptr %p, ptr %capacity_p, ptr %num_elements_p) { ; CHECK-LABEL: @test_04( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[CAPACITY:%.*]] = load i32, ptr [[CAPACITY_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: [[NUM_ELEMENTS:%.*]] = load i32, ptr [[NUM_ELEMENTS_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[BACKEDGE:%.*]] ] ; CHECK-NEXT: [[BYTES_TO_WRITE:%.*]] = sub nuw nsw i32 [[CAPACITY]], [[IV]] ; CHECK-NEXT: [[CAPACITY_CHECK:%.*]] = icmp sle i32 [[BYTES_TO_WRITE]], 3 ; CHECK-NEXT: br i1 [[CAPACITY_CHECK]], label [[OUT_OF_BOUNDS:%.*]], label [[BACKEDGE]] ; CHECK: backedge: ; CHECK-NEXT: [[EL_PTR:%.*]] = getelementptr i32, ptr [[P:%.*]], i32 [[IV]] ; CHECK-NEXT: store i32 1, ptr [[EL_PTR]], align 4 ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i32 [[IV]], 4 ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp slt i32 [[IV_NEXT]], [[NUM_ELEMENTS]] ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_NEXT_LCSSA:%.*]] = phi i32 [ [[IV_NEXT]], [[BACKEDGE]] ] ; CHECK-NEXT: ret i32 [[IV_NEXT_LCSSA]] ; CHECK: out_of_bounds: ; CHECK-NEXT: ret i32 -1 ; entry: %capacity = load i32, ptr %capacity_p, !range !0 %num_elements = load i32, ptr %num_elements_p, !range !0 br label %loop loop: %iv = phi i32 [0, %entry], [%iv.next, %backedge] %bytes_to_write = sub i32 %capacity, %iv %capacity_check = icmp sle i32 %bytes_to_write, 3 br i1 %capacity_check, label %out_of_bounds, label %backedge backedge: %el.ptr = getelementptr i32, ptr %p, i32 %iv store i32 1, ptr %el.ptr %iv.next = add nuw nsw i32 %iv, 4 %loop_cond = icmp slt i32 %iv.next, %num_elements br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv.next out_of_bounds: ret i32 -1 } define i32 @test_05(ptr %p, ptr %capacity_p, ptr %num_elements_p) { ; CHECK-LABEL: @test_05( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[CAPACITY:%.*]] = load i32, ptr [[CAPACITY_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: [[NUM_ELEMENTS:%.*]] = load i32, ptr [[NUM_ELEMENTS_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[BACKEDGE:%.*]] ] ; CHECK-NEXT: [[BYTES_TO_WRITE:%.*]] = sub nuw nsw i32 [[CAPACITY]], [[IV]] ; CHECK-NEXT: [[CAPACITY_CHECK:%.*]] = icmp ult i32 [[BYTES_TO_WRITE]], 4 ; CHECK-NEXT: br i1 [[CAPACITY_CHECK]], label [[OUT_OF_BOUNDS:%.*]], label [[BACKEDGE]] ; CHECK: backedge: ; CHECK-NEXT: [[EL_PTR:%.*]] = getelementptr i32, ptr [[P:%.*]], i32 [[IV]] ; CHECK-NEXT: store i32 1, ptr [[EL_PTR]], align 4 ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i32 [[IV]], 4 ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp slt i32 [[IV_NEXT]], [[NUM_ELEMENTS]] ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_NEXT_LCSSA:%.*]] = phi i32 [ [[IV_NEXT]], [[BACKEDGE]] ] ; CHECK-NEXT: ret i32 [[IV_NEXT_LCSSA]] ; CHECK: out_of_bounds: ; CHECK-NEXT: ret i32 -1 ; entry: %capacity = load i32, ptr %capacity_p, !range !0 %num_elements = load i32, ptr %num_elements_p, !range !0 br label %loop loop: %iv = phi i32 [0, %entry], [%iv.next, %backedge] %bytes_to_write = sub i32 %capacity, %iv %capacity_check = icmp ult i32 %bytes_to_write, 4 br i1 %capacity_check, label %out_of_bounds, label %backedge backedge: %el.ptr = getelementptr i32, ptr %p, i32 %iv store i32 1, ptr %el.ptr %iv.next = add nuw nsw i32 %iv, 4 %loop_cond = icmp slt i32 %iv.next, %num_elements br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv.next out_of_bounds: ret i32 -1 } define i32 @test_06(ptr %p, ptr %capacity_p, ptr %num_elements_p) { ; CHECK-LABEL: @test_06( ; CHECK-NEXT: entry: ; CHECK-NEXT: [[CAPACITY:%.*]] = load i32, ptr [[CAPACITY_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: [[NUM_ELEMENTS:%.*]] = load i32, ptr [[NUM_ELEMENTS_P:%.*]], align 4, !range [[RNG0]] ; CHECK-NEXT: br label [[LOOP:%.*]] ; CHECK: loop: ; CHECK-NEXT: [[IV:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[IV_NEXT:%.*]], [[BACKEDGE:%.*]] ] ; CHECK-NEXT: [[BYTES_TO_WRITE:%.*]] = sub nuw nsw i32 [[CAPACITY]], [[IV]] ; CHECK-NEXT: [[CAPACITY_CHECK:%.*]] = icmp ule i32 [[BYTES_TO_WRITE]], 3 ; CHECK-NEXT: br i1 [[CAPACITY_CHECK]], label [[OUT_OF_BOUNDS:%.*]], label [[BACKEDGE]] ; CHECK: backedge: ; CHECK-NEXT: [[EL_PTR:%.*]] = getelementptr i32, ptr [[P:%.*]], i32 [[IV]] ; CHECK-NEXT: store i32 1, ptr [[EL_PTR]], align 4 ; CHECK-NEXT: [[IV_NEXT]] = add nuw nsw i32 [[IV]], 4 ; CHECK-NEXT: [[LOOP_COND:%.*]] = icmp slt i32 [[IV_NEXT]], [[NUM_ELEMENTS]] ; CHECK-NEXT: br i1 [[LOOP_COND]], label [[LOOP]], label [[EXIT:%.*]] ; CHECK: exit: ; CHECK-NEXT: [[IV_NEXT_LCSSA:%.*]] = phi i32 [ [[IV_NEXT]], [[BACKEDGE]] ] ; CHECK-NEXT: ret i32 [[IV_NEXT_LCSSA]] ; CHECK: out_of_bounds: ; CHECK-NEXT: ret i32 -1 ; entry: %capacity = load i32, ptr %capacity_p, !range !0 %num_elements = load i32, ptr %num_elements_p, !range !0 br label %loop loop: %iv = phi i32 [0, %entry], [%iv.next, %backedge] %bytes_to_write = sub i32 %capacity, %iv %capacity_check = icmp ule i32 %bytes_to_write, 3 br i1 %capacity_check, label %out_of_bounds, label %backedge backedge: %el.ptr = getelementptr i32, ptr %p, i32 %iv store i32 1, ptr %el.ptr %iv.next = add nuw nsw i32 %iv, 4 %loop_cond = icmp slt i32 %iv.next, %num_elements br i1 %loop_cond, label %loop, label %exit exit: ret i32 %iv.next out_of_bounds: ret i32 -1 } !0 = !{i32 1, i32 2147483648}