Skip to content

Commit feaf59f

Browse files
committed
make Dict safe to use from finalizers
we may want to start indicating which APIs are safe to call from finalizers which generally will require disabling finalizers in most of the functions of that API to facilitate that, expose control from userspace of the finalizer-disable mechanism already used by codegen (modeled similarly to gc-disable) to make both easier to use, automatically restore them when an error is thrown fix #14445
1 parent 9b5fa5d commit feaf59f

9 files changed

+116
-64
lines changed

base/base.jl

+2-1
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,8 @@ end
8080
finalize(o::ANY) = ccall(:jl_finalize, Void, (Any,), o)
8181

8282
gc(full::Bool=true) = ccall(:jl_gc_collect, Void, (Cint,), full)
83-
gc_enable(on::Bool) = ccall(:jl_gc_enable, Cint, (Cint,), on)!=0
83+
gc_enable(on::Bool) = ccall(:jl_gc_enable, Cint, (Cint,), on) != 0
84+
gc_finalizers_enable(on::Bool) = ccall(:jl_finalizers_enable, Cint, (Cint,), on) != 0
8485

8586
bytestring(str::String) = str
8687

base/coreimg.jl

+1
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,7 @@ end
6464
include("reduce.jl")
6565

6666
## core structures
67+
gc_finalizers_enable(on::Bool) = true # Core.Inference doesn't use finalizers
6768
include("intset.jl")
6869
include("dict.jl")
6970
include("iterator.jl")

base/dict.jl

+71-43
Original file line numberDiff line numberDiff line change
@@ -537,6 +537,8 @@ function rehash!{K,V}(h::Dict{K,V}, newsz = length(h.keys))
537537
count = 0
538538
maxprobe = h.maxprobe
539539

540+
# make sure the following is atomic wrt finalizers
541+
prev_finalizers = gc_finalizers_enable(false)
540542
for i = 1:sz
541543
if olds[i] == 0x1
542544
k = oldk[i]
@@ -551,11 +553,6 @@ function rehash!{K,V}(h::Dict{K,V}, newsz = length(h.keys))
551553
keys[index] = k
552554
vals[index] = v
553555
count += 1
554-
555-
if h.age != age0
556-
# if `h` is changed by a finalizer, retry
557-
return rehash!(h, newsz)
558-
end
559556
end
560557
end
561558

@@ -565,7 +562,8 @@ function rehash!{K,V}(h::Dict{K,V}, newsz = length(h.keys))
565562
h.count = count
566563
h.ndel = 0
567564
h.maxprobe = maxprobe
568-
assert(h.age == age0)
565+
@assert h.age == age0
566+
gc_finalizers_enable(prev_finalizers)
569567

570568
return h
571569
end
@@ -584,6 +582,7 @@ function sizehint!(d::Dict, newsz)
584582
end
585583

586584
function empty!{K,V}(h::Dict{K,V})
585+
prev_finalizers = gc_finalizers_enable(false)
587586
fill!(h.slots, 0x0)
588587
sz = length(h.slots)
589588
empty!(h.keys)
@@ -594,6 +593,7 @@ function empty!{K,V}(h::Dict{K,V})
594593
h.count = 0
595594
h.age += 1
596595
h.idxfloor = 1
596+
gc_finalizers_enable(prev_finalizers)
597597
return h
598598
end
599599

@@ -698,6 +698,7 @@ function setindex!{K,V}(h::Dict{K,V}, v0, key0)
698698
end
699699
v = convert(V, v0)
700700

701+
prev_finalizers = gc_finalizers_enable(false)
701702
index = ht_keyindex2(h, key)
702703

703704
if index > 0
@@ -707,37 +708,31 @@ function setindex!{K,V}(h::Dict{K,V}, v0, key0)
707708
else
708709
_setindex!(h, v, key, -index)
709710
end
711+
gc_finalizers_enable(prev_finalizers)
710712

711713
return h
712714
end
713715

714-
function get!{K,V}(h::Dict{K,V}, key0, default)
715-
key = convert(K,key0)
716-
if !isequal(key,key0)
717-
throw(ArgumentError("$key0 is not a valid key for type $K"))
718-
end
719-
720-
index = ht_keyindex2(h, key)
721-
722-
index > 0 && return h.vals[index]
723-
724-
v = convert(V, default)
725-
_setindex!(h, v, key, -index)
726-
return v
727-
end
728-
716+
get!{K,V}(h::Dict{K,V}, key0, default) = get!(()->default, h, key0)
729717
function get!{K,V}(default::Callable, h::Dict{K,V}, key0)
730718
key = convert(K,key0)
731719
if !isequal(key,key0)
732720
throw(ArgumentError("$key0 is not a valid key for type $K"))
733721
end
734722

723+
prev_finalizers = gc_finalizers_enable(false)
735724
index = ht_keyindex2(h, key)
736725

737-
index > 0 && return h.vals[index]
726+
if index > 0
727+
val = h.vals[index]
728+
gc_finalizers_enable(prev_finalizers)
729+
return val
730+
end
738731

739732
age0 = h.age
740-
v = convert(V, default())
733+
gc_finalizers_enable(prev_finalizers)
734+
v = convert(V, default())
735+
prev_finalizers = gc_finalizers_enable(false)
741736
if h.age != age0
742737
index = ht_keyindex2(h, key)
743738
end
@@ -748,6 +743,7 @@ function get!{K,V}(default::Callable, h::Dict{K,V}, key0)
748743
else
749744
_setindex!(h, v, key, -index)
750745
end
746+
gc_finalizers_enable(prev_finalizers)
751747
return v
752748
end
753749

@@ -761,26 +757,47 @@ end
761757

762758

763759
function getindex{K,V}(h::Dict{K,V}, key)
760+
prev_finalizers = gc_finalizers_enable(false)
764761
index = ht_keyindex(h, key)
765-
return (index<0) ? throw(KeyError(key)) : h.vals[index]::V
762+
index < 0 && throw(KeyError(key))
763+
val = h.vals[index]::V
764+
gc_finalizers_enable(prev_finalizers)
765+
return val
766766
end
767767

768768
function get{K,V}(h::Dict{K,V}, key, default)
769+
prev_finalizers = gc_finalizers_enable(false)
769770
index = ht_keyindex(h, key)
770-
return (index<0) ? default : h.vals[index]::V
771+
val = (index < 0) ? default : h.vals[index]::V
772+
gc_finalizers_enable(prev_finalizers)
773+
return val
771774
end
772775

773776
function get{K,V}(default::Callable, h::Dict{K,V}, key)
777+
prev_finalizers = gc_finalizers_enable(false)
774778
index = ht_keyindex(h, key)
775-
return (index<0) ? default() : h.vals[index]::V
779+
if index < 0
780+
gc_finalizers_enable(prev_finalizers)
781+
return default()
782+
end
783+
val = h.vals[index]::V
784+
gc_finalizers_enable(prev_finalizers)
785+
return val
776786
end
777787

778788
haskey(h::Dict, key) = (ht_keyindex(h, key) >= 0)
779789
in{T<:Dict}(key, v::KeyIterator{T}) = (ht_keyindex(v.dict, key) >= 0)
780790

781791
function getkey{K,V}(h::Dict{K,V}, key, default)
792+
prev_finalizers = gc_finalizers_enable(false)
782793
index = ht_keyindex(h, key)
783-
return (index<0) ? default : h.keys[index]::K
794+
if index < 0
795+
gc_finalizers_enable(prev_finalizers)
796+
return default
797+
end
798+
val = h.keys[index]::V
799+
gc_finalizers_enable(prev_finalizers)
800+
return val
784801
end
785802

786803
function _pop!(h::Dict, index)
@@ -790,13 +807,24 @@ function _pop!(h::Dict, index)
790807
end
791808

792809
function pop!(h::Dict, key)
810+
prev_finalizers = gc_finalizers_enable(false)
793811
index = ht_keyindex(h, key)
794-
index > 0 ? _pop!(h, index) : throw(KeyError(key))
812+
index < 0 && throw(KeyError(key))
813+
val = _pop!(h, index)
814+
gc_finalizers_enable(prev_finalizers)
815+
return val
795816
end
796817

797818
function pop!(h::Dict, key, default)
819+
prev_finalizers = gc_finalizers_enable(false)
798820
index = ht_keyindex(h, key)
799-
index > 0 ? _pop!(h, index) : default
821+
if index < 0
822+
gc_finalizers_enable(prev_finalizers)
823+
return default
824+
end
825+
val = _pop!(h, index)
826+
gc_finalizers_enable(prev_finalizers)
827+
return val
800828
end
801829

802830
function _delete!(h::Dict, index)
@@ -806,13 +834,17 @@ function _delete!(h::Dict, index)
806834
h.ndel += 1
807835
h.count -= 1
808836
h.age += 1
809-
h
837+
return h
810838
end
811839

812840
function delete!(h::Dict, key)
841+
prev_finalizers = gc_finalizers_enable(false)
813842
index = ht_keyindex(h, key)
814-
if index > 0; _delete!(h, index); end
815-
h
843+
if index > 0
844+
_delete!(h, index)
845+
end
846+
gc_finalizers_enable(prev_finalizers)
847+
return h
816848
end
817849

818850
function skip_deleted(h::Dict, i)
@@ -840,32 +872,28 @@ next{T<:Dict}(v::ValueIterator{T}, i) = (v.dict.vals[i], skip_deleted(v.dict,i+1
840872
# weak key dictionaries
841873

842874
type WeakKeyDict{K,V} <: Associative{K,V}
843-
ht::Dict{Any,V}
844-
deleter::Function
845-
846-
WeakKeyDict() = new(Dict{Any,V}(), identity)
875+
ht::Dict{WeakRef,V}
876+
WeakKeyDict() = new(Dict{WeakRef,V}())
847877
end
848878
WeakKeyDict() = WeakKeyDict{Any,Any}()
849879

850-
function weak_key_delete!(t::Dict, k)
880+
function weak_key_delete!(t::Dict, k::WeakRef)
851881
# when a weak key is finalized, remove from dictionary if it is still there
852882
wk = getkey(t, k, secret_table_token)
853-
if !is(wk,secret_table_token) && is(wk.value, k)
883+
if is(wk, k)
854884
delete!(t, k)
855885
end
856886
end
857887

858888
function setindex!{K}(wkh::WeakKeyDict{K}, v, key)
859889
t = wkh.ht
860890
k = convert(K, key)
861-
if is(wkh.deleter, identity)
862-
wkh.deleter = x->weak_key_delete!(t, x)
863-
end
864-
t[WeakRef(k)] = v
891+
wr = WeakRef(k)
892+
t[wr] = v
865893
# TODO: it might be better to avoid the finalizer, allow
866894
# wiped WeakRefs to remain in the table, and delete them as
867895
# they are discovered by getindex and setindex!.
868-
finalizer(k, wkh.deleter)
896+
finalizer(k, x->weak_key_delete!(t, wr))
869897
return t
870898
end
871899

src/builtins.c

+8-4
Original file line numberDiff line numberDiff line change
@@ -195,12 +195,16 @@ JL_CALLABLE(jl_f_throw)
195195
JL_DLLEXPORT void jl_enter_handler(jl_handler_t *eh)
196196
{
197197
JL_SIGATOMIC_BEGIN();
198-
eh->prev = jl_current_task->eh;
199-
eh->gcstack = jl_pgcstack;
198+
jl_tls_states_t *ptls = jl_get_ptls_states();
199+
eh->prev = ptls->current_task->eh;
200+
eh->gcstack = ptls->pgcstack;
200201
#ifdef JULIA_ENABLE_THREADING
201-
eh->gc_state = jl_gc_state();
202-
eh->locks_len = jl_current_task->locks.len;
202+
eh->gc_state = ptls->gc_state;
203+
eh->locks_len = ptls->current_task->locks.len;
203204
#endif
205+
eh->gc_enabled = !ptls->disable_gc;
206+
eh->finalizers_enabled = !ptls->disable_finalizers;
207+
eh->prev = jl_current_task->eh;
204208
jl_current_task->eh = eh;
205209
// TODO: this should really go after setjmp(). see comment in
206210
// ctx_switch in task.c.

src/codegen.cpp

+4-4
Original file line numberDiff line numberDiff line change
@@ -813,15 +813,16 @@ static void to_function(jl_lambda_info_t *li)
813813
if (!nested_compile && dump_compiles_stream != NULL)
814814
last_time = jl_hrtime();
815815
nested_compile = true;
816-
jl_gc_inhibit_finalizers(nested_compile);
817816
std::unique_ptr<Module> m;
818817
Function *f = NULL, *specf = NULL;
819818
// actually do the work of emitting the function
820819
JL_TRY {
820+
int last_inhibited = jl_finalizers_enable(0);
821821
m = emit_function(li, &li->functionObjectsDecls);
822822
f = (Function*)li->functionObjectsDecls.functionObject;
823823
specf = (Function*)li->functionObjectsDecls.specFunctionObject;
824824
//n_emit++;
825+
jl_finalizers_enable(last_inhibited);
825826
}
826827
JL_CATCH {
827828
// something failed! this is very bad, since other WIP may be pointing to this function
@@ -871,7 +872,6 @@ static void to_function(jl_lambda_info_t *li)
871872
}
872873
li->inCompile = 0;
873874
nested_compile = last_n_c;
874-
jl_gc_inhibit_finalizers(nested_compile);
875875
JL_UNLOCK(&codegen_lock);
876876
JL_SIGATOMIC_END();
877877
if (dump_compiles_stream != NULL) {
@@ -3866,15 +3866,15 @@ static Function *jl_cfunction_object(jl_function_t *ff, jl_value_t *declrt, jl_t
38663866
DebugLoc olddl = builder.getCurrentDebugLocation();
38673867
bool last_n_c = nested_compile;
38683868
nested_compile = true;
3869-
jl_gc_inhibit_finalizers(nested_compile);
3869+
int last_inhibited = jl_finalizers_enable(0);
38703870
Function *f = gen_cfun_wrapper(ff, crt, (jl_tupletype_t*)argt, sf, declrt, (jl_tupletype_t*)sigt);
38713871
// Restore the previous compile context
38723872
if (old != NULL) {
38733873
builder.SetInsertPoint(old);
38743874
builder.SetCurrentDebugLocation(olddl);
38753875
}
38763876
nested_compile = last_n_c;
3877-
jl_gc_inhibit_finalizers(nested_compile);
3877+
jl_finalizers_enable(last_inhibited);
38783878
JL_SIGATOMIC_END();
38793879
JL_GC_POP();
38803880
return f;

src/gc.c

+24-11
Original file line numberDiff line numberDiff line change
@@ -443,7 +443,8 @@ static void jl_gc_signal_end(void)
443443

444444
#endif
445445

446-
static int jl_gc_finalizers_inhibited; // don't run finalizers during codegen #11956
446+
static volatile uint64_t jl_finalizers_disabled_counter; // don't run finalizers during codegen (for example, #11956)
447+
// counter keeps track of per-thread state
447448

448449
// malloc wrappers, aligned allocation
449450

@@ -539,16 +540,28 @@ static void run_finalizers(void)
539540
arraylist_free(&copied_list);
540541
}
541542

542-
void jl_gc_inhibit_finalizers(int state)
543+
JL_DLLEXPORT int jl_finalizers_enable(int on)
543544
{
544-
// NOTE: currently only called with the codegen lock held, but might need
545-
// more synchronization in the future
546-
if (jl_gc_finalizers_inhibited && !state && !jl_in_finalizer) {
547-
jl_in_finalizer = 1;
548-
run_finalizers();
549-
jl_in_finalizer = 0;
545+
jl_tls_states_t *ptls = jl_get_ptls_states();
546+
int prev = !ptls->disable_finalizers;
547+
ptls->disable_finalizers = (on == 0);
548+
if (on && !prev) {
549+
// disable -> enable
550+
// TODO: the atomics are in the wrong place / missing some thread locks
551+
jl_atomic_fetch_add(&jl_finalizers_disabled_counter, -1);
552+
if (jl_finalizers_disabled_counter == 0) {
553+
int8_t was_in_finalizer = jl_in_finalizer;
554+
jl_in_finalizer = 1;
555+
run_finalizers();
556+
jl_in_finalizer = was_in_finalizer;
557+
}
550558
}
551-
jl_gc_finalizers_inhibited = state;
559+
else if (prev && !on) {
560+
// enable -> disable
561+
jl_atomic_fetch_add(&jl_finalizers_disabled_counter, 1);
562+
// TODO: check if finalizers are running and wait for them to finish
563+
}
564+
return prev;
552565
}
553566

554567
static void schedule_all_finalizers(arraylist_t *flist)
@@ -2056,7 +2069,7 @@ static void post_mark(arraylist_t *list, int dryrun)
20562069
}
20572070

20582071
// collector entry point and control
2059-
static volatile uint64_t jl_gc_disable_counter = 0;
2072+
static volatile uint64_t jl_gc_disable_counter = 0; // counter keeps track of per-thread state
20602073

20612074
JL_DLLEXPORT int jl_gc_enable(int on)
20622075
{
@@ -2357,7 +2370,7 @@ JL_DLLEXPORT void jl_gc_collect(int full)
23572370
JL_SIGATOMIC_END();
23582371
jl_gc_state_set(old_state, JL_GC_STATE_WAITING);
23592372

2360-
if (!jl_gc_finalizers_inhibited) {
2373+
if (!jl_finalizers_disabled_counter) {
23612374
int8_t was_in_finalizer = jl_in_finalizer;
23622375
jl_in_finalizer = 1;
23632376
run_finalizers();

0 commit comments

Comments
 (0)