From 7080f68b548b83d97229929dc414eb2c92768af9 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Sun, 26 Jun 2022 04:37:23 -0500 Subject: [PATCH 01/11] Support adding backedges with invokesig This allows a MethodInstance to store dispatch tuples as well as other MethodInstances among their backedges. The motivation is to properly handle invalidation with `invoke`, which allows calling a less-specific method and thus runs afoul of our standard mechanisms to detect "outdated" dispatch. Additionally: - provide backedge-iterators for both C and Julia that abstracts the specific storage mechanism. - fix a bug in the CodeInstance `relocatability` field, where methods that only return a constant (and hence store `nothing` for `inferred`) were deemed non-relocatable. - fix a bug in which #43990 should have checked that the method had not been deleted. Tests passed formerly simply because we weren't caching external CodeInstances that inferred down to a `Const`; fixing that exposed the bug. This bug has been exposed since merging #43990 for non-`Const` inference, and would affect Revise etc. --- NEWS.md | 2 + base/compiler/abstractinterpretation.jl | 9 +- base/compiler/inferencestate.jl | 5 +- base/compiler/optimize.jl | 4 + base/compiler/ssair/inlining.jl | 31 ++-- base/compiler/typeinfer.jl | 13 +- base/compiler/utilities.jl | 59 ++++++++ src/dump.c | 187 ++++++++++++++++-------- src/gf.c | 47 ++++-- src/jltypes.c | 2 +- src/julia.h | 4 +- src/julia_internal.h | 6 +- src/method.c | 43 ++++++ test/precompile.jl | 125 +++++++++++++++- 14 files changed, 430 insertions(+), 107 deletions(-) diff --git a/NEWS.md b/NEWS.md index 421093b6e0681..25efd0eb665ee 100644 --- a/NEWS.md +++ b/NEWS.md @@ -35,6 +35,8 @@ Compiler/Runtime improvements `@nospecialize`-d call sites and avoiding excessive compilation. ([#44512]) * All the previous usages of `@pure`-macro in `Base` has been replaced with the preferred `Base.@assume_effects`-based annotations. ([#44776]) +* `invoke(f, invokesig, args...)` calls to a less-specific method than would normally be chosen + for `f(args...)` are no longer spuriously invalidated when loading package precompile files. ([#46010]) Command-line option changes --------------------------- diff --git a/base/compiler/abstractinterpretation.jl b/base/compiler/abstractinterpretation.jl index 31e34506a9046..0f504b9337c3f 100644 --- a/base/compiler/abstractinterpretation.jl +++ b/base/compiler/abstractinterpretation.jl @@ -801,6 +801,13 @@ function collect_const_args(argtypes::Vector{Any}) end for i = 2:length(argtypes) ] end +function invoke_signature(invokesig::Vector{Any}) + unwrapconst(x) = isa(x, Const) ? x.val : x + + f, argtyps = unwrapconst(invokesig[2]), unwrapconst(invokesig[3]) + return Tuple{typeof(f), unwrap_unionall(argtyps).parameters...} +end + function concrete_eval_call(interp::AbstractInterpreter, @nospecialize(f), result::MethodCallResult, arginfo::ArgInfo, sv::InferenceState) concrete_eval_eligible(interp, f, result, arginfo, sv) || return nothing @@ -1631,7 +1638,7 @@ function abstract_invoke(interp::AbstractInterpreter, (; fargs, argtypes)::ArgIn ti = tienv[1]; env = tienv[2]::SimpleVector result = abstract_call_method(interp, method, ti, env, false, sv) (; rt, edge, effects) = result - edge !== nothing && add_backedge!(edge::MethodInstance, sv) + edge !== nothing && add_backedge!(edge::MethodInstance, sv, argtypes) match = MethodMatch(ti, env, method, argtype <: method.sig) res = nothing sig = match.spec_types diff --git a/base/compiler/inferencestate.jl b/base/compiler/inferencestate.jl index d81bdccb7fa1c..732cd86e34fcc 100644 --- a/base/compiler/inferencestate.jl +++ b/base/compiler/inferencestate.jl @@ -479,12 +479,15 @@ function add_cycle_backedge!(frame::InferenceState, caller::InferenceState, curr end # temporarily accumulate our edges to later add as backedges in the callee -function add_backedge!(li::MethodInstance, caller::InferenceState) +function add_backedge!(li::MethodInstance, caller::InferenceState, invokesig::Union{Nothing,Vector{Any}}=nothing) isa(caller.linfo.def, Method) || return # don't add backedges to toplevel exprs edges = caller.stmt_edges[caller.currpc] if edges === nothing edges = caller.stmt_edges[caller.currpc] = [] end + if invokesig !== nothing + push!(edges, invoke_signature(invokesig)) + end push!(edges, li) return nothing end diff --git a/base/compiler/optimize.jl b/base/compiler/optimize.jl index e9c37a3054352..f19fbd014a04e 100644 --- a/base/compiler/optimize.jl +++ b/base/compiler/optimize.jl @@ -62,6 +62,10 @@ intersect!(et::EdgeTracker, range::WorldRange) = et.valid_worlds[] = intersect(et.valid_worlds[], range) push!(et::EdgeTracker, mi::MethodInstance) = push!(et.edges, mi) +function add_edge!(et::EdgeTracker, @nospecialize(invokesig), mi::MethodInstance) + invokesig === nothing && return push!(et.edges, mi) + push!(et.edges, invokesig, mi) +end function push!(et::EdgeTracker, ci::CodeInstance) intersect!(et, WorldRange(min_world(li), max_world(li))) push!(et, ci.def) diff --git a/base/compiler/ssair/inlining.jl b/base/compiler/ssair/inlining.jl index 44d9067679d52..a59d9e2ec41e2 100644 --- a/base/compiler/ssair/inlining.jl +++ b/base/compiler/ssair/inlining.jl @@ -29,7 +29,9 @@ pass to apply its own inlining policy decisions. struct DelayedInliningSpec match::Union{MethodMatch, InferenceResult} argtypes::Vector{Any} + invokesig # either nothing or a signature (signature is for an `invoke` call) end +DelayedInliningSpec(match, argtypes) = DelayedInliningSpec(match, argtypes, nothing) struct InliningTodo # The MethodInstance to be inlined @@ -37,11 +39,11 @@ struct InliningTodo spec::Union{ResolvedInliningSpec, DelayedInliningSpec} end -InliningTodo(mi::MethodInstance, match::MethodMatch, argtypes::Vector{Any}) = - InliningTodo(mi, DelayedInliningSpec(match, argtypes)) +InliningTodo(mi::MethodInstance, match::MethodMatch, argtypes::Vector{Any}, invokesig=nothing) = + InliningTodo(mi, DelayedInliningSpec(match, argtypes, invokesig)) -InliningTodo(result::InferenceResult, argtypes::Vector{Any}) = - InliningTodo(result.linfo, DelayedInliningSpec(result, argtypes)) +InliningTodo(result::InferenceResult, argtypes::Vector{Any}, invokesig=nothing) = + InliningTodo(result.linfo, DelayedInliningSpec(result, argtypes, invokesig)) struct ConstantCase val::Any @@ -810,7 +812,7 @@ end function resolve_todo(todo::InliningTodo, state::InliningState, flag::UInt8) mi = todo.mi - (; match, argtypes) = todo.spec::DelayedInliningSpec + (; match, argtypes, invokesig) = todo.spec::DelayedInliningSpec et = state.et #XXX: update_valid_age!(min_valid[1], max_valid[1], sv) @@ -818,7 +820,7 @@ function resolve_todo(todo::InliningTodo, state::InliningState, flag::UInt8) inferred_src = match.src if isa(inferred_src, ConstAPI) # use constant calling convention - et !== nothing && push!(et, mi) + et !== nothing && add_edge!(et, invokesig, mi) return ConstantCase(quoted(inferred_src.val)) else src = inferred_src # ::Union{Nothing,CodeInfo} for NativeInterpreter @@ -829,7 +831,7 @@ function resolve_todo(todo::InliningTodo, state::InliningState, flag::UInt8) if code isa CodeInstance if use_const_api(code) # in this case function can be inlined to a constant - et !== nothing && push!(et, mi) + et !== nothing && add_edge!(et, invokesig, mi) return ConstantCase(quoted(code.rettype_const)) else src = @atomic :monotonic code.inferred @@ -851,7 +853,7 @@ function resolve_todo(todo::InliningTodo, state::InliningState, flag::UInt8) src === nothing && return compileable_specialization(et, match, effects) - et !== nothing && push!(et, mi) + et !== nothing && add_edge!(et, invokesig, mi) return InliningTodo(mi, retrieve_ir_for_inlining(mi, src), effects) end @@ -873,7 +875,7 @@ function validate_sparams(sparams::SimpleVector) return true end -function analyze_method!(match::MethodMatch, argtypes::Vector{Any}, +function analyze_method!(match::MethodMatch, argtypes::Vector{Any}, invokesig, flag::UInt8, state::InliningState) method = match.method spec_types = match.spec_types @@ -905,7 +907,7 @@ function analyze_method!(match::MethodMatch, argtypes::Vector{Any}, mi = specialize_method(match; preexisting=true) # Union{Nothing, MethodInstance} isa(mi, MethodInstance) || return compileable_specialization(et, match, Effects()) - todo = InliningTodo(mi, match, argtypes) + todo = InliningTodo(mi, match, argtypes, invokesig) # If we don't have caches here, delay resolving this MethodInstance # until the batch inlining step (or an external post-processing pass) state.mi_cache === nothing && return todo @@ -1100,9 +1102,10 @@ function inline_invoke!( if isa(result, ConcreteResult) item = concrete_result_item(result, state) else + invokesig = invoke_signature(sig.argtypes) argtypes = invoke_rewrite(sig.argtypes) if isa(result, ConstPropResult) - (; mi) = item = InliningTodo(result.result, argtypes) + (; mi) = item = InliningTodo(result.result, argtypes, invokesig) validate_sparams(mi.sparam_vals) || return nothing if argtypes_to_type(argtypes) <: mi.def.sig state.mi_cache !== nothing && (item = resolve_todo(item, state, flag)) @@ -1110,7 +1113,7 @@ function inline_invoke!( return nothing end end - item = analyze_method!(match, argtypes, flag, state) + item = analyze_method!(match, argtypes, invokesig, flag, state) end handle_single_case!(ir, idx, stmt, item, todo, state.params, true) return nothing @@ -1328,7 +1331,7 @@ function handle_match!( # during abstract interpretation: for the purpose of inlining, we can just skip # processing this dispatch candidate _any(case->case.sig === spec_types, cases) && return true - item = analyze_method!(match, argtypes, flag, state) + item = analyze_method!(match, argtypes, nothing, flag, state) item === nothing && return false push!(cases, InliningCase(spec_types, item)) return true @@ -1475,7 +1478,7 @@ function assemble_inline_todo!(ir::IRCode, state::InliningState) if isa(result, ConcreteResult) item = concrete_result_item(result, state) else - item = analyze_method!(info.match, sig.argtypes, flag, state) + item = analyze_method!(info.match, sig.argtypes, nothing, flag, state) end handle_single_case!(ir, idx, stmt, item, todo, state.params) end diff --git a/base/compiler/typeinfer.jl b/base/compiler/typeinfer.jl index 2c66083b9024b..46897769046ce 100644 --- a/base/compiler/typeinfer.jl +++ b/base/compiler/typeinfer.jl @@ -312,7 +312,9 @@ function CodeInstance( const_flags = 0x00 end end - relocatability = isa(inferred_result, Vector{UInt8}) ? inferred_result[end] : UInt8(0) + relocatability = isa(inferred_result, Vector{UInt8}) ? inferred_result[end] : + inferred_result === nothing ? UInt8(1) : UInt8(0) + # relocatability = isa(inferred_result, Vector{UInt8}) ? inferred_result[end] : UInt8(0) return CodeInstance(result.linfo, widenconst(result_type), rettype_const, inferred_result, const_flags, first(valid_worlds), last(valid_worlds), @@ -561,17 +563,12 @@ function store_backedges(frame::InferenceResult, edges::Vector{Any}) end function store_backedges(caller::MethodInstance, edges::Vector{Any}) - i = 1 - while i <= length(edges) - to = edges[i] + for (typ, to) in BackedgeIterator(edges) if isa(to, MethodInstance) - ccall(:jl_method_instance_add_backedge, Cvoid, (Any, Any), to, caller) - i += 1 + ccall(:jl_method_instance_add_backedge, Cvoid, (Any, Any, Any), to, typ, caller) else typeassert(to, Core.MethodTable) - typ = edges[i + 1] ccall(:jl_method_table_add_backedge, Cvoid, (Any, Any, Any), to, typ, caller) - i += 2 end end end diff --git a/base/compiler/utilities.jl b/base/compiler/utilities.jl index d793660195492..975dc3d1a9b5d 100644 --- a/base/compiler/utilities.jl +++ b/base/compiler/utilities.jl @@ -223,6 +223,65 @@ Check if `method` is declared as `Base.@constprop :none`. """ is_no_constprop(method::Union{Method,CodeInfo}) = method.constprop == 0x02 +############# +# backedges # +############# + +""" + BackedgeIterator(mi::MethodInstance) + BackedgeIterator(backedges::Vector{Any}) + +Return an iterator over a list of backedges, which may be extracted +from `mi`. Iteration returns `(sig, caller)` elements, which will be one of +the following: + +- `(nothing, caller::MethodInstance)`: a call made by ordinary inferrable dispatch +- `(invokesig, caller::MethodInstance)`: a call made by `invoke(f, invokesig, args...)` +- `(specsig, mt::MethodTable)`: an abstract call + +# Examples + +```julia +julia> callme(x) = x+1 +callme (generic function with 1 method) + +julia> callyou(x) = callme(x) +callyou (generic function with 1 method) + +julia> callyou(2.0) +3.0 + +julia> mi = first(which(callme, (Any,)).specializations) +MethodInstance for callme(::Float64) + +julia> @eval Core.Compiler for (sig, caller) in BackedgeIterator(Main.mi) + println(sig) + println(caller) + end +nothing +callyou(Float64) from callyou(Any) +``` +""" +struct BackedgeIterator + backedges::Vector{Any} +end + +const empty_backedge_iter = BackedgeIterator(Any[]) + +function BackedgeIterator(mi::MethodInstance) + isdefined(mi, :backedges) || return empty_backedge_iter + return BackedgeIterator(mi.backedges) +end + +function iterate(iter::BackedgeIterator, i::Int=1) + backedges = iter.backedges + i > length(backedges) && return nothing + item = backedges[i] + isa(item, MethodInstance) && return (nothing, item), i+1 # regular dispatch + isa(item, Core.MethodTable) && return (backedges[i+1], item), i+2 # abstract dispatch + return (item, backedges[i+1]::MethodInstance), i+2 # `invoke` calls +end + ######### # types # ######### diff --git a/src/dump.c b/src/dump.c index 27c10254eac51..a3a04f0f6d9bb 100644 --- a/src/dump.c +++ b/src/dump.c @@ -337,9 +337,10 @@ static int has_backedge_to_worklist(jl_method_instance_t *mi, htable_t *visited) if (!mi->backedges) { return 0; } - size_t i, n = jl_array_len(mi->backedges); - for (i = 0; i < n; i++) { - jl_method_instance_t *be = (jl_method_instance_t*)jl_array_ptr_ref(mi->backedges, i); + size_t i = 0, n = jl_array_len(mi->backedges); + jl_method_instance_t *be; + while (i < n) { + i = get_next_backedge(mi->backedges, i, NULL, &be); if (has_backedge_to_worklist(be, visited)) { bp = ptrhash_bp(visited, mi); // re-acquire since rehashing might change the location *bp = (void*)((char*)HT_NOTFOUND + 2); // found @@ -369,7 +370,8 @@ static size_t queue_external_mis(jl_array_t *list) jl_code_instance_t *ci = mi->cache; int relocatable = 0; while (ci) { - relocatable |= ci->relocatability; + if (ci->max_world == ~(size_t)0) + relocatable |= ci->relocatability; ci = ci->next; } if (relocatable && ptrhash_get(&external_mis, mi) == HT_NOTFOUND) { @@ -947,12 +949,16 @@ static void jl_serialize_value_(jl_serializer_state *s, jl_value_t *v, int as_li if (backedges) { // filter backedges to only contain pointers // to items that we will actually store (internal >= 2) - size_t ins, i, l = jl_array_len(backedges); - jl_method_instance_t **b_edges = (jl_method_instance_t**)jl_array_data(backedges); - for (ins = i = 0; i < l; i++) { - jl_method_instance_t *backedge = b_edges[i]; + size_t ins = 0, i = 0, l = jl_array_len(backedges); + jl_value_t **b_edges = (jl_value_t**)jl_array_data(backedges); + jl_value_t *invokeTypes; + jl_method_instance_t *backedge; + while (i < l) { + i = get_next_backedge(backedges, i, &invokeTypes, &backedge); if (module_in_worklist(backedge->def.method->module) || method_instance_in_queue(backedge)) { - b_edges[ins++] = backedge; + if (invokeTypes) + b_edges[ins++] = invokeTypes; + b_edges[ins++] = (jl_value_t*)backedge; } } if (ins != l) @@ -1168,7 +1174,9 @@ static void jl_collect_missing_backedges_to_mod(jl_methtable_t *mt) jl_array_t **edges = (jl_array_t**)ptrhash_bp(&edges_map, (void*)caller); if (*edges == HT_NOTFOUND) *edges = jl_alloc_vec_any(0); - jl_array_ptr_1d_push(*edges, missing_callee); + // To stay synchronized with the format from MethodInstances (specifically for `invoke`d calls), + // we have to push a pair of values. But in this case the callee is unknown, so we leave it NULL. + push_backedge(*edges, missing_callee, NULL); } } } @@ -1178,13 +1186,15 @@ static void collect_backedges(jl_method_instance_t *callee) JL_GC_DISABLED { jl_array_t *backedges = callee->backedges; if (backedges) { - size_t i, l = jl_array_len(backedges); - for (i = 0; i < l; i++) { - jl_method_instance_t *caller = (jl_method_instance_t*)jl_array_ptr_ref(backedges, i); + size_t i = 0, l = jl_array_len(backedges); + jl_value_t *invokeTypes; + jl_method_instance_t *caller; + while (i < l) { + i = get_next_backedge(backedges, i, &invokeTypes, &caller); jl_array_t **edges = (jl_array_t**)ptrhash_bp(&edges_map, caller); if (*edges == HT_NOTFOUND) *edges = jl_alloc_vec_any(0); - jl_array_ptr_1d_push(*edges, (jl_value_t*)callee); + push_backedge(*edges, invokeTypes, callee); } } } @@ -1268,6 +1278,15 @@ static void jl_collect_extext_methods_from_mod(jl_array_t *s, jl_module_t *m) JL } } +static void register_backedge(htable_t *all_callees, jl_value_t *invokeTypes, jl_value_t *c) +{ + if (invokeTypes) + ptrhash_put(all_callees, invokeTypes, c); + else + ptrhash_put(all_callees, c, c); + +} + // flatten the backedge map reachable from caller into callees static void jl_collect_backedges_to(jl_method_instance_t *caller, htable_t *all_callees) JL_GC_DISABLED { @@ -1278,11 +1297,13 @@ static void jl_collect_backedges_to(jl_method_instance_t *caller, htable_t *all_ *callees = *pcallees; assert(callees != HT_NOTFOUND); *pcallees = (jl_array_t*) HT_NOTFOUND; - size_t i, l = jl_array_len(callees); - for (i = 0; i < l; i++) { - jl_value_t *c = jl_array_ptr_ref(callees, i); - ptrhash_put(all_callees, c, c); - if (jl_is_method_instance(c)) { + size_t i = 0, l = jl_array_len(callees); + jl_method_instance_t *c; + jl_value_t *invokeTypes; + while (i < l) { + i = get_next_backedge(callees, i, &invokeTypes, &c); + register_backedge(all_callees, invokeTypes, (jl_value_t*)c); + if (c && jl_is_method_instance(c)) { jl_collect_backedges_to((jl_method_instance_t*)c, all_callees); } } @@ -1297,6 +1318,8 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j htable_t all_callees; // MIs called by worklist methods (eff. Set{MethodInstance}) htable_new(&all_targets, 0); htable_new(&all_callees, 0); + jl_value_t *invokeTypes; + jl_method_instance_t *c; size_t i; void **table = edges_map.table; // edges is caller => callees size_t table_size = edges_map.size; @@ -1309,10 +1332,10 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j continue; assert(jl_is_method_instance(caller) && jl_is_method(caller->def.method)); if (module_in_worklist(caller->def.method->module) || method_instance_in_queue(caller)) { - size_t i, l = jl_array_len(callees); - for (i = 0; i < l; i++) { - jl_value_t *c = jl_array_ptr_ref(callees, i); - ptrhash_put(&all_callees, c, c); + size_t i = 0, l = jl_array_len(callees); + while (i < l) { + i = get_next_backedge(callees, i, &invokeTypes, &c); + register_backedge(&all_callees, invokeTypes, (jl_value_t*)c); if (jl_is_method_instance(c)) { jl_collect_backedges_to((jl_method_instance_t*)c, &all_callees); } @@ -1326,10 +1349,9 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j jl_value_t *callee = (jl_value_t*)pc[j]; void *target = ptrhash_get(&all_targets, (void*)callee); if (target == HT_NOTFOUND) { - jl_method_instance_t *callee_mi = (jl_method_instance_t*)callee; jl_value_t *sig; if (jl_is_method_instance(callee)) { - sig = callee_mi->specTypes; + sig = ((jl_method_instance_t*)callee)->specTypes; } else { sig = callee; @@ -2294,7 +2316,7 @@ void remove_code_instance_from_validation(jl_code_instance_t *codeinst) ptrhash_remove(&new_code_instance_validate, codeinst); } -static void jl_insert_method_instances(jl_array_t *list) +static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED { size_t i, l = jl_array_len(list); // Validate the MethodInstances @@ -2303,29 +2325,74 @@ static void jl_insert_method_instances(jl_array_t *list) size_t world = jl_atomic_load_acquire(&jl_world_counter); for (i = 0; i < l; i++) { jl_method_instance_t *mi = (jl_method_instance_t*)jl_array_ptr_ref(list, i); + int valid = 1; assert(jl_is_method_instance(mi)); if (jl_is_method(mi->def.method)) { - // Is this still the method we'd be calling? - jl_methtable_t *mt = jl_method_table_for(mi->specTypes); - struct jl_typemap_assoc search = {(jl_value_t*)mi->specTypes, world, NULL, 0, ~(size_t)0}; - jl_typemap_entry_t *entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/1); - if (entry) { - jl_value_t *mworld = entry->func.value; - if (jl_is_method(mworld) && mi->def.method != (jl_method_t*)mworld && jl_type_morespecific(((jl_method_t*)mworld)->sig, mi->def.method->sig)) { - jl_array_uint8_set(valids, i, 0); - invalidate_backedges(&remove_code_instance_from_validation, mi, world, "jl_insert_method_instance"); - // The codeinst of this mi haven't yet been removed - jl_code_instance_t *codeinst = mi->cache; - while (codeinst) { - remove_code_instance_from_validation(codeinst); - codeinst = codeinst->next; - } - if (_jl_debug_method_invalidation) { - jl_array_ptr_1d_push(_jl_debug_method_invalidation, mworld); - jl_array_ptr_1d_push(_jl_debug_method_invalidation, jl_cstr_to_string("jl_method_table_insert")); // GC disabled + jl_method_t *m = mi->def.method; + if (m->deleted_world != ~(size_t)0) { + // The method we depended on has been deleted, invalidate + valid = 0; + } else { + // Is this still the method we'd be calling? + jl_methtable_t *mt = jl_method_table_for(mi->specTypes); + struct jl_typemap_assoc search = {(jl_value_t*)mi->specTypes, world, NULL, 0, ~(size_t)0}; + jl_typemap_entry_t *entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/1); + if (entry) { + jl_value_t *mworld = entry->func.value; + if (jl_is_method(mworld) && mi->def.method != (jl_method_t*)mworld && jl_type_morespecific(((jl_method_t*)mworld)->sig, mi->def.method->sig)) { + // There's still a chance this is valid, if any caller made this via `invoke` and the invoke-signature is still valid + assert(mi->backedges); // should not be NULL if it's on `list` + jl_value_t *invokeTypes; + jl_method_instance_t *caller; + size_t jins = 0, j0, j = 0, nbe = jl_array_len(mi->backedges); + while (j < nbe) { + j0 = j; + j = get_next_backedge(mi->backedges, j, &invokeTypes, &caller); + if (invokeTypes) { + struct jl_typemap_assoc search = {invokeTypes, world, NULL, 0, ~(size_t)0}; + entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/1); + if (entry) { + jl_value_t *imworld = entry->func.value; + if (jl_is_method(imworld) && mi->def.method == (jl_method_t*)imworld) { + // this one is OK + // in case we deleted some earlier ones, move this earlier + for (; j0 < j; jins++, j0++) { + jl_array_ptr_set(mi->backedges, jins, jl_array_ptr_ref(mi->backedges, j0)); + } + continue; + } + } + } + invalidate_backedges(&remove_code_instance_from_validation, caller, world, "jl_insert_method_instance"); + // The codeinst of this mi haven't yet been removed + jl_code_instance_t *codeinst = caller->cache; + while (codeinst) { + remove_code_instance_from_validation(codeinst); + codeinst = codeinst->next; + } + } + jl_array_del_end(mi->backedges, j - jins); + if (jins == 0) { + m = (jl_method_t*)mworld; + valid = 0; + } } } } + if (!valid) { + // None of the callers were valid, so invalidate `mi` too + jl_array_uint8_set(valids, i, 0); + invalidate_backedges(&remove_code_instance_from_validation, mi, world, "jl_insert_method_instance"); + jl_code_instance_t *codeinst = mi->cache; + while (codeinst) { + remove_code_instance_from_validation(codeinst); + codeinst = codeinst->next; + } + if (_jl_debug_method_invalidation) { + jl_array_ptr_1d_push(_jl_debug_method_invalidation, (jl_value_t*)m); + jl_array_ptr_1d_push(_jl_debug_method_invalidation, jl_cstr_to_string("jl_method_table_insert")); // GC disabled + } + } } } // While it's tempting to just remove the invalidated MIs altogether, @@ -2342,36 +2409,42 @@ static void jl_insert_method_instances(jl_array_t *list) if (milive != mi) { // A previously-loaded module compiled this method, so the one we deserialized will be dropped. // But make sure the backedges are copied over. + jl_value_t *invokeTypes; + jl_method_instance_t *be, *belive; if (mi->backedges) { if (!milive->backedges) { // Copy all the backedges (after looking up the live ones) - size_t j, n = jl_array_len(mi->backedges); + size_t j = 0, jlive = 0, n = jl_array_len(mi->backedges); milive->backedges = jl_alloc_vec_any(n); jl_gc_wb(milive, milive->backedges); - for (j = 0; j < n; j++) { - jl_method_instance_t *be = (jl_method_instance_t*)jl_array_ptr_ref(mi->backedges, j); - jl_method_instance_t *belive = (jl_method_instance_t*)ptrhash_get(&uniquing_table, be); + while (j < n) { + j = get_next_backedge(mi->backedges, j, &invokeTypes, &be); + belive = (jl_method_instance_t*)ptrhash_get(&uniquing_table, be); if (belive == HT_NOTFOUND) belive = be; - jl_array_ptr_set(milive->backedges, j, belive); + jlive = set_next_backedge(milive->backedges, jlive, invokeTypes, belive); } } else { // Copy the missing backedges (this is an O(N^2) algorithm, but many methods have few MethodInstances) - size_t j, k, n = jl_array_len(mi->backedges), nlive = jl_array_len(milive->backedges); - for (j = 0; j < n; j++) { - jl_method_instance_t *be = (jl_method_instance_t*)jl_array_ptr_ref(mi->backedges, j); - jl_method_instance_t *belive = (jl_method_instance_t*)ptrhash_get(&uniquing_table, be); + size_t j = 0, k, n = jl_array_len(mi->backedges), nlive = jl_array_len(milive->backedges); + jl_value_t *invokeTypes2; + jl_method_instance_t *belive2; + while (j < n) { + j = get_next_backedge(mi->backedges, j, &invokeTypes, &be); + belive = (jl_method_instance_t*)ptrhash_get(&uniquing_table, be); if (belive == HT_NOTFOUND) belive = be; int found = 0; - for (k = 0; k < nlive; k++) { - if (belive == (jl_method_instance_t*)jl_array_ptr_ref(milive->backedges, k)) { + k = 0; + while (k < nlive) { + k = get_next_backedge(milive->backedges, k, &invokeTypes2, &belive2); + if (belive == belive2 && invokeTypes == invokeTypes2) { found = 1; break; } } if (!found) - jl_array_ptr_1d_push(milive->backedges, (jl_value_t*)belive); + push_backedge(milive->backedges, invokeTypes, belive); } } } @@ -2482,7 +2555,7 @@ static void jl_insert_backedges(jl_array_t *list, jl_array_t *targets) int32_t idx = idxs[j]; jl_value_t *callee = jl_array_ptr_ref(targets, idx * 2); if (jl_is_method_instance(callee)) { - jl_method_instance_add_backedge((jl_method_instance_t*)callee, caller); + jl_method_instance_add_backedge((jl_method_instance_t*)callee, NULL, caller); } else { jl_methtable_t *mt = jl_method_table_for(callee); diff --git a/src/gf.c b/src/gf.c index c9d91ec836a0b..6b2d9362198f7 100644 --- a/src/gf.c +++ b/src/gf.c @@ -1445,18 +1445,23 @@ static void invalidate_method_instance(void (*f)(jl_code_instance_t*), jl_method codeinst->max_world = max_world; } assert(codeinst->max_world <= max_world); + JL_GC_PUSH1(&codeinst); (*f)(codeinst); + JL_GC_POP(); codeinst = jl_atomic_load_relaxed(&codeinst->next); } // recurse to all backedges to update their valid range also jl_array_t *backedges = replaced->backedges; if (backedges) { + JL_GC_PUSH1(&backedges); replaced->backedges = NULL; - size_t i, l = jl_array_len(backedges); - for (i = 0; i < l; i++) { - jl_method_instance_t *replaced = (jl_method_instance_t*)jl_array_ptr_ref(backedges, i); + size_t i = 0, l = jl_array_len(backedges); + jl_method_instance_t *replaced; + while (i < l) { + i = get_next_backedge(backedges, i, NULL, &replaced); invalidate_method_instance(f, replaced, max_world, depth + 1); } + JL_GC_POP(); } JL_UNLOCK(&replaced->def.method->writelock); } @@ -1469,10 +1474,11 @@ void invalidate_backedges(void (*f)(jl_code_instance_t*), jl_method_instance_t * if (backedges) { // invalidate callers (if any) replaced_mi->backedges = NULL; - size_t i, l = jl_array_len(backedges); - jl_method_instance_t **replaced = (jl_method_instance_t**)jl_array_ptr_data(backedges); - for (i = 0; i < l; i++) { - invalidate_method_instance(f, replaced[i], max_world, 1); + size_t i = 0, l = jl_array_len(backedges); + jl_method_instance_t *replaced; + while (i < l) { + i = get_next_backedge(backedges, i, NULL, &replaced); + invalidate_method_instance(f, replaced, max_world, 1); } } JL_UNLOCK(&replaced_mi->def.method->writelock); @@ -1486,23 +1492,34 @@ void invalidate_backedges(void (*f)(jl_code_instance_t*), jl_method_instance_t * } // add a backedge from callee to caller -JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, jl_method_instance_t *caller) +JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, jl_value_t *invokesig, jl_method_instance_t *caller) { JL_LOCK(&callee->def.method->writelock); + if (invokesig == jl_nothing) + invokesig = NULL; // julia uses `nothing` but C uses NULL (#undef) if (!callee->backedges) { // lazy-init the backedges array - callee->backedges = jl_alloc_vec_any(1); + callee->backedges = jl_alloc_vec_any(0); jl_gc_wb(callee, callee->backedges); - jl_array_ptr_set(callee->backedges, 0, caller); + push_backedge(callee->backedges, invokesig, caller); } else { - size_t i, l = jl_array_len(callee->backedges); - for (i = 0; i < l; i++) { - if (jl_array_ptr_ref(callee->backedges, i) == (jl_value_t*)caller) + size_t i = 0, l = jl_array_len(callee->backedges); + int found = 0; + jl_value_t *invokeTypes; + jl_method_instance_t *mi; + while (i < l) { + i = get_next_backedge(callee->backedges, i, &invokeTypes, &mi); + // TODO: it would be better to canonicalize (how?) the Tuple-type so + // that we don't have to call `jl_egal` + if (mi == caller && ((invokesig == NULL && invokeTypes == NULL) || + (invokesig && invokeTypes && jl_egal(invokesig, invokeTypes)))) { + found = 1; break; + } } - if (i == l) { - jl_array_ptr_1d_push(callee->backedges, (jl_value_t*)caller); + if (!found) { + push_backedge(callee->backedges, invokesig, caller); } } JL_UNLOCK(&callee->def.method->writelock); diff --git a/src/jltypes.c b/src/jltypes.c index 8eb43076e46a5..2dc185db27c9b 100644 --- a/src/jltypes.c +++ b/src/jltypes.c @@ -2532,7 +2532,7 @@ void jl_init_types(void) JL_GC_DISABLED jl_any_type, jl_simplevector_type, jl_any_type, - jl_any_type, + jl_array_any_type, jl_any_type, jl_any_type, jl_bool_type, diff --git a/src/julia.h b/src/julia.h index 30a3533156775..f780522081c3c 100644 --- a/src/julia.h +++ b/src/julia.h @@ -362,7 +362,7 @@ struct _jl_method_instance_t { jl_value_t *specTypes; // argument types this was specialized for jl_svec_t *sparam_vals; // static parameter values, indexed by def.method->sparam_syms jl_value_t *uninferred; // cached uncompressed code, for generated functions, top-level thunks, or the interpreter - jl_array_t *backedges; // list of method-instances which contain a call into this method-instance + jl_array_t *backedges; // list of method-instances which call this method-instance; `invoke` records (invokesig, caller) pairs jl_array_t *callbacks; // list of callback functions to inform external caches about invalidations _Atomic(struct _jl_code_instance_t*) cache; uint8_t inInference; // flags to tell if inference is running on this object @@ -650,7 +650,7 @@ typedef struct _jl_methtable_t { intptr_t max_args; // max # of non-vararg arguments in a signature jl_value_t *kwsorter; // keyword argument sorter function jl_module_t *module; // used for incremental serialization to locate original binding - jl_array_t *backedges; + jl_array_t *backedges; // (sig, caller::MethodInstance) pairs jl_mutex_t writelock; uint8_t offs; // 0, or 1 to skip splitting typemap on first (function) argument uint8_t frozen; // whether this accepts adding new methods diff --git a/src/julia_internal.h b/src/julia_internal.h index 07ec5e8aedda2..8460e341e070d 100644 --- a/src/julia_internal.h +++ b/src/julia_internal.h @@ -604,6 +604,10 @@ JL_DLLEXPORT jl_code_info_t *jl_new_code_info_uninit(void); void jl_resolve_globals_in_ir(jl_array_t *stmts, jl_module_t *m, jl_svec_t *sparam_vals, int binding_effects); +int get_next_backedge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method_instance_t **caller) JL_NOTSAFEPOINT; +int set_next_backedge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_instance_t *caller); +void push_backedge(jl_array_t *list, jl_value_t *invokesig, jl_method_instance_t *caller); + JL_DLLEXPORT void jl_add_method_root(jl_method_t *m, jl_module_t *mod, jl_value_t* root); void jl_append_method_roots(jl_method_t *m, uint64_t modid, jl_array_t* roots); int get_root_reference(rle_reference *rr, jl_method_t *m, size_t i); @@ -949,7 +953,7 @@ JL_DLLEXPORT jl_value_t *jl_methtable_lookup(jl_methtable_t *mt, jl_value_t *typ JL_DLLEXPORT jl_method_instance_t *jl_specializations_get_linfo( jl_method_t *m JL_PROPAGATES_ROOT, jl_value_t *type, jl_svec_t *sparams); jl_method_instance_t *jl_specializations_get_or_insert(jl_method_instance_t *mi_ins); -JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, jl_method_instance_t *caller); +JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, jl_value_t *invokesig, jl_method_instance_t *caller); JL_DLLEXPORT void jl_method_table_add_backedge(jl_methtable_t *mt, jl_value_t *typ, jl_value_t *caller); uint32_t jl_module_next_counter(jl_module_t *m) JL_NOTSAFEPOINT; diff --git a/src/method.c b/src/method.c index 1abacfaa58e55..ce031cc1e5dc2 100644 --- a/src/method.c +++ b/src/method.c @@ -784,6 +784,49 @@ JL_DLLEXPORT jl_method_t *jl_new_method_uninit(jl_module_t *module) return m; } +// backedges ------------------------------------------------------------------ + +// Use this in a `while` loop to iterate over the backedges in a MethodInstance. +// `*invokesig` will be NULL if the call was made by ordinary dispatch, otherwise +// it will be the signature supplied in an `invoke` call. +// If you don't need `invokesig`, you can set it to NULL on input. +// Initialize iteration with `i = 0`. Returns `i` for the next backedge to be extracted. +int get_next_backedge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method_instance_t **caller) JL_NOTSAFEPOINT +{ + jl_value_t *item = jl_array_ptr_ref(list, i); + if (jl_is_method_instance(item)) { + // Not an `invoke` call, it's just the MethodInstance + if (invokesig != NULL) + *invokesig = NULL; + *caller = (jl_method_instance_t*)item; + return i + 1; + } + assert(jl_is_type(item)); + // An `invoke` call, it's a (sig, MethodInstance) pair + if (invokesig != NULL) + *invokesig = item; + *caller = (jl_method_instance_t*)jl_array_ptr_ref(list, i + 1); + if (*caller) + assert(jl_is_method_instance(*caller)); + return i + 2; +} + +int set_next_backedge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_instance_t *caller) +{ + if (invokesig) + jl_array_ptr_set(list, i++, invokesig); + jl_array_ptr_set(list, i++, caller); + return i; +} + +void push_backedge(jl_array_t *list, jl_value_t *invokesig, jl_method_instance_t *caller) +{ + if (invokesig) + jl_array_ptr_1d_push(list, invokesig); + jl_array_ptr_1d_push(list, (jl_value_t*)caller); + return; +} + // method definition ---------------------------------------------------------- jl_method_t *jl_make_opaque_closure_method(jl_module_t *module, jl_value_t *name, diff --git a/test/precompile.jl b/test/precompile.jl index ac2c63ff7af08..be4cbccf1172d 100644 --- a/test/precompile.jl +++ b/test/precompile.jl @@ -106,16 +106,18 @@ precompile_test_harness(false) do dir write(Foo2_file, """ module $Foo2_module - export override + export override, overridenc override(x::Integer) = 2 override(x::AbstractFloat) = Float64(override(1)) + overridenc(x::Integer) = rand()+1 + overridenc(x::AbstractFloat) = Float64(overridenc(1)) end """) write(Foo_file, """ module $Foo_module import $FooBase_module, $FooBase_module.typeA - import $Foo2_module: $Foo2_module, override + import $Foo2_module: $Foo2_module, override, overridenc import $FooBase_module.hash import Test module Inner @@ -221,6 +223,8 @@ precompile_test_harness(false) do dir g() = override(1.0) Test.@test g() === 2.0 # compile this + gnc() = overridenc(1.0) + Test.@test 1 < gnc() < 5 # compile this const abigfloat_f() = big"12.34" const abigfloat_x = big"43.21" @@ -257,6 +261,8 @@ precompile_test_harness(false) do dir Foo2 = Base.require(Main, Foo2_module) @eval $Foo2.override(::Int) = 'a' @eval $Foo2.override(::Float32) = 'b' + @eval $Foo2.overridenc(::Int) = rand() + 97.0 + @eval $Foo2.overridenc(::Float32) = rand() + 100.0 Foo = Base.require(Main, Foo_module) Base.invokelatest() do # use invokelatest to see the results of loading the compile @@ -265,9 +271,13 @@ precompile_test_harness(false) do dir # Issue #21307 @test Foo.g() === 97.0 + @test 96 < Foo.gnc() < 99 @test Foo.override(1.0e0) == Float64('a') @test Foo.override(1.0f0) == 'b' @test Foo.override(UInt(1)) == 2 + @test 96 < Foo.overridenc(1.0e0) < 99 + @test 99 < Foo.overridenc(1.0f0) < 102 + @test 0 < Foo.overridenc(UInt(1)) < 3 # Issue #15722 @test Foo.abigfloat_f()::BigFloat == big"12.34" @@ -873,6 +883,92 @@ precompile_test_harness("code caching") do dir @test hasvalid(mi, world) # was compiled with the new method end +precompile_test_harness("invoke") do dir + InvokeModule = :Invoke0x030e7e97c2365aad + CallerModule = :Caller0x030e7e97c2365aad + write(joinpath(dir, "$InvokeModule.jl"), + """ + module $InvokeModule + export f, g, h, fnc, gnc, hnc # nc variants do not infer to a Const + # f is for testing invoke that occurs within a dependency + f(x::Real) = 0 + f(x::Int) = x < 5 ? 1 : invoke(f, Tuple{Real}, x) + fnc(x::Real) = rand()-1 + fnc(x::Int) = x < 5 ? rand()+1 : invoke(fnc, Tuple{Real}, x) + # g is for testing invoke that occurs from a dependent + g(x::Real) = 0 + g(x::Int) = 1 + gnc(x::Real) = rand()-1 + gnc(x::Int) = rand()+1 + # h will be entirely superseded by a new method (full invalidation) + h(x::Real) = 0 + h(x::Int) = x < 5 ? 1 : invoke(h, Tuple{Integer}, x) + hnc(x::Real) = rand()-1 + hnc(x::Int) = x < 5 ? rand()+1 : invoke(hnc, Tuple{Integer}, x) + end + """) + write(joinpath(dir, "$CallerModule.jl"), + """ + module $CallerModule + using $InvokeModule + # involving external modules + callf(x) = f(x) + callg(x) = x < 5 ? g(x) : invoke(g, Tuple{Real}, x) + callh(x) = h(x) + callfnc(x) = fnc(x) + callgnc(x) = x < 5 ? gnc(x) : invoke(gnc, Tuple{Real}, x) + callhnc(x) = hnc(x) + + # Purely internal + internal(x::Real) = 0 + internal(x::Int) = x < 5 ? 1 : invoke(internal, Tuple{Real}, x) + internalnc(x::Real) = rand()-1 + internalnc(x::Int) = x < 5 ? rand()+1 : invoke(internalnc, Tuple{Real}, x) + + # force precompilation + begin + Base.Experimental.@force_compile + callf(3) + callg(3) + callh(3) + callfnc(3) + callgnc(3) + callhnc(3) + internal(3) + internalnc(3) + end + + # Now that we've precompiled, invalidate with a new method that overrides the `invoke` dispatch + $InvokeModule.h(x::Integer) = -1 + $InvokeModule.hnc(x::Integer) = rand() - 20 + end + """) + Base.compilecache(Base.PkgId(string(CallerModule))) + @eval using $CallerModule + M = getfield(@__MODULE__, CallerModule) + + function get_real_method(func) # return the method func(::Real) + for m in methods(func) + m.sig.parameters[end] === Real && return m + end + error("no ::Real method found for $func") + end + + for func in (M.f, M.g, M.internal, M.fnc, M.gnc, M.internalnc) + m = get_real_method(func) + mi = m.specializations[1] + @test length(mi.backedges) == 2 + @test mi.backedges[1] === Tuple{typeof(func), Real} + @test isa(mi.backedges[2], Core.MethodInstance) + @test mi.cache.max_world == typemax(mi.cache.max_world) + end + + m = get_real_method(M.h) + @test isempty(m.specializations) + m = get_real_method(M.hnc) + @test isempty(m.specializations) +end + # test --compiled-modules=no command line option precompile_test_harness("--compiled-modules=no") do dir Time_module = :Time4b3a94a1a081a8cb @@ -1069,14 +1165,22 @@ precompile_test_harness("delete_method") do dir """ module $A_module - export apc, anopc + export apc, anopc, apcnc, anopcnc + # Infer to a const apc(::Int, ::Int) = 1 apc(::Any, ::Any) = 2 anopc(::Int, ::Int) = 1 anopc(::Any, ::Any) = 2 + # Do not infer to a const + apcnc(::Int, ::Int) = rand() - 1 + apcnc(::Any, ::Any) = rand() + 1 + + anopcnc(::Int, ::Int) = rand() - 1 + anopcnc(::Any, ::Any) = rand() + 1 + end """) write(B_file, @@ -1087,19 +1191,26 @@ precompile_test_harness("delete_method") do dir bpc(x) = apc(x, x) bnopc(x) = anopc(x, x) + bpcnc(x) = apcnc(x, x) + bnopcnc(x) = anopcnc(x, x) precompile(bpc, (Int,)) precompile(bpc, (Float64,)) + precompile(bpcnc, (Int,)) + precompile(bpcnc, (Float64,)) end """) A = Base.require(Main, A_module) - for mths in (collect(methods(A.apc)), collect(methods(A.anopc))) - Base.delete_method(mths[1]) + for mths in (collect(methods(A.apc)), collect(methods(A.anopc)), collect(methods(A.apcnc)), collect(methods(A.anopcnc))) + idx = findfirst(m -> m.sig.parameters[end] === Int, mths) + Base.delete_method(mths[idx]) end B = Base.require(Main, B_module) - @test Base.invokelatest(B.bpc, 1) == Base.invokelatest(B.bpc, 1.0) == 2 - @test Base.invokelatest(B.bnopc, 1) == Base.invokelatest(B.bnopc, 1.0) == 2 + for f in (B.bpc, B.bnopc, B.bpcnc, B.bnopcnc) + @test Base.invokelatest(f, 1) > 1 + @test Base.invokelatest(f, 1.0) > 1 + end end precompile_test_harness("Issues #19030 and #25279") do load_path From 7d5f1a81b89d16baab5d18bb073e04fedaa3bc12 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Thu, 21 Jul 2022 10:59:04 -0500 Subject: [PATCH 02/11] don't include subtype --- src/dump.c | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/dump.c b/src/dump.c index a3a04f0f6d9bb..6143f69b4920c 100644 --- a/src/dump.c +++ b/src/dump.c @@ -2336,7 +2336,7 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED // Is this still the method we'd be calling? jl_methtable_t *mt = jl_method_table_for(mi->specTypes); struct jl_typemap_assoc search = {(jl_value_t*)mi->specTypes, world, NULL, 0, ~(size_t)0}; - jl_typemap_entry_t *entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/1); + jl_typemap_entry_t *entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/0); if (entry) { jl_value_t *mworld = entry->func.value; if (jl_is_method(mworld) && mi->def.method != (jl_method_t*)mworld && jl_type_morespecific(((jl_method_t*)mworld)->sig, mi->def.method->sig)) { @@ -2350,7 +2350,7 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED j = get_next_backedge(mi->backedges, j, &invokeTypes, &caller); if (invokeTypes) { struct jl_typemap_assoc search = {invokeTypes, world, NULL, 0, ~(size_t)0}; - entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/1); + entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/0); if (entry) { jl_value_t *imworld = entry->func.value; if (jl_is_method(imworld) && mi->def.method == (jl_method_t*)imworld) { From e937235c75f4483c2e0a6d290ce4e9212a1d4928 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Thu, 21 Jul 2022 12:59:07 -0500 Subject: [PATCH 03/11] Support precompile(::Method, argtypes) This mimics an `invoke` call --- base/loading.jl | 31 +++++++++++++++++++++++++++---- src/gf.c | 23 ++++++++++++++--------- src/jl_exported_funcs.inc | 1 + src/julia_internal.h | 1 + test/precompile.jl | 8 ++++++++ 5 files changed, 51 insertions(+), 13 deletions(-) diff --git a/base/loading.jl b/base/loading.jl index f8100fc0915a2..749fec736e61a 100644 --- a/base/loading.jl +++ b/base/loading.jl @@ -2262,12 +2262,12 @@ macro __DIR__() end """ - precompile(f, args::Tuple{Vararg{Any}}) + precompile(f, argtypes::Tuple{Vararg{Any}}) -Compile the given function `f` for the argument tuple (of types) `args`, but do not execute it. +Compile the given function `f` for the argument tuple (of types) `argtypes`, but do not execute it. """ -function precompile(@nospecialize(f), @nospecialize(args::Tuple)) - precompile(Tuple{Core.Typeof(f), args...}) +function precompile(@nospecialize(f), @nospecialize(argtypes::Tuple)) + precompile(Tuple{Core.Typeof(f), argtypes...}) end const ENABLE_PRECOMPILE_WARNINGS = Ref(false) @@ -2279,6 +2279,29 @@ function precompile(@nospecialize(argt::Type)) return ret end +# Variants that work for `invoke`d calls for which the signature may not be sufficient +precompile(mi::Core.MethodInstance, world=get_world_counter()) = (ccall(:jl_compile_method_instance, Cvoid, (Any, Any, UInt), mi, C_NULL, world); return true) + +""" + precompile(m::Method, argtypes::Tuple{Vararg{Any}}) + +Precompile a specific method for the given argument types. This may be used to precompile +a different method than the one that would ordinarily be chosen by dispatch. +""" +function precompile(m::Method, @nospecialize(argtypes::Tuple)) + world = get_world_counter() + f = getglobal(m.module, m.name) + matches = _methods(f, argtypes, -1, world)::Vector + for mtch in matches + mtch = mtch::Core.MethodMatch + if mtch.method == m + mi = Core.Compiler.specialize_method(m, mtch.spec_types, mtch.sparams) + return precompile(mi) + end + end + return false +end + precompile(include_package_for_output, (PkgId, String, Vector{String}, Vector{String}, Vector{String}, typeof(_concrete_dependencies), Nothing)) precompile(include_package_for_output, (PkgId, String, Vector{String}, Vector{String}, Vector{String}, typeof(_concrete_dependencies), String)) precompile(create_expr_cache, (PkgId, String, String, typeof(_concrete_dependencies), IO, IO)) diff --git a/src/gf.c b/src/gf.c index 6b2d9362198f7..26f5dd41f538a 100644 --- a/src/gf.c +++ b/src/gf.c @@ -2296,16 +2296,9 @@ static void jl_compile_now(jl_method_instance_t *mi) } } -JL_DLLEXPORT int jl_compile_hint(jl_tupletype_t *types) +JL_DLLEXPORT void jl_compile_method_instance(jl_method_instance_t *mi, jl_tupletype_t *types, size_t world) { - size_t world = jl_atomic_load_acquire(&jl_world_counter); size_t tworld = jl_typeinf_world; - size_t min_valid = 0; - size_t max_valid = ~(size_t)0; - jl_method_instance_t *mi = jl_get_specialization1(types, world, &min_valid, &max_valid, 1); - if (mi == NULL) - return 0; - JL_GC_PROMISE_ROOTED(mi); mi->precompiled = 1; if (jl_generating_output()) { jl_compile_now(mi); @@ -2314,7 +2307,7 @@ JL_DLLEXPORT int jl_compile_hint(jl_tupletype_t *types) // additional useful methods that should be compiled //ALT: if (jl_is_datatype(types) && ((jl_datatype_t*)types)->isdispatchtuple && !jl_egal(mi->specTypes, types)) //ALT: if (jl_subtype(types, mi->specTypes)) - if (!jl_subtype(mi->specTypes, (jl_value_t*)types)) { + if (types && !jl_subtype(mi->specTypes, (jl_value_t*)types)) { jl_svec_t *tpenv2 = jl_emptysvec; jl_value_t *types2 = NULL; JL_GC_PUSH2(&tpenv2, &types2); @@ -2335,6 +2328,18 @@ JL_DLLEXPORT int jl_compile_hint(jl_tupletype_t *types) // we should generate the native code immediately in preparation for use. (void)jl_compile_method_internal(mi, world); } +} + +JL_DLLEXPORT int jl_compile_hint(jl_tupletype_t *types) +{ + size_t world = jl_atomic_load_acquire(&jl_world_counter); + size_t min_valid = 0; + size_t max_valid = ~(size_t)0; + jl_method_instance_t *mi = jl_get_specialization1(types, world, &min_valid, &max_valid, 1); + if (mi == NULL) + return 0; + JL_GC_PROMISE_ROOTED(mi); + jl_compile_method_instance(mi, types, world); return 1; } diff --git a/src/jl_exported_funcs.inc b/src/jl_exported_funcs.inc index b77acc1eb3858..dfb6becea43e1 100644 --- a/src/jl_exported_funcs.inc +++ b/src/jl_exported_funcs.inc @@ -97,6 +97,7 @@ XX(jl_close_uv) \ XX(jl_code_for_staged) \ XX(jl_compile_hint) \ + XX(jl_compile_method_instance) \ XX(jl_compress_argnames) \ XX(jl_compress_ir) \ XX(jl_compute_fieldtypes) \ diff --git a/src/julia_internal.h b/src/julia_internal.h index 8460e341e070d..8399cfb36607a 100644 --- a/src/julia_internal.h +++ b/src/julia_internal.h @@ -596,6 +596,7 @@ JL_DLLEXPORT jl_code_instance_t *jl_get_method_inferred( jl_method_instance_t *jl_get_unspecialized_from_mi(jl_method_instance_t *method JL_PROPAGATES_ROOT); jl_method_instance_t *jl_get_unspecialized(jl_method_t *def JL_PROPAGATES_ROOT); +JL_DLLEXPORT void jl_compile_method_instance(jl_method_instance_t *mi, jl_tupletype_t *types, size_t world); JL_DLLEXPORT int jl_compile_hint(jl_tupletype_t *types); jl_code_info_t *jl_code_for_interpreter(jl_method_instance_t *lam JL_PROPAGATES_ROOT); int jl_code_requires_compiler(jl_code_info_t *src); diff --git a/test/precompile.jl b/test/precompile.jl index be4cbccf1172d..02936bd1e838f 100644 --- a/test/precompile.jl +++ b/test/precompile.jl @@ -967,6 +967,14 @@ precompile_test_harness("invoke") do dir @test isempty(m.specializations) m = get_real_method(M.hnc) @test isempty(m.specializations) + + # Precompile specific methods for arbitrary arg types + invokeme(x) = 1 + invokeme(::Int) = 2 + m_any, m_int = sort(collect(methods(invokeme)); by=m->m.line) + precompile(m_any, (Int,)) + @test m_any.specializations[1].specTypes === Tuple{typeof(invokeme), Int} + @test isempty(m_int.specializations) end # test --compiled-modules=no command line option From 737efc8a18448d5441e74f9b6f0a7eb238312185 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Thu, 21 Jul 2022 15:22:04 -0500 Subject: [PATCH 04/11] fix an ambiguity --- base/loading.jl | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/base/loading.jl b/base/loading.jl index 749fec736e61a..eebadb92928f4 100644 --- a/base/loading.jl +++ b/base/loading.jl @@ -2280,7 +2280,8 @@ function precompile(@nospecialize(argt::Type)) end # Variants that work for `invoke`d calls for which the signature may not be sufficient -precompile(mi::Core.MethodInstance, world=get_world_counter()) = (ccall(:jl_compile_method_instance, Cvoid, (Any, Any, UInt), mi, C_NULL, world); return true) +precompile(mi::Core.MethodInstance, world::UInt=get_world_counter()) = + (ccall(:jl_compile_method_instance, Cvoid, (Any, Any, UInt), mi, C_NULL, world); return true) """ precompile(m::Method, argtypes::Tuple{Vararg{Any}}) From 580854594a0d725dd7a1d57662b7c2c268b78f70 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Thu, 21 Jul 2022 17:56:15 -0500 Subject: [PATCH 05/11] Avoid name lookup --- base/loading.jl | 10 +++++----- test/precompile.jl | 2 +- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/base/loading.jl b/base/loading.jl index eebadb92928f4..cdd66414c8dae 100644 --- a/base/loading.jl +++ b/base/loading.jl @@ -2284,20 +2284,20 @@ precompile(mi::Core.MethodInstance, world::UInt=get_world_counter()) = (ccall(:jl_compile_method_instance, Cvoid, (Any, Any, UInt), mi, C_NULL, world); return true) """ - precompile(m::Method, argtypes::Tuple{Vararg{Any}}) + precompile(f, argtypes::Tuple{Vararg{Any}}, m::Method) Precompile a specific method for the given argument types. This may be used to precompile -a different method than the one that would ordinarily be chosen by dispatch. +a different method than the one that would ordinarily be chosen by dispatch, for example +mimicking `invoke`. """ -function precompile(m::Method, @nospecialize(argtypes::Tuple)) +function precompile(@nospecialize(f), @nospecialize(argtypes::Tuple), m::Method) world = get_world_counter() - f = getglobal(m.module, m.name) matches = _methods(f, argtypes, -1, world)::Vector for mtch in matches mtch = mtch::Core.MethodMatch if mtch.method == m mi = Core.Compiler.specialize_method(m, mtch.spec_types, mtch.sparams) - return precompile(mi) + return precompile(mi, world) end end return false diff --git a/test/precompile.jl b/test/precompile.jl index 02936bd1e838f..7b3a3184bac08 100644 --- a/test/precompile.jl +++ b/test/precompile.jl @@ -972,7 +972,7 @@ precompile_test_harness("invoke") do dir invokeme(x) = 1 invokeme(::Int) = 2 m_any, m_int = sort(collect(methods(invokeme)); by=m->m.line) - precompile(m_any, (Int,)) + precompile(invokeme, (Int,), m_any) @test m_any.specializations[1].specTypes === Tuple{typeof(invokeme), Int} @test isempty(m_int.specializations) end From fa5c237f9d3f113a25befa9029207825040ab307 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Thu, 21 Jul 2022 18:49:28 -0500 Subject: [PATCH 06/11] Fix sparams generation --- base/loading.jl | 19 ++++++++----------- test/precompile.jl | 4 ++-- 2 files changed, 10 insertions(+), 13 deletions(-) diff --git a/base/loading.jl b/base/loading.jl index cdd66414c8dae..8c0f37b1e224b 100644 --- a/base/loading.jl +++ b/base/loading.jl @@ -2287,20 +2287,17 @@ precompile(mi::Core.MethodInstance, world::UInt=get_world_counter()) = precompile(f, argtypes::Tuple{Vararg{Any}}, m::Method) Precompile a specific method for the given argument types. This may be used to precompile -a different method than the one that would ordinarily be chosen by dispatch, for example +a different method than the one that would ordinarily be chosen by dispatch, thus mimicking `invoke`. """ function precompile(@nospecialize(f), @nospecialize(argtypes::Tuple), m::Method) - world = get_world_counter() - matches = _methods(f, argtypes, -1, world)::Vector - for mtch in matches - mtch = mtch::Core.MethodMatch - if mtch.method == m - mi = Core.Compiler.specialize_method(m, mtch.spec_types, mtch.sparams) - return precompile(mi, world) - end - end - return false + precompile(Tuple{Core.Typeof(f), argtypes...}, m) +end + +function precompile(@nospecialize(argt::Type), m::Method) + atype, sparams = ccall(:jl_type_intersection_with_env, Any, (Any, Any), argt, m.sig)::SimpleVector + mi = Core.Compiler.specialize_method(m, atype, sparams) + return precompile(mi) end precompile(include_package_for_output, (PkgId, String, Vector{String}, Vector{String}, Vector{String}, typeof(_concrete_dependencies), Nothing)) diff --git a/test/precompile.jl b/test/precompile.jl index 7b3a3184bac08..ab49c175b6795 100644 --- a/test/precompile.jl +++ b/test/precompile.jl @@ -971,8 +971,8 @@ precompile_test_harness("invoke") do dir # Precompile specific methods for arbitrary arg types invokeme(x) = 1 invokeme(::Int) = 2 - m_any, m_int = sort(collect(methods(invokeme)); by=m->m.line) - precompile(invokeme, (Int,), m_any) + m_any, m_int = sort(collect(methods(invokeme)); by=m->(m.file,m.line)) + @test precompile(invokeme, (Int,), m_any) @test m_any.specializations[1].specTypes === Tuple{typeof(invokeme), Int} @test isempty(m_int.specializations) end From 1b9535b0882d8e0e197dbe430a4f8d0d386176b2 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Tue, 2 Aug 2022 15:47:59 -0500 Subject: [PATCH 07/11] Apply suggestions from code review Co-authored-by: Jameson Nash --- base/compiler/abstractinterpretation.jl | 8 +++----- base/compiler/utilities.jl | 4 ---- 2 files changed, 3 insertions(+), 9 deletions(-) diff --git a/base/compiler/abstractinterpretation.jl b/base/compiler/abstractinterpretation.jl index 0f504b9337c3f..dbc07ea985d1a 100644 --- a/base/compiler/abstractinterpretation.jl +++ b/base/compiler/abstractinterpretation.jl @@ -802,10 +802,8 @@ function collect_const_args(argtypes::Vector{Any}) end function invoke_signature(invokesig::Vector{Any}) - unwrapconst(x) = isa(x, Const) ? x.val : x - - f, argtyps = unwrapconst(invokesig[2]), unwrapconst(invokesig[3]) - return Tuple{typeof(f), unwrap_unionall(argtyps).parameters...} + ft, argtyps = widenconst(invokesig[2]), instanceof_tfunc(widenconst(invokesig[3]))[1] + return rewrap_unionall(Tuple{typeof(f), unwrap_unionall(argtyps).parameters...}, argtyps) end function concrete_eval_call(interp::AbstractInterpreter, @@ -1638,7 +1636,7 @@ function abstract_invoke(interp::AbstractInterpreter, (; fargs, argtypes)::ArgIn ti = tienv[1]; env = tienv[2]::SimpleVector result = abstract_call_method(interp, method, ti, env, false, sv) (; rt, edge, effects) = result - edge !== nothing && add_backedge!(edge::MethodInstance, sv, argtypes) + edge !== nothing && add_backedge!(edge::MethodInstance, sv, types) match = MethodMatch(ti, env, method, argtype <: method.sig) res = nothing sig = match.spec_types diff --git a/base/compiler/utilities.jl b/base/compiler/utilities.jl index 975dc3d1a9b5d..db08ae22c4517 100644 --- a/base/compiler/utilities.jl +++ b/base/compiler/utilities.jl @@ -268,10 +268,6 @@ end const empty_backedge_iter = BackedgeIterator(Any[]) -function BackedgeIterator(mi::MethodInstance) - isdefined(mi, :backedges) || return empty_backedge_iter - return BackedgeIterator(mi.backedges) -end function iterate(iter::BackedgeIterator, i::Int=1) backedges = iter.backedges From d1923b0055bf87bfd607d289b781b150a907cc01 Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Sat, 20 Aug 2022 07:17:32 -0500 Subject: [PATCH 08/11] Address review comments --- base/compiler/abstractinterpretation.jl | 2 +- base/compiler/inferencestate.jl | 4 ++-- base/compiler/utilities.jl | 8 +++---- src/dump.c | 28 ++++++++++++------------- src/gf.c | 12 ++++++----- src/julia_internal.h | 6 +++--- src/method.c | 6 +++--- 7 files changed, 33 insertions(+), 33 deletions(-) diff --git a/base/compiler/abstractinterpretation.jl b/base/compiler/abstractinterpretation.jl index dbc07ea985d1a..cde9a4ed2f4fa 100644 --- a/base/compiler/abstractinterpretation.jl +++ b/base/compiler/abstractinterpretation.jl @@ -803,7 +803,7 @@ end function invoke_signature(invokesig::Vector{Any}) ft, argtyps = widenconst(invokesig[2]), instanceof_tfunc(widenconst(invokesig[3]))[1] - return rewrap_unionall(Tuple{typeof(f), unwrap_unionall(argtyps).parameters...}, argtyps) + return rewrap_unionall(Tuple{ft, unwrap_unionall(argtyps).parameters...}, argtyps) end function concrete_eval_call(interp::AbstractInterpreter, diff --git a/base/compiler/inferencestate.jl b/base/compiler/inferencestate.jl index 732cd86e34fcc..d5e5bb4a5c3a6 100644 --- a/base/compiler/inferencestate.jl +++ b/base/compiler/inferencestate.jl @@ -479,14 +479,14 @@ function add_cycle_backedge!(frame::InferenceState, caller::InferenceState, curr end # temporarily accumulate our edges to later add as backedges in the callee -function add_backedge!(li::MethodInstance, caller::InferenceState, invokesig::Union{Nothing,Vector{Any}}=nothing) +function add_backedge!(li::MethodInstance, caller::InferenceState, invokesig::Union{Nothing,DataType}=nothing) isa(caller.linfo.def, Method) || return # don't add backedges to toplevel exprs edges = caller.stmt_edges[caller.currpc] if edges === nothing edges = caller.stmt_edges[caller.currpc] = [] end if invokesig !== nothing - push!(edges, invoke_signature(invokesig)) + push!(edges, invokesig) end push!(edges, li) return nothing diff --git a/base/compiler/utilities.jl b/base/compiler/utilities.jl index db08ae22c4517..c01c0dffec505 100644 --- a/base/compiler/utilities.jl +++ b/base/compiler/utilities.jl @@ -228,12 +228,10 @@ is_no_constprop(method::Union{Method,CodeInfo}) = method.constprop == 0x02 ############# """ - BackedgeIterator(mi::MethodInstance) BackedgeIterator(backedges::Vector{Any}) -Return an iterator over a list of backedges, which may be extracted -from `mi`. Iteration returns `(sig, caller)` elements, which will be one of -the following: +Return an iterator over a list of backedges. Iteration returns `(sig, caller)` elements, +which will be one of the following: - `(nothing, caller::MethodInstance)`: a call made by ordinary inferrable dispatch - `(invokesig, caller::MethodInstance)`: a call made by `invoke(f, invokesig, args...)` @@ -254,7 +252,7 @@ julia> callyou(2.0) julia> mi = first(which(callme, (Any,)).specializations) MethodInstance for callme(::Float64) -julia> @eval Core.Compiler for (sig, caller) in BackedgeIterator(Main.mi) +julia> @eval Core.Compiler for (sig, caller) in BackedgeIterator(Main.mi.backedges) println(sig) println(caller) end diff --git a/src/dump.c b/src/dump.c index 6143f69b4920c..3eb9d3d57e95d 100644 --- a/src/dump.c +++ b/src/dump.c @@ -340,7 +340,7 @@ static int has_backedge_to_worklist(jl_method_instance_t *mi, htable_t *visited) size_t i = 0, n = jl_array_len(mi->backedges); jl_method_instance_t *be; while (i < n) { - i = get_next_backedge(mi->backedges, i, NULL, &be); + i = get_next_edge(mi->backedges, i, NULL, &be); if (has_backedge_to_worklist(be, visited)) { bp = ptrhash_bp(visited, mi); // re-acquire since rehashing might change the location *bp = (void*)((char*)HT_NOTFOUND + 2); // found @@ -954,7 +954,7 @@ static void jl_serialize_value_(jl_serializer_state *s, jl_value_t *v, int as_li jl_value_t *invokeTypes; jl_method_instance_t *backedge; while (i < l) { - i = get_next_backedge(backedges, i, &invokeTypes, &backedge); + i = get_next_edge(backedges, i, &invokeTypes, &backedge); if (module_in_worklist(backedge->def.method->module) || method_instance_in_queue(backedge)) { if (invokeTypes) b_edges[ins++] = invokeTypes; @@ -1176,7 +1176,7 @@ static void jl_collect_missing_backedges_to_mod(jl_methtable_t *mt) *edges = jl_alloc_vec_any(0); // To stay synchronized with the format from MethodInstances (specifically for `invoke`d calls), // we have to push a pair of values. But in this case the callee is unknown, so we leave it NULL. - push_backedge(*edges, missing_callee, NULL); + push_edge(*edges, missing_callee, NULL); } } } @@ -1190,11 +1190,11 @@ static void collect_backedges(jl_method_instance_t *callee) JL_GC_DISABLED jl_value_t *invokeTypes; jl_method_instance_t *caller; while (i < l) { - i = get_next_backedge(backedges, i, &invokeTypes, &caller); + i = get_next_edge(backedges, i, &invokeTypes, &caller); jl_array_t **edges = (jl_array_t**)ptrhash_bp(&edges_map, caller); if (*edges == HT_NOTFOUND) *edges = jl_alloc_vec_any(0); - push_backedge(*edges, invokeTypes, callee); + push_edge(*edges, invokeTypes, callee); } } } @@ -1301,7 +1301,7 @@ static void jl_collect_backedges_to(jl_method_instance_t *caller, htable_t *all_ jl_method_instance_t *c; jl_value_t *invokeTypes; while (i < l) { - i = get_next_backedge(callees, i, &invokeTypes, &c); + i = get_next_edge(callees, i, &invokeTypes, &c); register_backedge(all_callees, invokeTypes, (jl_value_t*)c); if (c && jl_is_method_instance(c)) { jl_collect_backedges_to((jl_method_instance_t*)c, all_callees); @@ -1334,7 +1334,7 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j if (module_in_worklist(caller->def.method->module) || method_instance_in_queue(caller)) { size_t i = 0, l = jl_array_len(callees); while (i < l) { - i = get_next_backedge(callees, i, &invokeTypes, &c); + i = get_next_edge(callees, i, &invokeTypes, &c); register_backedge(&all_callees, invokeTypes, (jl_value_t*)c); if (jl_is_method_instance(c)) { jl_collect_backedges_to((jl_method_instance_t*)c, &all_callees); @@ -2347,7 +2347,7 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED size_t jins = 0, j0, j = 0, nbe = jl_array_len(mi->backedges); while (j < nbe) { j0 = j; - j = get_next_backedge(mi->backedges, j, &invokeTypes, &caller); + j = get_next_edge(mi->backedges, j, &invokeTypes, &caller); if (invokeTypes) { struct jl_typemap_assoc search = {invokeTypes, world, NULL, 0, ~(size_t)0}; entry = jl_typemap_assoc_by_type(mt->defs, &search, /*offs*/0, /*subtype*/0); @@ -2418,11 +2418,11 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED milive->backedges = jl_alloc_vec_any(n); jl_gc_wb(milive, milive->backedges); while (j < n) { - j = get_next_backedge(mi->backedges, j, &invokeTypes, &be); + j = get_next_edge(mi->backedges, j, &invokeTypes, &be); belive = (jl_method_instance_t*)ptrhash_get(&uniquing_table, be); if (belive == HT_NOTFOUND) belive = be; - jlive = set_next_backedge(milive->backedges, jlive, invokeTypes, belive); + jlive = set_next_edge(milive->backedges, jlive, invokeTypes, belive); } } else { // Copy the missing backedges (this is an O(N^2) algorithm, but many methods have few MethodInstances) @@ -2430,21 +2430,21 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED jl_value_t *invokeTypes2; jl_method_instance_t *belive2; while (j < n) { - j = get_next_backedge(mi->backedges, j, &invokeTypes, &be); + j = get_next_edge(mi->backedges, j, &invokeTypes, &be); belive = (jl_method_instance_t*)ptrhash_get(&uniquing_table, be); if (belive == HT_NOTFOUND) belive = be; int found = 0; k = 0; while (k < nlive) { - k = get_next_backedge(milive->backedges, k, &invokeTypes2, &belive2); - if (belive == belive2 && invokeTypes == invokeTypes2) { + k = get_next_edge(milive->backedges, k, &invokeTypes2, &belive2); + if (belive == belive2 && jl_egal(invokeTypes, invokeTypes2)) { found = 1; break; } } if (!found) - push_backedge(milive->backedges, invokeTypes, belive); + push_edge(milive->backedges, invokeTypes, belive); } } } diff --git a/src/gf.c b/src/gf.c index 26f5dd41f538a..802b80ebc57ac 100644 --- a/src/gf.c +++ b/src/gf.c @@ -1458,7 +1458,7 @@ static void invalidate_method_instance(void (*f)(jl_code_instance_t*), jl_method size_t i = 0, l = jl_array_len(backedges); jl_method_instance_t *replaced; while (i < l) { - i = get_next_backedge(backedges, i, NULL, &replaced); + i = get_next_edge(backedges, i, NULL, &replaced); invalidate_method_instance(f, replaced, max_world, depth + 1); } JL_GC_POP(); @@ -1474,12 +1474,14 @@ void invalidate_backedges(void (*f)(jl_code_instance_t*), jl_method_instance_t * if (backedges) { // invalidate callers (if any) replaced_mi->backedges = NULL; + JL_GC_PUSH1(&backedges); size_t i = 0, l = jl_array_len(backedges); jl_method_instance_t *replaced; while (i < l) { - i = get_next_backedge(backedges, i, NULL, &replaced); + i = get_next_edge(backedges, i, NULL, &replaced); invalidate_method_instance(f, replaced, max_world, 1); } + JL_GC_POP(); } JL_UNLOCK(&replaced_mi->def.method->writelock); if (why && _jl_debug_method_invalidation) { @@ -1501,7 +1503,7 @@ JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, // lazy-init the backedges array callee->backedges = jl_alloc_vec_any(0); jl_gc_wb(callee, callee->backedges); - push_backedge(callee->backedges, invokesig, caller); + push_edge(callee->backedges, invokesig, caller); } else { size_t i = 0, l = jl_array_len(callee->backedges); @@ -1509,7 +1511,7 @@ JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, jl_value_t *invokeTypes; jl_method_instance_t *mi; while (i < l) { - i = get_next_backedge(callee->backedges, i, &invokeTypes, &mi); + i = get_next_edge(callee->backedges, i, &invokeTypes, &mi); // TODO: it would be better to canonicalize (how?) the Tuple-type so // that we don't have to call `jl_egal` if (mi == caller && ((invokesig == NULL && invokeTypes == NULL) || @@ -1519,7 +1521,7 @@ JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, } } if (!found) { - push_backedge(callee->backedges, invokesig, caller); + push_edge(callee->backedges, invokesig, caller); } } JL_UNLOCK(&callee->def.method->writelock); diff --git a/src/julia_internal.h b/src/julia_internal.h index 8399cfb36607a..ccc542ba89f09 100644 --- a/src/julia_internal.h +++ b/src/julia_internal.h @@ -605,9 +605,9 @@ JL_DLLEXPORT jl_code_info_t *jl_new_code_info_uninit(void); void jl_resolve_globals_in_ir(jl_array_t *stmts, jl_module_t *m, jl_svec_t *sparam_vals, int binding_effects); -int get_next_backedge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method_instance_t **caller) JL_NOTSAFEPOINT; -int set_next_backedge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_instance_t *caller); -void push_backedge(jl_array_t *list, jl_value_t *invokesig, jl_method_instance_t *caller); +int get_next_edge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method_instance_t **caller) JL_NOTSAFEPOINT; +int set_next_edge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_instance_t *caller); +void push_edge(jl_array_t *list, jl_value_t *invokesig, jl_method_instance_t *caller); JL_DLLEXPORT void jl_add_method_root(jl_method_t *m, jl_module_t *mod, jl_value_t* root); void jl_append_method_roots(jl_method_t *m, uint64_t modid, jl_array_t* roots); diff --git a/src/method.c b/src/method.c index ce031cc1e5dc2..e5d6771b080d6 100644 --- a/src/method.c +++ b/src/method.c @@ -791,7 +791,7 @@ JL_DLLEXPORT jl_method_t *jl_new_method_uninit(jl_module_t *module) // it will be the signature supplied in an `invoke` call. // If you don't need `invokesig`, you can set it to NULL on input. // Initialize iteration with `i = 0`. Returns `i` for the next backedge to be extracted. -int get_next_backedge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method_instance_t **caller) JL_NOTSAFEPOINT +int get_next_edge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method_instance_t **caller) JL_NOTSAFEPOINT { jl_value_t *item = jl_array_ptr_ref(list, i); if (jl_is_method_instance(item)) { @@ -811,7 +811,7 @@ int get_next_backedge(jl_array_t *list, int i, jl_value_t** invokesig, jl_method return i + 2; } -int set_next_backedge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_instance_t *caller) +int set_next_edge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_instance_t *caller) { if (invokesig) jl_array_ptr_set(list, i++, invokesig); @@ -819,7 +819,7 @@ int set_next_backedge(jl_array_t *list, int i, jl_value_t *invokesig, jl_method_ return i; } -void push_backedge(jl_array_t *list, jl_value_t *invokesig, jl_method_instance_t *caller) +void push_edge(jl_array_t *list, jl_value_t *invokesig, jl_method_instance_t *caller) { if (invokesig) jl_array_ptr_1d_push(list, invokesig); From 5aaf76d03ec6bc04133ac8e735654d9406bdc4fe Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Mon, 22 Aug 2022 13:47:20 -0500 Subject: [PATCH 09/11] Cache invokesig in backedge data Encode the `invokesig` in `ext_targets` so it gets cached to disk. --- base/compiler/inferencestate.jl | 2 +- src/dump.c | 65 +++++++++++++++++++-------------- 2 files changed, 39 insertions(+), 28 deletions(-) diff --git a/base/compiler/inferencestate.jl b/base/compiler/inferencestate.jl index d5e5bb4a5c3a6..71559da0b2355 100644 --- a/base/compiler/inferencestate.jl +++ b/base/compiler/inferencestate.jl @@ -479,7 +479,7 @@ function add_cycle_backedge!(frame::InferenceState, caller::InferenceState, curr end # temporarily accumulate our edges to later add as backedges in the callee -function add_backedge!(li::MethodInstance, caller::InferenceState, invokesig::Union{Nothing,DataType}=nothing) +function add_backedge!(li::MethodInstance, caller::InferenceState, invokesig::Union{Nothing,Type}=nothing) isa(caller.linfo.def, Method) || return # don't add backedges to toplevel exprs edges = caller.stmt_edges[caller.currpc] if edges === nothing diff --git a/src/dump.c b/src/dump.c index 3eb9d3d57e95d..21d6d038ea8e2 100644 --- a/src/dump.c +++ b/src/dump.c @@ -1312,7 +1312,7 @@ static void jl_collect_backedges_to(jl_method_instance_t *caller, htable_t *all_ // Extract `edges` and `ext_targets` from `edges_map` // This identifies internal->external edges in the call graph, pulling them out for special treatment. -static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ jl_array_t *t) +static void jl_collect_backedges(jl_array_t *edges, jl_array_t *ext_targets) { htable_t all_targets; // target => tgtindex mapping htable_t all_callees; // MIs called by worklist methods (eff. Set{MethodInstance}) @@ -1336,7 +1336,7 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j while (i < l) { i = get_next_edge(callees, i, &invokeTypes, &c); register_backedge(&all_callees, invokeTypes, (jl_value_t*)c); - if (jl_is_method_instance(c)) { + if (c && jl_is_method_instance(c)) { jl_collect_backedges_to((jl_method_instance_t*)c, &all_callees); } } @@ -1344,6 +1344,7 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j void **pc = all_callees.table; size_t j; int valid = 1; + int mode; for (j = 0; valid && j < all_callees.size; j += 2) { if (pc[j + 1] != HT_NOTFOUND) { jl_value_t *callee = (jl_value_t*)pc[j]; @@ -1352,9 +1353,12 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j jl_value_t *sig; if (jl_is_method_instance(callee)) { sig = ((jl_method_instance_t*)callee)->specTypes; + mode = 1; } else { sig = callee; + callee = (jl_value_t*)pc[j+1]; + mode = 2; } size_t min_valid = 0; size_t max_valid = ~(size_t)0; @@ -1369,9 +1373,10 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j jl_method_match_t *match = (jl_method_match_t *)jl_array_ptr_ref(matches, k); jl_array_ptr_set(matches, k, match->method); } - jl_array_ptr_1d_push(t, callee); - jl_array_ptr_1d_push(t, matches); - target = (char*)HT_NOTFOUND + jl_array_len(t) / 2; + jl_array_ptr_1d_push(ext_targets, mode == 1 ? NULL : sig); + jl_array_ptr_1d_push(ext_targets, callee); + jl_array_ptr_1d_push(ext_targets, matches); + target = (char*)HT_NOTFOUND + jl_array_len(ext_targets) / 3; ptrhash_put(&all_targets, (void*)callee, target); } jl_array_grow_end(callees, 1); @@ -1380,8 +1385,8 @@ static void jl_collect_backedges( /* edges */ jl_array_t *s, /* ext_targets */ j } htable_reset(&all_callees, 100); if (valid) { - jl_array_ptr_1d_push(s, (jl_value_t*)caller); - jl_array_ptr_1d_push(s, (jl_value_t*)callees); + jl_array_ptr_1d_push(edges, (jl_value_t*)caller); + jl_array_ptr_1d_push(edges, (jl_value_t*)callees); } } } @@ -2471,23 +2476,24 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED // verify that these edges intersect with the same methods as before static void jl_verify_edges(jl_array_t *targets, jl_array_t **pvalids) { - size_t i, l = jl_array_len(targets) / 2; + size_t i, l = jl_array_len(targets) / 3; jl_array_t *valids = jl_alloc_array_1d(jl_array_uint8_type, l); memset(jl_array_data(valids), 1, l); jl_value_t *loctag = NULL; JL_GC_PUSH1(&loctag); *pvalids = valids; for (i = 0; i < l; i++) { - jl_value_t *callee = jl_array_ptr_ref(targets, i * 2); + jl_value_t *invokesig = jl_array_ptr_ref(targets, i * 3); + jl_value_t *callee = jl_array_ptr_ref(targets, i * 3 + 1); jl_method_instance_t *callee_mi = (jl_method_instance_t*)callee; jl_value_t *sig; - if (jl_is_method_instance(callee)) { - sig = callee_mi->specTypes; + if (callee && jl_is_method_instance(callee)) { + sig = invokesig == NULL ? callee_mi->specTypes : invokesig; } else { - sig = callee; + sig = callee == NULL ? invokesig : callee; } - jl_array_t *expected = (jl_array_t*)jl_array_ptr_ref(targets, i * 2 + 1); + jl_array_t *expected = (jl_array_t*)jl_array_ptr_ref(targets, i * 3 + 2); assert(jl_is_array(expected)); int valid = 1; size_t min_valid = 0; @@ -2527,20 +2533,20 @@ static void jl_verify_edges(jl_array_t *targets, jl_array_t **pvalids) } // Restore backedges to external targets -// `targets` is [callee1, matches1, ...], the global set of non-worklist callees of worklist-owned methods. -// `list` = [caller1, targets_indexes1, ...], the list of worklist-owned methods calling external methods. -static void jl_insert_backedges(jl_array_t *list, jl_array_t *targets) +// `edges` = [caller1, targets_indexes1, ...], the list of worklist-owned methods calling external methods. +// `ext_targets` is [invokesig1, callee1, matches1, ...], the global set of non-worklist callees of worklist-owned methods. +static void jl_insert_backedges(jl_array_t *edges, jl_array_t *ext_targets) { - // map(enable, ((list[i] => targets[list[i + 1] .* 2]) for i in 1:2:length(list) if all(valids[list[i + 1]]))) - size_t i, l = jl_array_len(list); + // foreach(enable, ((edges[2i-1] => ext_targets[edges[2i] .* 3]) for i in 1:length(edges)รท2 if all(valids[edges[2i]]))) + size_t i, l = jl_array_len(edges); jl_array_t *valids = NULL; jl_value_t *loctag = NULL; JL_GC_PUSH2(&valids, &loctag); - jl_verify_edges(targets, &valids); + jl_verify_edges(ext_targets, &valids); for (i = 0; i < l; i += 2) { - jl_method_instance_t *caller = (jl_method_instance_t*)jl_array_ptr_ref(list, i); + jl_method_instance_t *caller = (jl_method_instance_t*)jl_array_ptr_ref(edges, i); assert(jl_is_method_instance(caller) && jl_is_method(caller->def.method)); - jl_array_t *idxs_array = (jl_array_t*)jl_array_ptr_ref(list, i + 1); + jl_array_t *idxs_array = (jl_array_t*)jl_array_ptr_ref(edges, i + 1); assert(jl_isa((jl_value_t*)idxs_array, jl_array_int32_type)); int32_t *idxs = (int32_t*)jl_array_data(idxs_array); int valid = 1; @@ -2553,18 +2559,20 @@ static void jl_insert_backedges(jl_array_t *list, jl_array_t *targets) // if this callee is still valid, add all the backedges for (j = 0; j < jl_array_len(idxs_array); j++) { int32_t idx = idxs[j]; - jl_value_t *callee = jl_array_ptr_ref(targets, idx * 2); - if (jl_is_method_instance(callee)) { - jl_method_instance_add_backedge((jl_method_instance_t*)callee, NULL, caller); + jl_value_t *callee = jl_array_ptr_ref(ext_targets, idx * 3 + 1); + if (callee && jl_is_method_instance(callee)) { + jl_value_t *invokesig = jl_array_ptr_ref(ext_targets, idx * 3); + jl_method_instance_add_backedge((jl_method_instance_t*)callee, invokesig, caller); } else { - jl_methtable_t *mt = jl_method_table_for(callee); + jl_value_t *sig = callee == NULL ? jl_array_ptr_ref(ext_targets, idx * 3) : callee; + jl_methtable_t *mt = jl_method_table_for(sig); // FIXME: rarely, `callee` has an unexpected `Union` signature, // see https://github.com/JuliaLang/julia/pull/43990#issuecomment-1030329344 // Fix the issue and turn this back into an `assert((jl_value_t*)mt != jl_nothing)` // This workaround exposes us to (rare) 265-violations. if ((jl_value_t*)mt != jl_nothing) - jl_method_table_add_backedge(mt, callee, (jl_value_t*)caller); + jl_method_table_add_backedge(mt, sig, (jl_value_t*)caller); } } // then enable it @@ -2812,7 +2820,10 @@ JL_DLLEXPORT int jl_save_incremental(const char *fname, jl_array_t *worklist) int en = jl_gc_enable(0); // edges map is not gc-safe jl_array_t *extext_methods = jl_alloc_vec_any(0); // [method1, simplesig1, ...], worklist-owned "extending external" methods added to functions owned by modules outside the worklist - jl_array_t *ext_targets = jl_alloc_vec_any(0); // [callee1, matches1, ...] non-worklist callees of worklist-owned methods + jl_array_t *ext_targets = jl_alloc_vec_any(0); // [invokesig1, callee1, matches1, ...] non-worklist callees of worklist-owned methods + // ordinary dispatch: invokesig=NULL, callee is MethodInstance + // `invoke` dispatch: invokesig is signature, callee is MethodInstance + // abstract call: callee is signature jl_array_t *edges = jl_alloc_vec_any(0); // [caller1, ext_targets_indexes1, ...] for worklist-owned methods calling external methods int n_ext_mis = queue_external_mis(newly_inferred); From 945bcfc672cd3f81eae7962a88793632af25a02f Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Wed, 24 Aug 2022 11:42:53 -0500 Subject: [PATCH 10/11] Fix jl_egal call --- src/dump.c | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/dump.c b/src/dump.c index 21d6d038ea8e2..a52523e56a71f 100644 --- a/src/dump.c +++ b/src/dump.c @@ -2443,7 +2443,8 @@ static void jl_insert_method_instances(jl_array_t *list) JL_GC_DISABLED k = 0; while (k < nlive) { k = get_next_edge(milive->backedges, k, &invokeTypes2, &belive2); - if (belive == belive2 && jl_egal(invokeTypes, invokeTypes2)) { + if (belive == belive2 && ((invokeTypes == NULL && invokeTypes2 == NULL) || + (invokeTypes && invokeTypes2 && jl_egal(invokeTypes, invokeTypes2)))) { found = 1; break; } From 8a9a87c75b5e75092226280c49f2316aced4966f Mon Sep 17 00:00:00 2001 From: Tim Holy Date: Wed, 24 Aug 2022 11:44:31 -0500 Subject: [PATCH 11/11] Teach jl_method_table_insert about invoke backedges --- src/gf.c | 38 +++++++++++++++++++++++++++++---- test/precompile.jl | 52 ++++++++++++++++++++++++++++++++++++++++------ 2 files changed, 80 insertions(+), 10 deletions(-) diff --git a/src/gf.c b/src/gf.c index 802b80ebc57ac..907adfea43dae 100644 --- a/src/gf.c +++ b/src/gf.c @@ -1851,11 +1851,41 @@ JL_DLLEXPORT void jl_method_table_insert(jl_methtable_t *mt, jl_method_t *method if (k != n) continue; } - jl_array_ptr_1d_push(oldmi, (jl_value_t*)mi); - invalidate_external(mi, max_world); + // Before deciding whether to invalidate `mi`, check each backedge for `invoke`s if (mi->backedges) { - invalidated = 1; - invalidate_backedges(&do_nothing_with_codeinst, mi, max_world, "jl_method_table_insert"); + jl_array_t *backedges = mi->backedges; + size_t ib = 0, insb = 0, nb = jl_array_len(backedges); + jl_value_t *invokeTypes; + jl_method_instance_t *caller; + while (ib < nb) { + ib = get_next_edge(backedges, ib, &invokeTypes, &caller); + if (!invokeTypes) { + // ordinary dispatch, invalidate + invalidate_method_instance(&do_nothing_with_codeinst, caller, max_world, 1); + invalidated = 1; + } else { + // invoke-dispatch, check invokeTypes for validity + struct jl_typemap_assoc search = {invokeTypes, method->primary_world, NULL, 0, ~(size_t)0}; + oldentry = jl_typemap_assoc_by_type(jl_atomic_load_relaxed(&mt->defs), &search, /*offs*/0, /*subtype*/0); + assert(oldentry); + if (oldentry->func.method == mi->def.method) { + jl_array_ptr_set(backedges, insb++, invokeTypes); + jl_array_ptr_set(backedges, insb++, caller); + continue; + } + invalidate_method_instance(&do_nothing_with_codeinst, caller, max_world, 1); + invalidated = 1; + } + } + jl_array_del_end(backedges, nb - insb); + } + if (!mi->backedges || jl_array_len(mi->backedges) == 0) { + jl_array_ptr_1d_push(oldmi, (jl_value_t*)mi); + invalidate_external(mi, max_world); + if (mi->backedges) { + invalidated = 1; + invalidate_backedges(&do_nothing_with_codeinst, mi, max_world, "jl_method_table_insert"); + } } } } diff --git a/test/precompile.jl b/test/precompile.jl index ab49c175b6795..21c8a8a2468dc 100644 --- a/test/precompile.jl +++ b/test/precompile.jl @@ -889,7 +889,7 @@ precompile_test_harness("invoke") do dir write(joinpath(dir, "$InvokeModule.jl"), """ module $InvokeModule - export f, g, h, fnc, gnc, hnc # nc variants do not infer to a Const + export f, g, h, q, fnc, gnc, hnc, qnc # nc variants do not infer to a Const # f is for testing invoke that occurs within a dependency f(x::Real) = 0 f(x::Int) = x < 5 ? 1 : invoke(f, Tuple{Real}, x) @@ -905,6 +905,9 @@ precompile_test_harness("invoke") do dir h(x::Int) = x < 5 ? 1 : invoke(h, Tuple{Integer}, x) hnc(x::Real) = rand()-1 hnc(x::Int) = x < 5 ? rand()+1 : invoke(hnc, Tuple{Integer}, x) + # q will have some callers invalidated + q(x::Integer) = 0 + qnc(x::Integer) = rand()-1 end """) write(joinpath(dir, "$CallerModule.jl"), @@ -915,9 +918,13 @@ precompile_test_harness("invoke") do dir callf(x) = f(x) callg(x) = x < 5 ? g(x) : invoke(g, Tuple{Real}, x) callh(x) = h(x) + callq(x) = q(x) + callqi(x) = invoke(q, Tuple{Integer}, x) callfnc(x) = fnc(x) callgnc(x) = x < 5 ? gnc(x) : invoke(gnc, Tuple{Real}, x) callhnc(x) = hnc(x) + callqnc(x) = qnc(x) + callqnci(x) = invoke(qnc, Tuple{Integer}, x) # Purely internal internal(x::Real) = 0 @@ -931,9 +938,13 @@ precompile_test_harness("invoke") do dir callf(3) callg(3) callh(3) + callq(3) + callqi(3) callfnc(3) callgnc(3) callhnc(3) + callqnc(3) + callqnci(3) internal(3) internalnc(3) end @@ -941,32 +952,61 @@ precompile_test_harness("invoke") do dir # Now that we've precompiled, invalidate with a new method that overrides the `invoke` dispatch $InvokeModule.h(x::Integer) = -1 $InvokeModule.hnc(x::Integer) = rand() - 20 + # ...and for q, override with a more specialized method that should leave only the invoked version still valid + $InvokeModule.q(x::Int) = -1 + $InvokeModule.qnc(x::Int) = rand()+1 end """) Base.compilecache(Base.PkgId(string(CallerModule))) @eval using $CallerModule M = getfield(@__MODULE__, CallerModule) - function get_real_method(func) # return the method func(::Real) + function get_method_for_type(func, @nospecialize(T)) # return the method func(::T) for m in methods(func) - m.sig.parameters[end] === Real && return m + m.sig.parameters[end] === T && return m end error("no ::Real method found for $func") end + function nvalid(mi::Core.MethodInstance) + isdefined(mi, :cache) || return 0 + ci = mi.cache + n = Int(ci.max_world == typemax(UInt)) + while isdefined(ci, :next) + ci = ci.next + n += ci.max_world == typemax(UInt) + end + return n + end for func in (M.f, M.g, M.internal, M.fnc, M.gnc, M.internalnc) - m = get_real_method(func) + m = get_method_for_type(func, Real) mi = m.specializations[1] @test length(mi.backedges) == 2 @test mi.backedges[1] === Tuple{typeof(func), Real} @test isa(mi.backedges[2], Core.MethodInstance) @test mi.cache.max_world == typemax(mi.cache.max_world) end + for func in (M.q, M.qnc) + m = get_method_for_type(func, Integer) + mi = m.specializations[1] + @test length(mi.backedges) == 2 + @test mi.backedges[1] === Tuple{typeof(func), Integer} + @test isa(mi.backedges[2], Core.MethodInstance) + @test mi.cache.max_world == typemax(mi.cache.max_world) + end - m = get_real_method(M.h) + m = get_method_for_type(M.h, Real) @test isempty(m.specializations) - m = get_real_method(M.hnc) + m = get_method_for_type(M.hnc, Real) @test isempty(m.specializations) + m = only(methods(M.callq)) + @test isempty(m.specializations) || nvalid(m.specializations[1]) == 0 + m = only(methods(M.callqnc)) + @test isempty(m.specializations) || nvalid(m.specializations[1]) == 0 + m = only(methods(M.callqi)) + @test m.specializations[1].specTypes == Tuple{typeof(M.callqi), Int} + m = only(methods(M.callqnci)) + @test m.specializations[1].specTypes == Tuple{typeof(M.callqnci), Int} # Precompile specific methods for arbitrary arg types invokeme(x) = 1