-
-
Notifications
You must be signed in to change notification settings - Fork 5.6k
/
Copy pathsetup.jl
72 lines (65 loc) · 2.69 KB
/
setup.jl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
include(normpath(@__DIR__, "EAUtils.jl"))
using Test, Core.Compiler.EscapeAnalysis, .EAUtils
import Core: Argument, SSAValue, ReturnNode
const EA = Core.Compiler.EscapeAnalysis
import .EA: ignore_argescape
isT(T) = (@nospecialize x) -> x === T
isreturn(@nospecialize x) = isa(x, Core.ReturnNode) && isdefined(x, :val)
isthrow(@nospecialize x) = Meta.isexpr(x, :call) && Core.Compiler.is_throw_call(x)
isnew(@nospecialize x) = Meta.isexpr(x, :new)
isϕ(@nospecialize x) = isa(x, Core.PhiNode)
function with_normalized_name(@nospecialize(f), @nospecialize(x))
if Meta.isexpr(x, :foreigncall)
name = x.args[1]
nn = EA.normalize(name)
return isa(nn, Symbol) && f(nn)
end
return false
end
isarrayalloc(@nospecialize x) = with_normalized_name(nn->!isnothing(Core.Compiler.alloc_array_ndims(nn)), x)
isarrayresize(@nospecialize x) = with_normalized_name(nn->!isnothing(EA.array_resize_info(nn)), x)
isarraycopy(@nospecialize x) = with_normalized_name(nn->EA.is_array_copy(nn), x)
import Core.Compiler: argextype, singleton_type
iscall(y) = @nospecialize(x) -> iscall(y, x)
function iscall((ir, f), @nospecialize(x))
return iscall(x) do @nospecialize x
singleton_type(Core.Compiler.argextype(x, ir, Any[])) === f
end
end
iscall(pred::Function, @nospecialize(x)) = Meta.isexpr(x, :call) && pred(x.args[1])
# check if `x` is a statically-resolved call of a function whose name is `sym`
isinvoke(y) = @nospecialize(x) -> isinvoke(y, x)
isinvoke(sym::Symbol, @nospecialize(x)) = isinvoke(mi->mi.def.name===sym, x)
isinvoke(pred::Function, @nospecialize(x)) = Meta.isexpr(x, :invoke) && pred(x.args[1]::Core.MethodInstance)
"""
is_load_forwardable(x::EscapeInfo) -> Bool
Queries if `x` is elibigle for store-to-load forwarding optimization.
"""
function is_load_forwardable(x::EA.EscapeInfo)
AliasInfo = x.AliasInfo
# NOTE technically we also need to check `!has_thrown_escape(x)` here as well,
# but we can also do equivalent check during forwarding
return isa(AliasInfo, EA.IndexableFields) || isa(AliasInfo, EA.IndexableElements)
end
let setup_ex = quote
mutable struct SafeRef{T}
x::T
end
Base.getindex(s::SafeRef) = getfield(s, 1)
Base.setindex!(s::SafeRef, x) = setfield!(s, 1, x)
mutable struct SafeRefs{S,T}
x1::S
x2::T
end
Base.getindex(s::SafeRefs, idx::Int) = getfield(s, idx)
Base.setindex!(s::SafeRefs, x, idx::Int) = setfield!(s, idx, x)
global GV::Any
const global GR = Ref{Any}()
end
global function EATModule(setup_ex = setup_ex)
M = Module()
Core.eval(M, setup_ex)
return M
end
Core.eval(@__MODULE__, setup_ex)
end