Skip to content

Commit a72c744

Browse files
timholyKristofferC
andauthoredMar 8, 2021
Reduce latency (#101)
The following benchmark: ```julia julia> using Example julia> id = Base.module_keys[Example] Example [7876af07-990d-54b4-ab0e-23690620f79a] julia> @time begin @eval module SomeMod using Requires __init__() = @require Example="7876af07-990d-54b4-ab0e-23690620f79a" f(x) = x^2 end end 0.335103 seconds (512.40 k allocations: 30.980 MiB, 8.35% gc time) Main.SomeMod ``` has substantial latency. This PR cuts it to about 60ms, more than a five-fold improvment. It also makes the precompiles less fragile. Co-authored-by: Kristoffer Carlsson <kcarlsson89@gmail.com>
1 parent 8b1543e commit a72c744

File tree

2 files changed

+47
-32
lines changed

2 files changed

+47
-32
lines changed
 

‎src/Requires.jl

+20-10
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,12 @@
11
module Requires
22

3+
if isdefined(Base, :Experimental) && isdefined(Base.Experimental, Symbol("@compiler_options"))
4+
@eval Base.Experimental.@compiler_options compile=min optimize=0 infer=false
5+
end
6+
37
using UUIDs
48

5-
function _include_path(relpath)
9+
function _include_path(relpath::String)
610
# Reproduces include()'s runtime relative path logic
711
# See Base._include_dependency()
812
prev = Base.source_path(nothing)
@@ -23,7 +27,7 @@ string literal, not an expression.
2327
2428
`@require` blocks insert this automatically when you use `include`.
2529
"""
26-
macro include(relpath)
30+
macro include(relpath::String)
2731
compiletime_path = joinpath(dirname(String(__source__.file)), relpath)
2832
s = String(read(compiletime_path))
2933
quote
@@ -48,14 +52,20 @@ function __init__()
4852
end
4953

5054
if isprecompiling()
51-
@assert precompile(loadpkg, (Base.PkgId,))
52-
@assert precompile(withpath, (Any, String))
53-
@assert precompile(err, (Any, Module, String))
54-
@assert precompile(parsepkg, (Expr,))
55-
@assert precompile(listenpkg, (Any, Base.PkgId))
56-
@assert precompile(callbacks, (Base.PkgId,))
57-
@assert precompile(withnotifications, (Vararg{Any},))
58-
@assert precompile(withnotifications, (Any, Vararg{Any},))
55+
precompile(loadpkg, (Base.PkgId,)) || @warn "Requires failed to precompile `loadpkg`"
56+
precompile(withpath, (Any, String)) || @warn "Requires failed to precompile `withpath`"
57+
precompile(err, (Any, Module, String)) || @warn "Requires failed to precompile `err`"
58+
precompile(parsepkg, (Expr,)) || @warn "Requires failed to precompile `parsepkg`"
59+
precompile(listenpkg, (Any, Base.PkgId)) || @warn "Requires failed to precompile `listenpkg`"
60+
precompile(callbacks, (Base.PkgId,)) || @warn "Requires failed to precompile `callbacks`"
61+
precompile(withnotifications, (String, Module, String, String, Expr)) || @warn "Requires failed to precompile `withnotifications`"
62+
precompile(replace_include, (Expr, LineNumberNode)) || @warn "Requires failed to precompile `replace_include`"
63+
precompile(getfield(Requires, Symbol("@require")), (LineNumberNode, Module, Expr, Any)) || @warn "Requires failed to precompile `@require`"
64+
65+
precompile(_include_path, (String,)) || @warn "Requires failed to precompile `_include_path`"
66+
precompile(getfield(Requires, Symbol("@include")), (LineNumberNode, Module, String)) || @warn "Requires failed to precompile `@include`"
67+
68+
precompile(__init__, ()) || @warn "Requires failed to precompile `__init__`"
5969
end
6070

6171
end # module

‎src/require.jl

+27-22
Original file line numberDiff line numberDiff line change
@@ -1,29 +1,34 @@
11
using Base: PkgId, loaded_modules, package_callbacks
22
using Base.Meta: isexpr
3+
if isdefined(Base, :mapany)
4+
const mapany = Base.mapany
5+
else
6+
mapany(f, A::AbstractVector) = map!(f, Vector{Any}(undef, length(A)), A)
7+
end
38

49
export @require
510

611
isprecompiling() = ccall(:jl_generating_output, Cint, ()) == 1
712

8-
loaded(pkg) = haskey(Base.loaded_modules, pkg)
13+
loaded(pkg::PkgId) = haskey(Base.loaded_modules, pkg)
914

1015
const notified_pkgs = [Base.PkgId(UUID(0x295af30fe4ad537b898300126c2a3abe), "Revise")]
1116

1217
const _callbacks = Dict{PkgId, Vector{Function}}()
13-
callbacks(pkg) = get!(Vector{Function}, _callbacks, pkg)
18+
callbacks(pkg::PkgId) = get!(Vector{Function}, _callbacks, pkg)
1419

15-
listenpkg(@nospecialize(f), pkg) =
20+
listenpkg(@nospecialize(f), pkg::PkgId) =
1621
loaded(pkg) ? f() : push!(callbacks(pkg), f)
1722

18-
function loadpkg(pkg::Base.PkgId)
23+
function loadpkg(pkg::PkgId)
1924
if haskey(_callbacks, pkg)
2025
fs = _callbacks[pkg]
2126
delete!(_callbacks, pkg)
2227
foreach(Base.invokelatest, fs)
2328
end
2429
end
2530

26-
function withpath(@nospecialize(f), path)
31+
function withpath(@nospecialize(f), path::String)
2732
tls = task_local_storage()
2833
hassource = haskey(tls, :SOURCE_PATH)
2934
hassource && (path′ = tls[:SOURCE_PATH])
@@ -37,19 +42,19 @@ function withpath(@nospecialize(f), path)
3742
end
3843
end
3944

40-
function err(@nospecialize(f), listener, mod)
45+
function err(@nospecialize(f), listener::Module, modname::String)
4146
try
4247
f()
4348
catch exc
44-
@warn "Error requiring `$mod` from `$listener`" exception=(exc,catch_backtrace())
49+
@warn "Error requiring `$modname` from `$listener`" exception=(exc,catch_backtrace())
4550
end
4651
end
4752

48-
function parsepkg(ex)
53+
function parsepkg(ex::Expr)
4954
isexpr(ex, :(=)) || @goto fail
5055
mod, id = ex.args
5156
(mod isa Symbol && id isa String) || @goto fail
52-
return id, String(mod)
57+
return id::String, String(mod::Symbol)
5358
@label fail
5459
error("Requires syntax is: `@require Pkg=\"uuid\"`")
5560
end
@@ -67,29 +72,29 @@ function withnotifications(@nospecialize(args...))
6772
return nothing
6873
end
6974

70-
function replace_include(ex, source)
71-
if isexpr(ex, :call) && ex.args[1] == :include && ex.args[2] isa String
72-
return Expr(:macrocall, :($Requires.$(Symbol("@include"))), source, ex.args[2])
73-
elseif ex isa Expr
74-
Expr(ex.head, replace_include.(ex.args, (source,))...)
75-
else
76-
return ex
75+
function replace_include(ex::Expr, source::LineNumberNode)
76+
if ex.head == :call && ex.args[1] === :include && ex.args[2] isa String
77+
return Expr(:macrocall, :($Requires.$(Symbol("@include"))), source, ex.args[2]::String)
7778
end
79+
return Expr(ex.head, (mapany(ex.args) do arg
80+
isa(arg, Expr) ? replace_include(arg, source) : arg
81+
end)...)
7882
end
7983

80-
macro require(pkg, expr)
84+
macro require(pkg::Union{Symbol,Expr}, expr)
8185
pkg isa Symbol &&
8286
return Expr(:macrocall, Symbol("@warn"), __source__,
8387
"Requires now needs a UUID; please see the readme for changes in 0.7.")
84-
id, modname = parsepkg(pkg)
85-
pkg = :(Base.PkgId(Base.UUID($id), $modname))
86-
expr = replace_include(expr, __source__)
88+
idstr, modname = parsepkg(pkg)
89+
pkg = :(Base.PkgId(Base.UUID($idstr), $modname))
90+
expr = isa(expr, Expr) ? replace_include(expr, __source__) : expr
8791
expr = macroexpand(__module__, expr)
92+
srcfile = string(__source__.file)
8893
quote
8994
if !isprecompiling()
9095
listenpkg($pkg) do
91-
$withnotifications($(string(__source__.file)), $__module__, $id, $modname, $(esc(Expr(:quote, expr))))
92-
withpath($(string(__source__.file))) do
96+
$withnotifications($srcfile, $__module__, $idstr, $modname, $(esc(Expr(:quote, expr))))
97+
withpath($srcfile) do
9398
err($__module__, $modname) do
9499
$(esc(:(eval($(Expr(:quote, Expr(:block,
95100
:(const $(Symbol(modname)) = Base.require($pkg)),

0 commit comments

Comments
 (0)
Please sign in to comment.