Skip to content

Commit 4525784

Browse files
committed
Use original computed edges during serialization instead of trying to guess them
1 parent d0f9e68 commit 4525784

8 files changed

+305
-439
lines changed

base/compiler/abstractinterpretation.jl

-3
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,6 @@ function abstract_call_gf_by_type(interp::AbstractInterpreter, @nospecialize(f),
5454
update_valid_age!(sv, valid_worlds)
5555
napplicable = length(applicable)
5656
rettype = exctype = Bottom
57-
edges = MethodInstance[]
5857
conditionals = nothing # keeps refinement information of call argument types when the return type is boolean
5958
seen = 0 # number of signatures actually inferred
6059
const_results = nothing # or const_results::Vector{Union{Nothing,ConstResult}} if any const results are available
@@ -110,7 +109,6 @@ function abstract_call_gf_by_type(interp::AbstractInterpreter, @nospecialize(f),
110109
end
111110
const_results[i] = const_result
112111
end
113-
edge === nothing || push!(edges, edge)
114112
this_rt = this_rt ₚ rt
115113
this_exct = this_exct ₚ exct
116114
if bail_out_call(interp, this_rt, sv)
@@ -167,7 +165,6 @@ function abstract_call_gf_by_type(interp::AbstractInterpreter, @nospecialize(f),
167165
end
168166
const_results[i] = const_result
169167
end
170-
edge === nothing || push!(edges, edge)
171168
end
172169
@assert !(this_conditional isa Conditional || this_rt isa MustAlias) "invalid lattice element returned from inter-procedural context"
173170
seen += 1

base/compiler/typeinfer.jl

+28-14
Original file line numberDiff line numberDiff line change
@@ -649,8 +649,8 @@ end
649649

650650
add_edges!(edges::Vector{Any}, info::MethodResultPure) = add_edges!(edges, info.info)
651651
add_edges!(edges::Vector{Any}, info::ConstCallInfo) = add_edges!(edges, info.call)
652-
add_edges!(edges::Vector{Any}, info::OpaqueClosureCreateInfo) = nothing # merely creating the object does not imply edges
653-
add_edges!(edges::Vector{Any}, info::OpaqueClosureCallInfo) = nothing # TODO: inference appears to have always mis-accounted for these backedges
652+
add_edges!(edges::Vector{Any}, info::OpaqueClosureCreateInfo) = add_edges!(edges, info.unspec.info) # merely creating the object implies edges for OC, unlike normal objects, since calling them doesn't normally have edges in contrast
653+
add_edges!(edges::Vector{Any}, info::OpaqueClosureCallInfo) = add_one_edge!(edges, specialize_method(info.match))
654654
add_edges!(edges::Vector{Any}, info::ReturnTypeCallInfo) = add_edges!(edges, info.info)
655655
function add_edges!(edges::Vector{Any}, info::ApplyCallInfo)
656656
add_edges!(edges, info.call)
@@ -668,15 +668,6 @@ add_edges!(edges::Vector{Any}, info::FinalizerInfo) = nothing # merely allocatin
668668
add_edges!(edges::Vector{Any}, info::NoCallInfo) = nothing
669669
function add_edges!(edges::Vector{Any}, info::MethodMatchInfo)
670670
matches = info.results.matches
671-
if length(matches) != 1
672-
# TODO: add check for whether this info already exists in the edges
673-
push!(edges, length(matches))
674-
push!(edges, info.atype)
675-
end
676-
for m in matches
677-
mi = specialize_method(m)
678-
length(matches) == 1 ? add_one_edge!(edges, mi) : push!(edges, mi)
679-
end
680671
if isempty(matches) || !(matches[end]::Core.MethodMatch).fully_covers
681672
# add legacy-style missing backedge info also
682673
exists = false
@@ -691,6 +682,29 @@ function add_edges!(edges::Vector{Any}, info::MethodMatchInfo)
691682
push!(edges, info.atype)
692683
end
693684
end
685+
if length(matches) == 1
686+
# try the optimized format for the representation, if possible and applicable
687+
# if this doesn't succeed, the backedge will be less precise,
688+
# but the forward edge will maintain the precision
689+
m = matches[1]::Core.MethodMatch
690+
mi = specialize_method(m)
691+
if mi.specTypes === m.spec_types
692+
add_one_edge!(edges, mi)
693+
return
694+
end
695+
end
696+
# add check for whether this lookup already existed in the edges list
697+
for i in 1:length(edges)
698+
if edges[i] === length(matches) && edges[i + 1] == info.atype
699+
return
700+
end
701+
end
702+
push!(edges, length(matches))
703+
push!(edges, info.atype)
704+
for m in matches
705+
mi = specialize_method(m::Core.MethodMatch)
706+
push!(edges, mi)
707+
end
694708
nothing
695709
end
696710
add_edges!(edges::Vector{Any}, info::InvokeCallInfo) = add_invoke_edge!(edges, info.atype, specialize_method(info.match))
@@ -716,14 +730,14 @@ function add_one_edge!(edges::Vector{Any}, mi::MethodInstance)
716730
nothing
717731
end
718732

719-
720733
function compute_edges!(sv::InferenceState)
721734
edges = sv.edges
722735
for i in 1:length(sv.stmt_info)
723736
info = sv.stmt_info[i]
724737
#rt = sv.ssavaluetypes[i]
725-
#effects = EFFECTS_TOTAL # sv.stmt_effects[i]
726-
#if rt === Any && effects === Effects()
738+
#et = sv.exectiontypes[i]
739+
#effects = EFFECTS_TOTAL # TODO: sv.stmt_effects[i]
740+
#if rt === Any && et === Any && effects === Effects()
727741
# continue
728742
#end
729743
add_edges!(edges, info)

src/gf.c

+4-1
Original file line numberDiff line numberDiff line change
@@ -1821,9 +1821,12 @@ static void invalidate_backedges(jl_method_instance_t *replaced_mi, size_t max_w
18211821
// add a backedge from callee to caller
18221822
JL_DLLEXPORT void jl_method_instance_add_backedge(jl_method_instance_t *callee, jl_value_t *invokesig, jl_method_instance_t *caller)
18231823
{
1824-
JL_LOCK(&callee->def.method->writelock);
18251824
if (invokesig == jl_nothing)
18261825
invokesig = NULL; // julia uses `nothing` but C uses NULL (#undef)
1826+
assert(jl_is_method_instance(callee));
1827+
assert(jl_is_method_instance(caller));
1828+
assert(invokesig == NULL || jl_is_type(invokesig));
1829+
JL_LOCK(&callee->def.method->writelock);
18271830
int found = 0;
18281831
// TODO: use jl_cache_type_(invokesig) like cache_method does to save memory
18291832
if (!callee->backedges) {

src/staticdata.c

+28-40
Original file line numberDiff line numberDiff line change
@@ -2617,25 +2617,21 @@ JL_DLLEXPORT jl_value_t *jl_as_global_root(jl_value_t *val, int insert)
26172617
}
26182618

26192619
static void jl_prepare_serialization_data(jl_array_t *mod_array, jl_array_t *newly_inferred, uint64_t worklist_key,
2620-
/* outputs */ jl_array_t **extext_methods, jl_array_t **new_ext_cis,
2621-
jl_array_t **method_roots_list, jl_array_t **ext_targets, jl_array_t **edges)
2620+
/* outputs */ jl_array_t **extext_methods JL_REQUIRE_ROOTED_SLOT,
2621+
jl_array_t **new_ext_cis JL_REQUIRE_ROOTED_SLOT,
2622+
jl_array_t **method_roots_list JL_REQUIRE_ROOTED_SLOT,
2623+
jl_array_t **edges JL_REQUIRE_ROOTED_SLOT)
26222624
{
26232625
// extext_methods: [method1, ...], worklist-owned "extending external" methods added to functions owned by modules outside the worklist
2624-
// ext_targets: [invokesig1, callee1, matches1, ...] non-worklist callees of worklist-owned methods
2625-
// ordinary dispatch: invokesig=NULL, callee is MethodInstance
2626-
// `invoke` dispatch: invokesig is signature, callee is MethodInstance
2627-
// abstract call: callee is signature
2628-
// edges: [caller1, ext_targets_indexes1, ...] for worklist-owned methods calling external methods
2629-
assert(edges_map == NULL);
2626+
// edges: [caller1, ext_targets, ...] for worklist-owned methods calling external methods
26302627

26312628
// Save the inferred code from newly inferred, external methods
26322629
*new_ext_cis = queue_external_cis(newly_inferred);
26332630

26342631
// Collect method extensions and edges data
2635-
JL_GC_PUSH1(&edges_map);
2636-
if (edges)
2637-
edges_map = jl_alloc_memory_any(0);
26382632
*extext_methods = jl_alloc_vec_any(0);
2633+
internal_methods = jl_alloc_vec_any(0);
2634+
JL_GC_PUSH1(&internal_methods);
26392635
jl_collect_methtable_from_mod(jl_type_type_mt, *extext_methods);
26402636
jl_collect_methtable_from_mod(jl_nonfunction_mt, *extext_methods);
26412637
size_t i, len = jl_array_len(mod_array);
@@ -2648,18 +2644,15 @@ static void jl_prepare_serialization_data(jl_array_t *mod_array, jl_array_t *new
26482644

26492645
if (edges) {
26502646
size_t world = jl_atomic_load_acquire(&jl_world_counter);
2651-
jl_collect_missing_backedges(jl_type_type_mt);
2652-
jl_collect_missing_backedges(jl_nonfunction_mt);
2653-
// jl_collect_extext_methods_from_mod and jl_collect_missing_backedges also accumulate data in callers_with_edges.
2654-
// Process this to extract `edges` and `ext_targets`.
2655-
*ext_targets = jl_alloc_vec_any(0);
2656-
*edges = jl_alloc_vec_any(0);
2647+
// jl_collect_extext_methods_from_mod accumulate data in callers_with_edges.
2648+
// Process this to extract `new_ext_cis` and `edges`
26572649
*method_roots_list = jl_alloc_vec_any(0);
26582650
// Collect the new method roots for external specializations
26592651
jl_collect_new_roots(&relocatable_ext_cis, *method_roots_list, *new_ext_cis, worklist_key);
2660-
jl_collect_edges(*edges, *ext_targets, *new_ext_cis, world);
2652+
*edges = jl_alloc_vec_any(0);
2653+
jl_collect_edges(*edges, *new_ext_cis, world);
26612654
}
2662-
assert(edges_map == NULL); // jl_collect_edges clears this when done
2655+
internal_methods = NULL;
26632656

26642657
JL_GC_POP();
26652658
}
@@ -2668,7 +2661,7 @@ static void jl_prepare_serialization_data(jl_array_t *mod_array, jl_array_t *new
26682661
static void jl_save_system_image_to_stream(ios_t *f, jl_array_t *mod_array,
26692662
jl_array_t *worklist, jl_array_t *extext_methods,
26702663
jl_array_t *new_ext_cis, jl_array_t *method_roots_list,
2671-
jl_array_t *ext_targets, jl_array_t *edges) JL_GC_DISABLED
2664+
jl_array_t *edges) JL_GC_DISABLED
26722665
{
26732666
htable_new(&field_replace, 0);
26742667
// strip metadata and IR when requested
@@ -2771,7 +2764,6 @@ static void jl_save_system_image_to_stream(ios_t *f, jl_array_t *mod_array,
27712764
}
27722765
// step 1.1: as needed, serialize the data needed for insertion into the running system
27732766
if (extext_methods) {
2774-
assert(ext_targets);
27752767
assert(edges);
27762768
// Queue method extensions
27772769
jl_queue_for_serialization(&s, extext_methods);
@@ -2780,7 +2772,6 @@ static void jl_save_system_image_to_stream(ios_t *f, jl_array_t *mod_array,
27802772
// Queue the new roots
27812773
jl_queue_for_serialization(&s, method_roots_list);
27822774
// Queue the edges
2783-
jl_queue_for_serialization(&s, ext_targets);
27842775
jl_queue_for_serialization(&s, edges);
27852776
}
27862777
jl_serialize_reachable(&s);
@@ -2934,7 +2925,6 @@ static void jl_save_system_image_to_stream(ios_t *f, jl_array_t *mod_array,
29342925
jl_write_value(&s, extext_methods);
29352926
jl_write_value(&s, new_ext_cis);
29362927
jl_write_value(&s, method_roots_list);
2937-
jl_write_value(&s, ext_targets);
29382928
jl_write_value(&s, edges);
29392929
}
29402930
write_uint32(f, jl_array_len(s.link_ids_gctags));
@@ -3019,18 +3009,18 @@ JL_DLLEXPORT void jl_create_system_image(void **_native_data, jl_array_t *workli
30193009
}
30203010

30213011
jl_array_t *mod_array = NULL, *extext_methods = NULL, *new_ext_cis = NULL;
3022-
jl_array_t *method_roots_list = NULL, *ext_targets = NULL, *edges = NULL;
3012+
jl_array_t *method_roots_list = NULL, *edges = NULL;
30233013
int64_t checksumpos = 0;
30243014
int64_t checksumpos_ff = 0;
30253015
int64_t datastartpos = 0;
3026-
JL_GC_PUSH6(&mod_array, &extext_methods, &new_ext_cis, &method_roots_list, &ext_targets, &edges);
3016+
JL_GC_PUSH5(&mod_array, &extext_methods, &new_ext_cis, &method_roots_list, &edges);
30273017

30283018
if (worklist) {
30293019
mod_array = jl_get_loaded_modules(); // __toplevel__ modules loaded in this session (from Base.loaded_modules_array)
30303020
// Generate _native_data`
30313021
if (_native_data != NULL) {
30323022
jl_prepare_serialization_data(mod_array, newly_inferred, jl_worklist_key(worklist),
3033-
&extext_methods, &new_ext_cis, NULL, NULL, NULL);
3023+
&extext_methods, &new_ext_cis, NULL, NULL);
30343024
jl_precompile_toplevel_module = (jl_module_t*)jl_array_ptr_ref(worklist, jl_array_len(worklist)-1);
30353025
*_native_data = jl_precompile_worklist(worklist, extext_methods, new_ext_cis);
30363026
jl_precompile_toplevel_module = NULL;
@@ -3059,7 +3049,7 @@ JL_DLLEXPORT void jl_create_system_image(void **_native_data, jl_array_t *workli
30593049
if (worklist) {
30603050
htable_new(&relocatable_ext_cis, 0);
30613051
jl_prepare_serialization_data(mod_array, newly_inferred, jl_worklist_key(worklist),
3062-
&extext_methods, &new_ext_cis, &method_roots_list, &ext_targets, &edges);
3052+
&extext_methods, &new_ext_cis, &method_roots_list, &edges);
30633053
if (!emit_split) {
30643054
write_int32(f, 0); // No clone_targets
30653055
write_padding(f, LLT_ALIGN(ios_pos(f), JL_CACHE_BYTE_ALIGNMENT) - ios_pos(f));
@@ -3071,7 +3061,7 @@ JL_DLLEXPORT void jl_create_system_image(void **_native_data, jl_array_t *workli
30713061
}
30723062
if (_native_data != NULL)
30733063
native_functions = *_native_data;
3074-
jl_save_system_image_to_stream(ff, mod_array, worklist, extext_methods, new_ext_cis, method_roots_list, ext_targets, edges);
3064+
jl_save_system_image_to_stream(ff, mod_array, worklist, extext_methods, new_ext_cis, method_roots_list, edges);
30753065
if (_native_data != NULL)
30763066
native_functions = NULL;
30773067
if (worklist)
@@ -3161,7 +3151,7 @@ static void jl_restore_system_image_from_stream_(ios_t *f, jl_image_t *image, jl
31613151
/* outputs */ jl_array_t **restored, jl_array_t **init_order,
31623152
jl_array_t **extext_methods, jl_array_t **internal_methods,
31633153
jl_array_t **new_ext_cis, jl_array_t **method_roots_list,
3164-
jl_array_t **ext_targets, jl_array_t **edges,
3154+
jl_array_t **edges,
31653155
char **base, arraylist_t *ccallable_list, pkgcachesizes *cachesizes) JL_GC_DISABLED
31663156
{
31673157
jl_task_t *ct = jl_current_task;
@@ -3232,7 +3222,7 @@ static void jl_restore_system_image_from_stream_(ios_t *f, jl_image_t *image, jl
32323222
assert(!ios_eof(f));
32333223
s.s = f;
32343224
uintptr_t offset_restored = 0, offset_init_order = 0, offset_extext_methods = 0, offset_new_ext_cis = 0, offset_method_roots_list = 0;
3235-
uintptr_t offset_ext_targets = 0, offset_edges = 0;
3225+
uintptr_t offset_edges = 0;
32363226
if (!s.incremental) {
32373227
size_t i;
32383228
for (i = 0; tags[i] != NULL; i++) {
@@ -3265,7 +3255,6 @@ static void jl_restore_system_image_from_stream_(ios_t *f, jl_image_t *image, jl
32653255
offset_extext_methods = jl_read_offset(&s);
32663256
offset_new_ext_cis = jl_read_offset(&s);
32673257
offset_method_roots_list = jl_read_offset(&s);
3268-
offset_ext_targets = jl_read_offset(&s);
32693258
offset_edges = jl_read_offset(&s);
32703259
}
32713260
s.buildid_depmods_idxs = depmod_to_imageidx(depmods);
@@ -3292,13 +3281,12 @@ static void jl_restore_system_image_from_stream_(ios_t *f, jl_image_t *image, jl
32923281
uint32_t external_fns_begin = read_uint32(f);
32933282
jl_read_arraylist(s.s, ccallable_list ? ccallable_list : &s.ccallable_list);
32943283
if (s.incremental) {
3295-
assert(restored && init_order && extext_methods && internal_methods && new_ext_cis && method_roots_list && ext_targets && edges);
3284+
assert(restored && init_order && extext_methods && internal_methods && new_ext_cis && method_roots_list && edges);
32963285
*restored = (jl_array_t*)jl_delayed_reloc(&s, offset_restored);
32973286
*init_order = (jl_array_t*)jl_delayed_reloc(&s, offset_init_order);
32983287
*extext_methods = (jl_array_t*)jl_delayed_reloc(&s, offset_extext_methods);
32993288
*new_ext_cis = (jl_array_t*)jl_delayed_reloc(&s, offset_new_ext_cis);
33003289
*method_roots_list = (jl_array_t*)jl_delayed_reloc(&s, offset_method_roots_list);
3301-
*ext_targets = (jl_array_t*)jl_delayed_reloc(&s, offset_ext_targets);
33023290
*edges = (jl_array_t*)jl_delayed_reloc(&s, offset_edges);
33033291
*internal_methods = jl_alloc_vec_any(0);
33043292
}
@@ -3736,9 +3724,9 @@ static jl_value_t *jl_restore_package_image_from_stream(void* pkgimage_handle, i
37363724
arraylist_t ccallable_list;
37373725

37383726
jl_value_t *restored = NULL;
3739-
jl_array_t *init_order = NULL, *extext_methods = NULL, *internal_methods = NULL, *new_ext_cis = NULL, *method_roots_list = NULL, *ext_targets = NULL, *edges = NULL;
3727+
jl_array_t *init_order = NULL, *extext_methods = NULL, *internal_methods = NULL, *new_ext_cis = NULL, *method_roots_list = NULL, *edges = NULL;
37403728
jl_svec_t *cachesizes_sv = NULL;
3741-
JL_GC_PUSH9(&restored, &init_order, &extext_methods, &internal_methods, &new_ext_cis, &method_roots_list, &ext_targets, &edges, &cachesizes_sv);
3729+
JL_GC_PUSH8(&restored, &init_order, &extext_methods, &internal_methods, &new_ext_cis, &method_roots_list, &edges, &cachesizes_sv);
37423730

37433731
{ // make a permanent in-memory copy of f (excluding the header)
37443732
ios_bufmode(f, bm_none);
@@ -3763,7 +3751,7 @@ static jl_value_t *jl_restore_package_image_from_stream(void* pkgimage_handle, i
37633751
ios_static_buffer(f, sysimg, len);
37643752
pkgcachesizes cachesizes;
37653753
jl_restore_system_image_from_stream_(f, image, depmods, checksum, (jl_array_t**)&restored, &init_order, &extext_methods, &internal_methods, &new_ext_cis, &method_roots_list,
3766-
&ext_targets, &edges, &base, &ccallable_list, &cachesizes);
3754+
&edges, &base, &ccallable_list, &cachesizes);
37673755
JL_SIGATOMIC_END();
37683756

37693757
// No special processing of `new_ext_cis` is required because recaching handled it
@@ -3777,7 +3765,7 @@ static jl_value_t *jl_restore_package_image_from_stream(void* pkgimage_handle, i
37773765
// allow users to start running in this updated world
37783766
jl_atomic_store_release(&jl_world_counter, world);
37793767
// but one of those immediate users is going to be our cache updates
3780-
jl_insert_backedges((jl_array_t*)edges, (jl_array_t*)ext_targets, (jl_array_t*)new_ext_cis, world); // restore external backedges (needs to be last)
3768+
jl_insert_backedges((jl_array_t*)edges, (jl_array_t*)new_ext_cis, world); // restore external backedges (needs to be last)
37813769
// now permit more methods to be added again
37823770
JL_UNLOCK(&world_counter_lock);
37833771
// reinit ccallables
@@ -3793,9 +3781,9 @@ static jl_value_t *jl_restore_package_image_from_stream(void* pkgimage_handle, i
37933781
jl_svecset(cachesizes_sv, 4, jl_box_long(cachesizes.reloclist));
37943782
jl_svecset(cachesizes_sv, 5, jl_box_long(cachesizes.gvarlist));
37953783
jl_svecset(cachesizes_sv, 6, jl_box_long(cachesizes.fptrlist));
3796-
restored = (jl_value_t*)jl_svec(8, restored, init_order, extext_methods,
3784+
restored = (jl_value_t*)jl_svec(7, restored, init_order, extext_methods,
37973785
new_ext_cis ? (jl_value_t*)new_ext_cis : jl_nothing,
3798-
method_roots_list, ext_targets, edges, cachesizes_sv);
3786+
method_roots_list, edges, cachesizes_sv);
37993787
}
38003788
else {
38013789
restored = (jl_value_t*)jl_svec(2, restored, init_order);
@@ -3810,7 +3798,7 @@ static jl_value_t *jl_restore_package_image_from_stream(void* pkgimage_handle, i
38103798
static void jl_restore_system_image_from_stream(ios_t *f, jl_image_t *image, uint32_t checksum)
38113799
{
38123800
JL_TIMING(LOAD_IMAGE, LOAD_Sysimg);
3813-
jl_restore_system_image_from_stream_(f, image, NULL, checksum | ((uint64_t)0xfdfcfbfa << 32), NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
3801+
jl_restore_system_image_from_stream_(f, image, NULL, checksum | ((uint64_t)0xfdfcfbfa << 32), NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL);
38143802
}
38153803

38163804
JL_DLLEXPORT jl_value_t *jl_restore_incremental_from_buf(void* pkgimage_handle, const char *buf, jl_image_t *image, size_t sz, jl_array_t *depmods, int completeinfo, const char *pkgname, int needs_permalloc)

0 commit comments

Comments
 (0)