2019-08-22 01:00:57 +08:00
|
|
|
; RUN: opt -S -basicaa -licm -licm-n2-threshold=0 -enable-mssa-loop-dependency=false %s | FileCheck %s
|
|
|
|
; RUN: opt -S -basicaa -licm -licm-n2-threshold=0 -enable-mssa-loop-dependency=true -verify-memoryssa %s | FileCheck %s --check-prefix=ALIAS-N2
|
2019-04-17 12:52:47 +08:00
|
|
|
; RUN: opt -licm -basicaa -licm-n2-threshold=200 < %s -S | FileCheck %s --check-prefix=ALIAS-N2
|
2019-08-22 01:00:57 +08:00
|
|
|
|
2019-04-17 12:52:47 +08:00
|
|
|
; RUN: opt -aa-pipeline=basic-aa -licm-n2-threshold=0 -passes='require<aa>,require<targetir>,require<scalar-evolution>,require<opt-remark-emit>,loop(licm)' < %s -S | FileCheck %s
|
2019-08-22 01:00:57 +08:00
|
|
|
; RUN: opt -aa-pipeline=basic-aa -licm-n2-threshold=0 -passes='require<aa>,require<targetir>,require<scalar-evolution>,require<opt-remark-emit>,loop-mssa(licm)' < %s -S | FileCheck %s --check-prefix=ALIAS-N2
|
2019-04-17 12:52:47 +08:00
|
|
|
; RUN: opt -aa-pipeline=basic-aa -licm-n2-threshold=200 -passes='require<aa>,require<targetir>,require<scalar-evolution>,require<opt-remark-emit>,loop(licm)' < %s -S | FileCheck %s --check-prefix=ALIAS-N2
|
|
|
|
|
|
|
|
declare i32 @foo() readonly argmemonly nounwind
|
|
|
|
declare i32 @foo2() readonly nounwind
|
|
|
|
declare i32 @bar(i32* %loc2) readonly argmemonly nounwind
|
|
|
|
|
|
|
|
define void @test(i32* %loc) {
|
|
|
|
; CHECK-LABEL: @test
|
|
|
|
; CHECK: @foo
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; ALIAS-N2-LABEL: @test
|
|
|
|
; ALIAS-N2: @foo
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%res = call i32 @foo()
|
|
|
|
store i32 %res, i32* %loc
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
; Negative test: show argmemonly is required
|
|
|
|
define void @test_neg(i32* %loc) {
|
|
|
|
; CHECK-LABEL: @test_neg
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; CHECK: @foo
|
|
|
|
; ALIAS-N2-LABEL: @test_neg
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
; ALIAS-N2: @foo
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%res = call i32 @foo2()
|
|
|
|
store i32 %res, i32* %loc
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
define void @test2(i32* noalias %loc, i32* noalias %loc2) {
|
|
|
|
; CHECK-LABEL: @test2
|
|
|
|
; CHECK: @bar
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; ALIAS-N2-LABEL: @test2
|
|
|
|
; ALIAS-N2: @bar
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%res = call i32 @bar(i32* %loc2)
|
|
|
|
store i32 %res, i32* %loc
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
; Negative test: %might clobber gep
|
|
|
|
define void @test3(i32* %loc) {
|
|
|
|
; CHECK-LABEL: @test3
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; CHECK: @bar
|
|
|
|
; ALIAS-N2-LABEL: @test3
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
; ALIAS-N2: @bar
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%res = call i32 @bar(i32* %loc)
|
|
|
|
%gep = getelementptr i32, i32 *%loc, i64 1000000
|
|
|
|
store i32 %res, i32* %gep
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
; Negative test: %loc might alias %loc2
|
|
|
|
define void @test4(i32* %loc, i32* %loc2) {
|
|
|
|
; CHECK-LABEL: @test4
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; CHECK: @bar
|
|
|
|
; ALIAS-N2-LABEL: @test4
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
; ALIAS-N2: @bar
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%res = call i32 @bar(i32* %loc2)
|
|
|
|
store i32 %res, i32* %loc
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
declare i32 @foo_new(i32*) readonly
|
|
|
|
; With the default AST mechanism used by LICM for alias analysis,
|
|
|
|
; we clump foo_new with bar.
|
|
|
|
; With the N2 Alias analysis diagnostic tool, we are able to hoist the
|
|
|
|
; argmemonly bar call out of the loop.
|
|
|
|
; Using MemorySSA we can also hoist bar.
|
|
|
|
|
|
|
|
define void @test5(i32* %loc2, i32* noalias %loc) {
|
|
|
|
; ALIAS-N2-LABEL: @test5
|
|
|
|
; ALIAS-N2: @bar
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
|
|
|
|
; CHECK-LABEL: @test5
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; CHECK: @bar
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%res1 = call i32 @bar(i32* %loc2)
|
|
|
|
%res = call i32 @foo_new(i32* %loc2)
|
|
|
|
store volatile i32 %res1, i32* %loc
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
; memcpy doesn't write to it's source argument, so loads to that location
|
|
|
|
; can still be hoisted
|
|
|
|
define void @test6(i32* noalias %loc, i32* noalias %loc2) {
|
|
|
|
; CHECK-LABEL: @test6
|
|
|
|
; CHECK: %val = load i32, i32* %loc2
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; CHECK: @llvm.memcpy
|
|
|
|
; ALIAS-N2-LABEL: @test6
|
|
|
|
; ALIAS-N2: %val = load i32, i32* %loc2
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
; ALIAS-N2: @llvm.memcpy
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%val = load i32, i32* %loc2
|
|
|
|
store i32 %val, i32* %loc
|
|
|
|
%dest = bitcast i32* %loc to i8*
|
|
|
|
%src = bitcast i32* %loc2 to i8*
|
|
|
|
call void @llvm.memcpy.p0i8.p0i8.i64(i8* %dest, i8* %src, i64 8, i1 false)
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
define void @test7(i32* noalias %loc, i32* noalias %loc2) {
|
|
|
|
; CHECK-LABEL: @test7
|
|
|
|
; CHECK: %val = load i32, i32* %loc2
|
|
|
|
; CHECK-LABEL: loop:
|
|
|
|
; CHECK: @custom_memcpy
|
|
|
|
; ALIAS-N2-LABEL: @test7
|
|
|
|
; ALIAS-N2: %val = load i32, i32* %loc2
|
|
|
|
; ALIAS-N2-LABEL: loop:
|
|
|
|
; ALIAS-N2: @custom_memcpy
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%val = load i32, i32* %loc2
|
|
|
|
store i32 %val, i32* %loc
|
|
|
|
%dest = bitcast i32* %loc to i8*
|
|
|
|
%src = bitcast i32* %loc2 to i8*
|
|
|
|
call void @custom_memcpy(i8* %dest, i8* %src)
|
|
|
|
br label %loop
|
|
|
|
}
|
|
|
|
|
|
|
|
declare void @llvm.memcpy.p0i8.p0i8.i64(i8* nocapture writeonly, i8* nocapture readonly, i64, i1)
|
|
|
|
declare void @custom_memcpy(i8* nocapture writeonly, i8* nocapture readonly) argmemonly nounwind
|