| ; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --function-signature --check-attributes |
| ; RUN: opt -aa-pipeline=basic-aa,globals-aa -passes='require<globals-aa>,gvn' -S < %s | FileCheck %s |
| |
| ; Make sure we do not hoist the load before the intrinsic, unknown function, or |
| ; optnone function except if we know the unknown function is nosync and nocallback. |
| |
| @G1 = internal global i32 undef |
| @G2 = internal global i32 undef |
| @G3 = internal global i32 undef |
| @G4 = internal global i32 undef |
| |
| define void @test_barrier(i1 %c) { |
| ; CHECK-LABEL: define {{[^@]+}}@test_barrier |
| ; CHECK-SAME: (i1 [[C:%.*]]) { |
| ; CHECK-NEXT: br i1 [[C]], label [[INIT:%.*]], label [[CHECK:%.*]] |
| ; CHECK: init: |
| ; CHECK-NEXT: store i32 0, ptr @G1, align 4 |
| ; CHECK-NEXT: br label [[CHECK]] |
| ; CHECK: check: |
| ; CHECK-NEXT: call void @llvm.amdgcn.s.barrier() |
| ; CHECK-NEXT: [[V:%.*]] = load i32, ptr @G1, align 4 |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[V]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| br i1 %c, label %init, label %check |
| init: |
| store i32 0, ptr @G1 |
| br label %check |
| check: |
| call void @llvm.amdgcn.s.barrier() |
| %v = load i32, ptr @G1 |
| %cmp = icmp eq i32 %v, 0 |
| call void @llvm.assume(i1 %cmp) |
| ret void |
| } |
| |
| define void @test_unknown(i1 %c) { |
| ; CHECK-LABEL: define {{[^@]+}}@test_unknown |
| ; CHECK-SAME: (i1 [[C:%.*]]) { |
| ; CHECK-NEXT: br i1 [[C]], label [[INIT:%.*]], label [[CHECK:%.*]] |
| ; CHECK: init: |
| ; CHECK-NEXT: store i32 0, ptr @G2, align 4 |
| ; CHECK-NEXT: br label [[CHECK]] |
| ; CHECK: check: |
| ; CHECK-NEXT: call void @unknown() |
| ; CHECK-NEXT: [[V:%.*]] = load i32, ptr @G2, align 4 |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[V]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| br i1 %c, label %init, label %check |
| init: |
| store i32 0, ptr @G2 |
| br label %check |
| check: |
| call void @unknown() |
| %v = load i32, ptr @G2 |
| %cmp = icmp eq i32 %v, 0 |
| call void @llvm.assume(i1 %cmp) |
| ret void |
| } |
| |
| define void @test_optnone(i1 %c) { |
| ; CHECK-LABEL: define {{[^@]+}}@test_optnone |
| ; CHECK-SAME: (i1 [[C:%.*]]) { |
| ; CHECK-NEXT: br i1 [[C]], label [[INIT:%.*]], label [[CHECK:%.*]] |
| ; CHECK: init: |
| ; CHECK-NEXT: store i32 0, ptr @G3, align 4 |
| ; CHECK-NEXT: br label [[CHECK]] |
| ; CHECK: check: |
| ; CHECK-NEXT: call void @optnone() |
| ; CHECK-NEXT: [[V:%.*]] = load i32, ptr @G3, align 4 |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[V]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| br i1 %c, label %init, label %check |
| init: |
| store i32 0, ptr @G3 |
| br label %check |
| check: |
| call void @optnone() |
| %v = load i32, ptr @G3 |
| %cmp = icmp eq i32 %v, 0 |
| call void @llvm.assume(i1 %cmp) |
| ret void |
| } |
| |
| define void @optnone() optnone nosync nocallback noinline { |
| ; CHECK: Function Attrs: nocallback noinline nosync optnone |
| ; CHECK-LABEL: define {{[^@]+}}@optnone |
| ; CHECK-SAME: () #[[ATTR0:[0-9]+]] { |
| ; CHECK-NEXT: ret void |
| ; |
| ret void |
| } |
| |
| ; Here hoisting is legal and we use it to verify it will happen. |
| define void @test_unknown_annotated(i1 %c) { |
| ; CHECK-LABEL: define {{[^@]+}}@test_unknown_annotated |
| ; CHECK-SAME: (i1 [[C:%.*]]) { |
| ; CHECK-NEXT: br i1 [[C]], label [[INIT:%.*]], label [[DOTCHECK_CRIT_EDGE:%.*]] |
| ; CHECK: .check_crit_edge: |
| ; CHECK-NEXT: [[V_PRE:%.*]] = load i32, ptr @G4, align 4 |
| ; CHECK-NEXT: br label [[CHECK:%.*]] |
| ; CHECK: init: |
| ; CHECK-NEXT: store i32 0, ptr @G4, align 4 |
| ; CHECK-NEXT: br label [[CHECK]] |
| ; CHECK: check: |
| ; CHECK-NEXT: [[V:%.*]] = phi i32 [ [[V_PRE]], [[DOTCHECK_CRIT_EDGE]] ], [ 0, [[INIT]] ] |
| ; CHECK-NEXT: call void @unknown_nosync_nocallback() |
| ; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[V]], 0 |
| ; CHECK-NEXT: call void @llvm.assume(i1 [[CMP]]) |
| ; CHECK-NEXT: ret void |
| ; |
| br i1 %c, label %init, label %check |
| init: |
| store i32 0, ptr @G4 |
| br label %check |
| check: |
| call void @unknown_nosync_nocallback() |
| %v = load i32, ptr @G4 |
| %cmp = icmp eq i32 %v, 0 |
| call void @llvm.assume(i1 %cmp) |
| ret void |
| } |
| |
| declare void @unknown() |
| declare void @unknown_nosync_nocallback() nosync nocallback |
| declare void @llvm.amdgcn.s.barrier() |
| declare void @llvm.assume(i1 noundef) |
| |