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