From ad370215297c3d78efd9b1135f7731b115c2ccc7 Mon Sep 17 00:00:00 2001 From: Jameson Nash Date: Mon, 18 Apr 2016 21:00:08 -0400 Subject: [PATCH] do a better job at generating code for known functions closes #15048 --- base/loading.jl | 28 +++++++++----- src/gf.c | 98 ++++++++++++++++++++++++++++++++++++++++++------- src/init.c | 6 +-- 3 files changed, 105 insertions(+), 27 deletions(-) diff --git a/base/loading.jl b/base/loading.jl index 8b522437c03cc..25a6ae5183e89 100644 --- a/base/loading.jl +++ b/base/loading.jl @@ -80,9 +80,19 @@ else end end +function try_path(prefix::ByteString, base::ByteString, name::ByteString) + path = joinpath(prefix, name) + isfile_casesensitive(path) && return abspath(path) + path = joinpath(prefix, base, "src", name) + isfile_casesensitive(path) && return abspath(path) + path = joinpath(prefix, name, "src", name) + isfile_casesensitive(path) && return abspath(path) + return nothing +end + # `wd` is a working directory to search. defaults to current working directory. # if `wd === nothing`, no extra path is searched. -function find_in_path(name::AbstractString, wd = pwd()) +function find_in_path(name::ByteString, wd = pwd()) isabspath(name) && return name base = name if endswith(name,".jl") @@ -93,18 +103,16 @@ function find_in_path(name::AbstractString, wd = pwd()) if wd !== nothing isfile_casesensitive(joinpath(wd,name)) && return joinpath(wd,name) end - for prefix in [Pkg.dir(); LOAD_PATH] - path = joinpath(prefix, name) - isfile_casesensitive(path) && return abspath(path) - path = joinpath(prefix, base, "src", name) - isfile_casesensitive(path) && return abspath(path) - path = joinpath(prefix, name, "src", name) - isfile_casesensitive(path) && return abspath(path) + p = try_path(Pkg.dir(), base, name) + p !== nothing && return p + for prefix in LOAD_PATH + p = try_path(prefix, base, name) + p !== nothing && return p end return nothing end -function find_in_node_path(name, srcpath, node::Int=1) +function find_in_node_path(name::ByteString, srcpath, node::Int=1) if myid() == node find_in_path(name, srcpath) else @@ -112,7 +120,7 @@ function find_in_node_path(name, srcpath, node::Int=1) end end -function find_source_file(file) +function find_source_file(file::ByteString) (isabspath(file) || isfile(file)) && return file file2 = find_in_path(file) file2 !== nothing && return file2 diff --git a/src/gf.c b/src/gf.c index 16c137e9c208d..6fe7d6597bea4 100644 --- a/src/gf.c +++ b/src/gf.c @@ -1888,6 +1888,11 @@ jl_lambda_info_t *jl_get_specialization1(jl_tupletype_t *types) return NULL; } +JL_DLLEXPORT void jl_compile_hint(jl_tupletype_t *types) +{ + (void)jl_get_specialization1(types); +} + // add type of `f` to front of argument tuple type jl_tupletype_t *jl_argtype_with_function(jl_function_t *f, jl_tupletype_t *types) { @@ -2158,13 +2163,6 @@ static int _compile_all_enq(jl_typemap_entry_t *ml, void *env) return 1; } -static void _compile_all_enq_mt(jl_methtable_t *mt, jl_array_t *found) -{ - if (mt == NULL || (jl_value_t*)mt == jl_nothing) return; - jl_typemap_visitor(mt->defs, _compile_all_enq, (void*)found); - jl_typemap_visitor(mt->cache, _compile_all_enq, (void*)found); -} - static void _compile_all_enq_module(jl_module_t *m, jl_array_t *found) { // scan through all types reachable from 'v' and @@ -2178,7 +2176,11 @@ static void _compile_all_enq_module(jl_module_t *m, jl_array_t *found) if (jl_is_datatype(v)) { jl_typename_t *tn = ((jl_datatype_t*)v)->name; if (tn->module == m && tn->name == b->name) { - _compile_all_enq_mt(tn->mt, found); + jl_methtable_t *mt = tn->mt; + if (mt != NULL && (jl_value_t*)mt != jl_nothing) { + jl_typemap_visitor(mt->defs, _compile_all_enq, (void*)found); + jl_typemap_visitor(mt->cache, _compile_all_enq, (void*)found); + } } } else if (jl_is_module(v)) { @@ -2193,11 +2195,10 @@ static void _compile_all_enq_module(jl_module_t *m, jl_array_t *found) } } -void jl_compile_all(void) +static void jl_compile_all(void) { // this "found" array will contain - // LambdaInfos that need to be compiled - // and (generic-function, method) pairs that may be optimized (and need to be compiled) + // TypeMapEntries for Methods and LambdaInfos that need to be compiled jl_array_t *m = jl_alloc_cell_1d(0); JL_GC_PUSH1(&m); while (1) { @@ -2210,12 +2211,81 @@ void jl_compile_all(void) } JL_GC_POP(); } -// -JL_DLLEXPORT void jl_compile_hint(jl_tupletype_t *types) + +static int _precompile_enq_tfunc(jl_typemap_entry_t *l, void *closure) { - (void)jl_get_specialization1(types); + if (jl_is_lambda_info(l->func.value) && !l->func.linfo->functionID) + jl_cell_1d_push(closure, (jl_value_t*)l->func.linfo->specTypes); + return 1; +} + +static int _precompile_enq_spec(jl_typemap_entry_t *def, void *closure) +{ + jl_array_t *spec = def->func.method->specializations; + if (spec == NULL) + return 1; + size_t i, l; + for (i = 0, l = jl_array_len(spec); i < l; i++) { + jl_value_t *li = jl_cellref(spec, i); + if (jl_is_lambda_info(li) && !((jl_lambda_info_t*)li)->functionID) + jl_cell_1d_push(closure, (jl_value_t*)((jl_lambda_info_t*)li)->specTypes); + } + jl_typemap_visitor(def->func.method->tfunc, _precompile_enq_tfunc, closure); + return 1; +} + +static void _precompile_enq_module(jl_module_t *m, jl_array_t *unspec) +{ + // removes all method caches + size_t i; + void **table = m->bindings.table; + for(i=1; i < m->bindings.size; i+=2) { + if (table[i] != HT_NOTFOUND) { + jl_binding_t *b = (jl_binding_t*)table[i]; + if (b->owner == m && b->value && b->constp) { + if (jl_is_datatype(b->value)) { + jl_typename_t *tn = ((jl_datatype_t*)b->value)->name; + if (tn->module == m && tn->name == b->name) { + jl_methtable_t *mt = tn->mt; + if (mt != NULL && (jl_value_t*)mt != jl_nothing) { + jl_typemap_visitor(mt->defs, _precompile_enq_spec, (void*)unspec); + } + } + } + else if (jl_is_module(b->value)) { + jl_module_t *child = (jl_module_t*)b->value; + if (child != m && child->parent == m && child->name == b->name) { + // this is the original/primary binding for the submodule + _precompile_enq_module((jl_module_t*)b->value, unspec); + } + } + } + } + } } +static void jl_compile_specializations(void) +{ + // this "found" array will contain function + // type signatures that were inferred but haven't been compiled + jl_array_t *m = jl_alloc_cell_1d(0); + JL_GC_PUSH1(&m); + _precompile_enq_module(jl_main_module, m); + size_t i, l; + for (i = 0, l = jl_array_len(m); i < l; i++) { + jl_compile_hint((jl_tupletype_t*)jl_cellref(m, i)); + } + JL_GC_POP(); +} + +void jl_precompile(int all) { + jl_compile_specializations(); + if (all) + jl_compile_all(); +} + +// + #ifdef JL_TRACE static int trace_en = 0; static int error_en = 1; diff --git a/src/init.c b/src/init.c index aa92a14cb4fa5..1c8f2b12a16a3 100644 --- a/src/init.c +++ b/src/init.c @@ -715,15 +715,15 @@ JL_DLLEXPORT int jl_generating_output(void) return jl_options.outputo || jl_options.outputbc || jl_options.outputji; } -void jl_compile_all(void); +void jl_precompile(int all); static void julia_save(void) { if (!jl_generating_output()) return; - if (jl_options.compile_enabled == JL_OPTIONS_COMPILE_ALL) - jl_compile_all(); + if (!jl_options.incremental) + jl_precompile(jl_options.compile_enabled == JL_OPTIONS_COMPILE_ALL); if (!jl_module_init_order) { jl_printf(JL_STDERR, "WARNING: --output requested, but no modules defined during run\n");