Skip to content

Commit

Permalink
fix anonymous function serialization logic
Browse files Browse the repository at this point in the history
should_send_whole_type was sending too many regular type that weren't functions
and deserialize was trying to hide that error,
but was then not deserializing things that were anonymous functions

fix #16091
  • Loading branch information
vtjnash committed Jul 29, 2016
1 parent 5144387 commit d584142
Show file tree
Hide file tree
Showing 5 changed files with 95 additions and 60 deletions.
23 changes: 9 additions & 14 deletions base/clusterserialize.jl
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
# This file is a part of Julia. License is MIT: http://julialang.org/license

import .Serializer: known_object_data, object_number, serialize_cycle, deserialize_cycle, writetag,
__deserialized_types__, serialize_typename_body, deserialize_typename_body,
__deserialized_types__, serialize_typename, deserialize_typename,
TYPENAME_TAG, object_numbers

type ClusterSerializer{I<:IO} <: AbstractSerializer
Expand All @@ -17,16 +17,16 @@ ClusterSerializer(io::IO) = ClusterSerializer{typeof(io)}(io)

function deserialize(s::ClusterSerializer, ::Type{TypeName})
full_body_sent = deserialize(s)
number = read(s.io, UInt64)
if !full_body_sent
number = read(s.io, UInt64)
tn = get(known_object_data, number, nothing)::TypeName
if !haskey(object_numbers, tn)
# setup reverse mapping for serialize
object_numbers[tn] = number
end
deserialize_cycle(s, tn)
else
tn = invoke(deserialize, (AbstractSerializer, Type{TypeName}), s, TypeName)
tn = deserialize_typename(s, number)
end
return tn
end
Expand All @@ -36,18 +36,13 @@ function serialize(s::ClusterSerializer, t::TypeName)
writetag(s.io, TYPENAME_TAG)

identifier = object_number(t)
if !(identifier in s.sent_objects)
serialize(s, true)
write(s.io, identifier)
serialize(s, t.name)
serialize(s, t.module)
serialize_typename_body(s, t)
send_whole = !(identifier in s.sent_objects)
serialize(s, send_whole)
write(s.io, identifier)
if send_whole
serialize_typename(s, t)
push!(s.sent_objects, identifier)
# println(t.module, ":", t.name, ", id:", identifier, " sent")
else
serialize(s, false)
write(s.io, identifier)
# println(t.module, ":", t.name, ", id:", identifier, " NOT sent")
end
# println(t.module, ":", t.name, ", id:", identifier, send_whole ? " sent" : " NOT sent")
nothing
end
1 change: 1 addition & 0 deletions base/methodshow.jl
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,7 @@ function show(io::IO, m::Method; kwtype::Nullable{DataType}=Nullable{DataType}()
decls = Any[(), ("...", "")]
elseif ft <: Function &&
isdefined(ft.name.module, ft.name.mt.name) &&
# TODO: more accurate test? (tn.name === "#" name)
ft == typeof(getfield(ft.name.module, ft.name.mt.name))
print(io, ft.name.mt.name)
elseif isa(ft, DataType) && is(ft.name, Type.name) && isleaftype(ft)
Expand Down
88 changes: 42 additions & 46 deletions base/serialize.jl
Original file line number Diff line number Diff line change
Expand Up @@ -374,7 +374,7 @@ end
function serialize(s::AbstractSerializer, g::GlobalRef)
writetag(s.io, GLOBALREF_TAG)
if g.mod === Main && isdefined(g.mod, g.name) && isconst(g.mod, g.name)
v = eval(g)
v = getfield(g.mod, g.name)
if isa(v, DataType) && v === v.name.primary && should_send_whole_type(s, v)
# handle references to types in Main by sending the whole type.
# needed to be able to send nested functions (#15451).
Expand All @@ -393,17 +393,16 @@ function serialize(s::AbstractSerializer, t::TypeName)
serialize_cycle(s, t) && return
writetag(s.io, TYPENAME_TAG)
write(s.io, object_number(t))
serialize(s, t.name)
serialize(s, t.module)
serialize_typename_body(s, t)
serialize_typename(s, t)
end

function serialize_typename_body(s::AbstractSerializer, t::TypeName)
function serialize_typename(s::AbstractSerializer, t::TypeName)
serialize(s, t.name)
serialize(s, t.names)
serialize(s, t.primary.super)
serialize(s, t.primary.parameters)
serialize(s, t.primary.types)
serialize(s, t.primary.size)
serialize(s, isdefined(t.primary, :instance))
serialize(s, t.primary.abstract)
serialize(s, t.primary.mutable)
serialize(s, t.primary.ninitialized)
Expand All @@ -423,24 +422,28 @@ function serialize_typename_body(s::AbstractSerializer, t::TypeName)
end

# decide whether to send all data for a type (instead of just its name)
function should_send_whole_type(s, t::ANY)
function should_send_whole_type(s, t::DataType)
tn = t.name
if isdefined(tn, :mt)
# TODO improve somehow
# send whole type for anonymous functions in Main
fname = tn.mt.name
name = tn.mt.name
mod = tn.module
toplevel = isdefined(mod, fname) && isdefined(t, :instance) &&
getfield(mod, fname) === t.instance
ishidden = unsafe_load(unsafe_convert(Ptr{UInt8}, fname))==UInt8('#')
return mod === __deserialized_types__ || (mod === Main && (ishidden || !toplevel))
isanonfunction = mod === Main && # only Main
t.super === Function && # only Functions
unsafe_load(unsafe_convert(Ptr{UInt8}, tn.name)) == UInt8('#') && # hidden type
(!isdefined(mod, name) || t != typeof(getfield(mod, name))) # XXX: 95% accurate test for this being an inner function
# TODO: more accurate test? (tn.name !== "#" name)
#TODO: iskw = startswith(tn.name, "#kw#") && ???
#TODO: iskw && return send-as-kwftype
return mod === __deserialized_types__ || isanonfunction
end
return false
end

# `type_itself` means we are serializing a type object. when it's false, we are
# sending the type tag part of some other object's representation.
function serialize_type_data(s, t::ANY, type_itself::Bool)
function serialize_type_data(s, t::DataType, type_itself::Bool)
whole = should_send_whole_type(s, t)
form = type_itself ? UInt8(0) : UInt8(1)
if whole
Expand Down Expand Up @@ -725,28 +728,25 @@ function deserialize(s::AbstractSerializer, ::Type{Union})
Union{types...}
end

module __deserialized_types__
end

module __deserialized_types__ end

function deserialize(s::AbstractSerializer, ::Type{TypeName})
# the deserialize_cycle call can be delayed, since neither
# Symbol nor Module will use the backref table
number = read(s.io, UInt64)
return deserialize_typename(s, number)
end

function deserialize_typename(s::AbstractSerializer, number)
name = deserialize(s)::Symbol
mod = deserialize(s)::Module
tn = get(known_object_data, number, nothing)
if tn !== nothing
makenew = false
elseif mod !== __deserialized_types__ && isdefined(mod, name)
tn = getfield(mod, name).name
# TODO: confirm somehow that the types match
makenew = false
known_object_data[number] = tn
else
name = gensym()
mod = __deserialized_types__
tn = ccall(:jl_new_typename_in, Ref{TypeName}, (Any, Any), name, mod)
# reuse the same name for the type, if possible, for nicer debugging
tn_name = isdefined(__deserialized_types__, name) ? gensym() : name
tn = ccall(:jl_new_typename_in, Ref{TypeName}, (Any, Any),
tn_name, __deserialized_types__)
makenew = true
known_object_data[number] = tn
end
Expand All @@ -755,19 +755,15 @@ function deserialize(s::AbstractSerializer, ::Type{TypeName})
object_numbers[tn] = number
end
deserialize_cycle(s, tn)
deserialize_typename_body(s, tn, number, name, mod, makenew)
return tn
end

function deserialize_typename_body(s::AbstractSerializer, tn, number, name, mod, makenew)
names = deserialize(s)
super = deserialize(s)
parameters = deserialize(s)
types = deserialize(s)
size = deserialize(s)
abstr = deserialize(s)
mutable = deserialize(s)
ninitialized = deserialize(s)
names = deserialize(s)::SimpleVector
super = deserialize(s)::Type
parameters = deserialize(s)::SimpleVector
types = deserialize(s)::SimpleVector
has_instance = deserialize(s)::Bool
abstr = deserialize(s)::Bool
mutable = deserialize(s)::Bool
ninitialized = deserialize(s)::Int32

if makenew
tn.names = names
Expand All @@ -778,22 +774,20 @@ function deserialize_typename_body(s::AbstractSerializer, tn, number, name, mod,
tn, super, parameters, names, types,
abstr, mutable, ninitialized)
ty = tn.primary
ccall(:jl_set_const, Void, (Any, Any, Any), mod, name, ty)
if !isdefined(ty, :instance)
if isempty(parameters) && !abstr && size == 0 && (!mutable || isempty(names))
# use setfield! directly to avoid `fieldtype` lowering expecting to see a Singleton object already on ty
Core.setfield!(ty, :instance, ccall(:jl_new_struct, Any, (Any, Any...), ty))
end
ccall(:jl_set_const, Void, (Any, Any, Any), tn.module, tn.name, ty)
if has_instance && !isdefined(ty, :instance)
# use setfield! directly to avoid `fieldtype` lowering expecting to see a Singleton object already on ty
Core.setfield!(ty, :instance, ccall(:jl_new_struct, Any, (Any, Any...), ty))
end
end

tag = Int32(read(s.io, UInt8)::UInt8)
if tag != UNDEFREF_TAG
mtname = handle_deserialize(s, tag)
defs = deserialize(s)
maxa = deserialize(s)
maxa = deserialize(s)::Int
if makenew
tn.mt = ccall(:jl_new_method_table, Any, (Any, Any), name, mod)
tn.mt = ccall(:jl_new_method_table, Any, (Any, Any), name, tn.module)
tn.mt.name = mtname
tn.mt.defs = defs
tn.mt.max_args = maxa
Expand All @@ -806,6 +800,7 @@ function deserialize_typename_body(s::AbstractSerializer, tn, number, name, mod,
end
end
end
return tn::TypeName
end

function deserialize_datatype(s::AbstractSerializer)
Expand Down Expand Up @@ -853,7 +848,8 @@ function deserialize(s::AbstractSerializer, t::DataType)
return ccall(:jl_new_struct, Any, (Any,Any...), t)
elseif isbits(t)
if nf == 1
return ccall(:jl_new_struct, Any, (Any,Any...), t, deserialize(s))
f1 = deserialize(s)
return ccall(:jl_new_struct, Any, (Any,Any...), t, f1)
elseif nf == 2
f1 = deserialize(s)
f2 = deserialize(s)
Expand Down
28 changes: 28 additions & 0 deletions test/parallel_exec.jl
Original file line number Diff line number Diff line change
Expand Up @@ -1023,6 +1023,34 @@ end
# issue #15451
@test remotecall_fetch(x->(y->2y)(x)+1, workers()[1], 3) == 7

# issue #16091
type T16091 end
wid = workers()[1]
@test try
remotecall_fetch(()->T16091, wid)
false
catch ex
((ex::RemoteException).captured::CapturedException).ex === UndefVarError(:T16091)
end
@test try
remotecall_fetch(identity, wid, T16091)
false
catch ex
((ex::RemoteException).captured::CapturedException).ex === UndefVarError(:T16091)
end

f16091a() = 1
remotecall_fetch(()->eval(:(f16091a() = 2)), wid)
@test remotecall_fetch(f16091a, wid) === 2
@test remotecall_fetch((myid)->remotecall_fetch(f16091a, myid), wid, myid()) === 1

# these will only heisen-fail, since it depends on the gensym counter collisions:
f16091b = () -> 1
remotecall_fetch(()->eval(:(f16091b = () -> 2)), wid)
@test remotecall_fetch(f16091b, 2) === 1
@test remotecall_fetch((myid)->remotecall_fetch(f16091b, myid), wid, myid()) === 2


# issue #16451
rng=RandomDevice()
retval = @parallel (+) for _ in 1:10
Expand Down
15 changes: 15 additions & 0 deletions test/serialize.jl
Original file line number Diff line number Diff line change
Expand Up @@ -278,6 +278,21 @@ create_serialization_stream() do s # Base generic function
@test deserialize(s)() === 1
end

# Anonymous Functions
create_serialization_stream() do s
local g() = :magic_token_anon_fun_test
serialize(s, g)
serialize(s, g)

seekstart(s)
local g2 = deserialize(s)
@test g2 !== g
@test g2() == :magic_token_anon_fun_test
@test g2() == :magic_token_anon_fun_test
@test deserialize(s) === g2
end


# Task
create_serialization_stream() do s # user-defined type array
f = () -> begin task_local_storage(:v, 2); return 1+1 end
Expand Down

0 comments on commit d584142

Please sign in to comment.