mirror of
https://github.com/RPCS3/llvm.git
synced 2025-01-07 20:40:28 +00:00
9f0a2a8bd5
This functionality is currently turned off by default. Part of the motivation for introducing scoped-noalias metadata is to enable the preservation of noalias parameter attribute information after inlining. Sometimes this can be inferred from the code in the caller after inlining, but often we simply lose valuable information. The overall process if fairly simple: 1. Create a new unqiue scope domain. 2. For each (used) noalias parameter, create a new alias scope. 3. For each pointer, collect the underlying objects. Add a noalias scope for each noalias parameter from which we're not derived (and has not been captured prior to that point). 4. Add an alias.scope for each noalias parameter from which we might be derived (or has been captured before that point). Note that the capture checks apply only if one of the underlying objects is not an identified function-local object. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@213949 91177308-0d34-0410-b5e6-96231b3b80d8
98 lines
4.5 KiB
LLVM
98 lines
4.5 KiB
LLVM
; RUN: opt -inline -enable-noalias-to-md-conversion -S < %s | FileCheck %s
|
|
target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-s0:64:64-f80:128:128-n8:16:32:64-S128"
|
|
target triple = "x86_64-unknown-linux-gnu"
|
|
|
|
define void @hello(float* noalias nocapture %a, float* noalias nocapture readonly %c) #0 {
|
|
entry:
|
|
%0 = load float* %c, align 4
|
|
%arrayidx = getelementptr inbounds float* %a, i64 5
|
|
store float %0, float* %arrayidx, align 4
|
|
ret void
|
|
}
|
|
|
|
define void @foo(float* noalias nocapture %a, float* noalias nocapture readonly %c) #0 {
|
|
entry:
|
|
tail call void @hello(float* %a, float* %c)
|
|
%0 = load float* %c, align 4
|
|
%arrayidx = getelementptr inbounds float* %a, i64 7
|
|
store float %0, float* %arrayidx, align 4
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @foo(float* noalias nocapture %a, float* noalias nocapture readonly %c) #0 {
|
|
; CHECK: entry:
|
|
; CHECK: %0 = load float* %c, align 4, !alias.scope !0, !noalias !3
|
|
; CHECK: %arrayidx.i = getelementptr inbounds float* %a, i64 5
|
|
; CHECK: store float %0, float* %arrayidx.i, align 4, !alias.scope !3, !noalias !0
|
|
; CHECK: %1 = load float* %c, align 4
|
|
; CHECK: %arrayidx = getelementptr inbounds float* %a, i64 7
|
|
; CHECK: store float %1, float* %arrayidx, align 4
|
|
; CHECK: ret void
|
|
; CHECK: }
|
|
|
|
define void @hello2(float* noalias nocapture %a, float* noalias nocapture %b, float* nocapture readonly %c) #0 {
|
|
entry:
|
|
%0 = load float* %c, align 4
|
|
%arrayidx = getelementptr inbounds float* %a, i64 6
|
|
store float %0, float* %arrayidx, align 4
|
|
%arrayidx1 = getelementptr inbounds float* %b, i64 8
|
|
store float %0, float* %arrayidx1, align 4
|
|
ret void
|
|
}
|
|
|
|
; Check that when hello() is inlined into foo(), and then foo() is inlined into
|
|
; foo2(), the noalias scopes are properly concatenated.
|
|
define void @foo2(float* nocapture %a, float* nocapture %b, float* nocapture readonly %c) #0 {
|
|
entry:
|
|
tail call void @foo(float* %a, float* %c)
|
|
tail call void @hello2(float* %a, float* %b, float* %c)
|
|
%0 = load float* %c, align 4
|
|
%arrayidx = getelementptr inbounds float* %a, i64 7
|
|
store float %0, float* %arrayidx, align 4
|
|
ret void
|
|
}
|
|
|
|
; CHECK: define void @foo2(float* nocapture %a, float* nocapture %b, float* nocapture readonly %c) #0 {
|
|
; CHECK: entry:
|
|
; CHECK: %0 = load float* %c, align 4, !alias.scope !5, !noalias !10
|
|
; CHECK: %arrayidx.i.i = getelementptr inbounds float* %a, i64 5
|
|
; CHECK: store float %0, float* %arrayidx.i.i, align 4, !alias.scope !10, !noalias !5
|
|
; CHECK: %1 = load float* %c, align 4, !alias.scope !13, !noalias !14
|
|
; CHECK: %arrayidx.i = getelementptr inbounds float* %a, i64 7
|
|
; CHECK: store float %1, float* %arrayidx.i, align 4, !alias.scope !14, !noalias !13
|
|
; CHECK: %2 = load float* %c, align 4, !noalias !15
|
|
; CHECK: %arrayidx.i1 = getelementptr inbounds float* %a, i64 6
|
|
; CHECK: store float %2, float* %arrayidx.i1, align 4, !alias.scope !19, !noalias !20
|
|
; CHECK: %arrayidx1.i = getelementptr inbounds float* %b, i64 8
|
|
; CHECK: store float %2, float* %arrayidx1.i, align 4, !alias.scope !20, !noalias !19
|
|
; CHECK: %3 = load float* %c, align 4
|
|
; CHECK: %arrayidx = getelementptr inbounds float* %a, i64 7
|
|
; CHECK: store float %3, float* %arrayidx, align 4
|
|
; CHECK: ret void
|
|
; CHECK: }
|
|
|
|
; CHECK: !0 = metadata !{metadata !1}
|
|
; CHECK: !1 = metadata !{metadata !1, metadata !2, metadata !"hello: %c"}
|
|
; CHECK: !2 = metadata !{metadata !2, metadata !"hello"}
|
|
; CHECK: !3 = metadata !{metadata !4}
|
|
; CHECK: !4 = metadata !{metadata !4, metadata !2, metadata !"hello: %a"}
|
|
; CHECK: !5 = metadata !{metadata !6, metadata !8}
|
|
; CHECK: !6 = metadata !{metadata !6, metadata !7, metadata !"hello: %c"}
|
|
; CHECK: !7 = metadata !{metadata !7, metadata !"hello"}
|
|
; CHECK: !8 = metadata !{metadata !8, metadata !9, metadata !"foo: %c"}
|
|
; CHECK: !9 = metadata !{metadata !9, metadata !"foo"}
|
|
; CHECK: !10 = metadata !{metadata !11, metadata !12}
|
|
; CHECK: !11 = metadata !{metadata !11, metadata !7, metadata !"hello: %a"}
|
|
; CHECK: !12 = metadata !{metadata !12, metadata !9, metadata !"foo: %a"}
|
|
; CHECK: !13 = metadata !{metadata !8}
|
|
; CHECK: !14 = metadata !{metadata !12}
|
|
; CHECK: !15 = metadata !{metadata !16, metadata !18}
|
|
; CHECK: !16 = metadata !{metadata !16, metadata !17, metadata !"hello2: %a"}
|
|
; CHECK: !17 = metadata !{metadata !17, metadata !"hello2"}
|
|
; CHECK: !18 = metadata !{metadata !18, metadata !17, metadata !"hello2: %b"}
|
|
; CHECK: !19 = metadata !{metadata !16}
|
|
; CHECK: !20 = metadata !{metadata !18}
|
|
|
|
attributes #0 = { nounwind uwtable }
|
|
|