; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 3 ; RUN: opt -p constraint-elimination -S %s | FileCheck %s declare void @use(i1) declare void @llvm.assume(i1) define void @add_rec_decreasing_cond_true_constant(i8 noundef %len) { ; CHECK-LABEL: define void @add_rec_decreasing_cond_true_constant( ; CHECK-SAME: i8 noundef [[LEN:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ 4, [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], 0 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: call void @use(i1 true) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -1 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: br label %loop.header loop.header: %k.0 = phi i8 [ 4, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, 0 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, 5 call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -1 br label %loop.header exit: ret void } define void @add_rec_decreasing_cond_not_true_constant(i8 noundef %len) { ; CHECK-LABEL: define void @add_rec_decreasing_cond_not_true_constant( ; CHECK-SAME: i8 noundef [[LEN:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ 4, [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], 0 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: [[CMP_NOT_I:%.*]] = icmp ult i8 [[K_0]], 4 ; CHECK-NEXT: call void @use(i1 [[CMP_NOT_I]]) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -1 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: br label %loop.header loop.header: %k.0 = phi i8 [ 4, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, 0 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, 4 call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -1 br label %loop.header exit: ret void } define void @add_rec_decreasing_cond_true_start_signed_positive(i8 noundef %start) { ; CHECK-LABEL: define void @add_rec_decreasing_cond_true_start_signed_positive( ; CHECK-SAME: i8 noundef [[START:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: [[PRECOND:%.*]] = icmp sge i8 [[START]], 1 ; CHECK-NEXT: call void @llvm.assume(i1 [[PRECOND]]) ; CHECK-NEXT: [[START_1:%.*]] = add i8 [[START]], -1 ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ [[START_1]], [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], 0 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: call void @use(i1 true) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -1 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: %precond = icmp sge i8 %start, 1 call void @llvm.assume(i1 %precond) %start.1 = add i8 %start, -1 br label %loop.header loop.header: %k.0 = phi i8 [ %start.1, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, 0 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, %start call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -1 br label %loop.header exit: ret void } define void @add_rec_decreasing_cond_not_true_start_signed_positive(i8 noundef %start) { ; CHECK-LABEL: define void @add_rec_decreasing_cond_not_true_start_signed_positive( ; CHECK-SAME: i8 noundef [[START:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: [[PRECOND:%.*]] = icmp sge i8 [[START]], 1 ; CHECK-NEXT: call void @llvm.assume(i1 [[PRECOND]]) ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ [[START]], [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], 0 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: [[CMP_NOT_I:%.*]] = icmp ult i8 [[K_0]], [[START]] ; CHECK-NEXT: call void @use(i1 [[CMP_NOT_I]]) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -1 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: %precond = icmp sge i8 %start, 1 call void @llvm.assume(i1 %precond) br label %loop.header loop.header: %k.0 = phi i8 [ %start, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, 0 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, %start call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -1 br label %loop.header exit: ret void } define void @add_rec_decreasing_add_rec_positive_to_negative(i8 noundef %len) { ; CHECK-LABEL: define void @add_rec_decreasing_add_rec_positive_to_negative( ; CHECK-SAME: i8 noundef [[LEN:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ 4, [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], -2 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: [[CMP_NOT_I:%.*]] = icmp ult i8 [[K_0]], 5 ; CHECK-NEXT: call void @use(i1 [[CMP_NOT_I]]) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -1 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: br label %loop.header loop.header: %k.0 = phi i8 [ 4, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, -2 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, 5 call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -1 br label %loop.header exit: ret void } define void @add_rec_decreasing_2_cond_true_constant(i8 noundef %len) { ; CHECK-LABEL: define void @add_rec_decreasing_2_cond_true_constant( ; CHECK-SAME: i8 noundef [[LEN:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ 4, [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], 0 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: [[CMP_NOT_I:%.*]] = icmp ult i8 [[K_0]], 5 ; CHECK-NEXT: call void @use(i1 [[CMP_NOT_I]]) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -2 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: br label %loop.header loop.header: %k.0 = phi i8 [ 4, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, 0 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, 5 call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -2 br label %loop.header exit: ret void } define void @add_rec_decreasing_2_cond_not_true_constant(i8 noundef %len) { ; CHECK-LABEL: define void @add_rec_decreasing_2_cond_not_true_constant( ; CHECK-SAME: i8 noundef [[LEN:%.*]]) { ; CHECK-NEXT: entry: ; CHECK-NEXT: br label [[LOOP_HEADER:%.*]] ; CHECK: loop.header: ; CHECK-NEXT: [[K_0:%.*]] = phi i8 [ 5, [[ENTRY:%.*]] ], [ [[K_DEC:%.*]], [[LOOP_LATCH:%.*]] ] ; CHECK-NEXT: [[CMP2_NOT:%.*]] = icmp eq i8 [[K_0]], 0 ; CHECK-NEXT: br i1 [[CMP2_NOT]], label [[EXIT:%.*]], label [[LOOP_LATCH]] ; CHECK: loop.latch: ; CHECK-NEXT: [[CMP_NOT_I:%.*]] = icmp ult i8 [[K_0]], 5 ; CHECK-NEXT: call void @use(i1 [[CMP_NOT_I]]) ; CHECK-NEXT: [[K_DEC]] = add i8 [[K_0]], -2 ; CHECK-NEXT: br label [[LOOP_HEADER]] ; CHECK: exit: ; CHECK-NEXT: ret void ; entry: br label %loop.header loop.header: %k.0 = phi i8 [ 5, %entry], [ %k.dec, %loop.latch ] %cmp2.not = icmp eq i8 %k.0, 0 br i1 %cmp2.not, label %exit, label %loop.latch loop.latch: %cmp.not.i = icmp ult i8 %k.0, 5 call void @use(i1 %cmp.not.i) %k.dec = add i8 %k.0, -2 br label %loop.header exit: ret void }