| ; 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 |
| } |