| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py |
| ; RUN: opt -passes=simplifycfg -simplifycfg-require-and-preserve-domtree=1 -S < %s | FileCheck %s |
| |
| define void @assume_false_to_unreachable1() { |
| ; CHECK-LABEL: @assume_false_to_unreachable1( |
| ; CHECK-NEXT: unreachable |
| ; |
| call void @llvm.assume(i1 0) |
| ret void |
| |
| } |
| |
| define void @assume_undef_to_unreachable() { |
| ; CHECK-LABEL: @assume_undef_to_unreachable( |
| ; CHECK-NEXT: unreachable |
| ; |
| call void @llvm.assume(i1 undef) |
| ret void |
| |
| } |
| |
| define i32 @speculate_block_with_assume_basic(i1 %c, i32 %x) { |
| ; CHECK-LABEL: @speculate_block_with_assume_basic( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 1, i32 0 |
| ; CHECK-NEXT: ret i32 [[SPEC_SELECT]] |
| ; |
| entry: |
| br i1 %c, label %if, label %join |
| |
| if: |
| %cmp = icmp ne i32 %x, 0 |
| call void @llvm.assume(i1 %cmp) |
| br label %join |
| |
| join: |
| %phi = phi i32 [ 0, %entry ], [ 1, %if ] |
| ret i32 %phi |
| } |
| |
| define i32 @speculate_block_with_assume_extra_instr(i1 %c, i32 %x) { |
| ; CHECK-LABEL: @speculate_block_with_assume_extra_instr( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[ADD:%.*]] = add i32 [[X:%.*]], 1 |
| ; CHECK-NEXT: [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 [[ADD]], i32 0 |
| ; CHECK-NEXT: ret i32 [[SPEC_SELECT]] |
| ; |
| entry: |
| br i1 %c, label %if, label %join |
| |
| if: |
| %add = add i32 %x, 1 |
| %cmp = icmp ne i32 %add, 0 |
| call void @llvm.assume(i1 %cmp) |
| br label %join |
| |
| join: |
| %phi = phi i32 [ 0, %entry ], [ %add, %if ] |
| ret i32 %phi |
| } |
| |
| ; We only allow speculating one instruction. Here %add and %add2 are used by |
| ; the assume, but not ephemeral, because they are also used by %phi. |
| define i32 @speculate_block_with_assume_extra_instrs_too_many(i1 %c, i32 %x) { |
| ; CHECK-LABEL: @speculate_block_with_assume_extra_instrs_too_many( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: br i1 [[C:%.*]], label [[IF:%.*]], label [[JOIN:%.*]] |
| ; CHECK: if: |
| ; CHECK-NEXT: [[ADD:%.*]] = add i32 [[X:%.*]], 1 |
| ; CHECK-NEXT: [[ADD2:%.*]] = add i32 [[ADD]], 1 |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp ne i32 [[ADD2]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: br label [[JOIN]] |
| ; CHECK: join: |
| ; CHECK-NEXT: [[PHI:%.*]] = phi i32 [ 0, [[ENTRY:%.*]] ], [ [[ADD2]], [[IF]] ] |
| ; CHECK-NEXT: ret i32 [[PHI]] |
| ; |
| entry: |
| br i1 %c, label %if, label %join |
| |
| if: |
| %add = add i32 %x, 1 |
| %add2 = add i32 %add, 1 |
| %cmp = icmp ne i32 %add2, 0 |
| call void @llvm.assume(i1 %cmp) |
| br label %join |
| |
| join: |
| %phi = phi i32 [ 0, %entry ], [ %add2, %if ] |
| ret i32 %phi |
| } |
| |
| define i32 @speculate_block_with_assume_extra_instrs_okay(i1 %c, i32 %x) { |
| ; CHECK-LABEL: @speculate_block_with_assume_extra_instrs_okay( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[ADD:%.*]] = add i32 [[X:%.*]], 1 |
| ; CHECK-NEXT: [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 [[ADD]], i32 0 |
| ; CHECK-NEXT: ret i32 [[SPEC_SELECT]] |
| ; |
| entry: |
| br i1 %c, label %if, label %join |
| |
| if: |
| %add = add i32 %x, 1 |
| %add2 = add i32 %add, 1 |
| %cmp = icmp ne i32 %add2, 0 |
| call void @llvm.assume(i1 %cmp) |
| br label %join |
| |
| join: |
| %phi = phi i32 [ 0, %entry ], [ %add, %if ] |
| ret i32 %phi |
| } |
| |
| define i32 @speculate_block_with_assume_operand_bundle(i1 %c, ptr %p) { |
| ; CHECK-LABEL: @speculate_block_with_assume_operand_bundle( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: [[SPEC_SELECT:%.*]] = select i1 [[C:%.*]], i32 1, i32 0 |
| ; CHECK-NEXT: ret i32 [[SPEC_SELECT]] |
| ; |
| entry: |
| br i1 %c, label %if, label %join |
| |
| if: |
| call void @llvm.assume(i1 true) ["nonnull"(ptr %p)] |
| br label %join |
| |
| join: |
| %phi = phi i32 [ 0, %entry ], [ 1, %if ] |
| ret i32 %phi |
| } |
| |
| define void @empty_block_with_assume(i1 %c, i32 %x) { |
| ; CHECK-LABEL: @empty_block_with_assume( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: br i1 [[C:%.*]], label [[IF:%.*]], label [[ELSE:%.*]] |
| ; CHECK: if: |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp ne i32 [[X:%.*]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: br label [[JOIN:%.*]] |
| ; CHECK: else: |
| ; CHECK-NEXT: call void @dummy() |
| ; CHECK-NEXT: br label [[JOIN]] |
| ; CHECK: join: |
| ; CHECK-NEXT: ret void |
| ; |
| entry: |
| br i1 %c, label %if, label %else |
| |
| if: |
| %cmp = icmp ne i32 %x, 0 |
| call void @llvm.assume(i1 %cmp) |
| br label %join |
| |
| else: |
| call void @dummy() |
| br label %join |
| |
| join: |
| ret void |
| } |
| |
| define void @not_empty_block_with_assume(i1 %c) { |
| ; CHECK-LABEL: @not_empty_block_with_assume( |
| ; CHECK-NEXT: entry: |
| ; CHECK-NEXT: br i1 [[C:%.*]], label [[IF:%.*]], label [[ELSE:%.*]] |
| ; CHECK: if: |
| ; CHECK-NEXT: [[X:%.*]] = call i32 @may_have_side_effect() |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp ne i32 [[X]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: br label [[JOIN:%.*]] |
| ; CHECK: else: |
| ; CHECK-NEXT: call void @dummy() |
| ; CHECK-NEXT: br label [[JOIN]] |
| ; CHECK: join: |
| ; CHECK-NEXT: ret void |
| ; |
| entry: |
| br i1 %c, label %if, label %else |
| |
| if: |
| %x = call i32 @may_have_side_effect() |
| %cmp = icmp ne i32 %x, 0 |
| call void @llvm.assume(i1 %cmp) |
| br label %join |
| |
| else: |
| call void @dummy() |
| br label %join |
| |
| join: |
| ret void |
| } |
| |
| declare void @dummy() |
| declare i32 @may_have_side_effect() |
| declare void @llvm.assume(i1) nounwind |
| |