Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit d131863

Browse files
committedJun 7, 2020
Make reinterpretting Unions of nothings/missing and a bitstype work
remove mistakenly included files remove mistakenly included files
1 parent 98e678f commit d131863

File tree

2 files changed

+69
-0
lines changed

2 files changed

+69
-0
lines changed
 

‎base/missing.jl

+27
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,33 @@ convert(::Type{T}, x::T) where {T>:Union{Missing, Nothing}} = x
6969
convert(::Type{T}, x) where {T>:Missing} = convert(nonmissingtype_checked(T), x)
7070
convert(::Type{T}, x) where {T>:Union{Missing, Nothing}} = convert(nonmissingtype_checked(nonnothingtype_checked(T)), x)
7171

72+
for Cs in ((:Missing,), (:Nothing,), (:Missing, :Nothing))
73+
@eval function Base.reinterpret(
74+
::Type{<:T}, # Note this is effectively same as Type{T} since will error if T is abstract
75+
xs::Array{Union{T, $(Cs...)},N}
76+
) where {T,N}
77+
isbitstype(T) || throw(ArgumentError("cannot reinterpret `$(eltype(xs))`, type `$(T)` is not a bits type"))
78+
return unsafe_wrap(Array{T,N}, Ptr{T}(pointer(xs)), length(xs))
79+
end
80+
81+
#==
82+
@eval function Base.convert(
83+
::Type{Array{Q,N}},
84+
xs::Array{Union{T, $(Cs...)},N}
85+
) where {N} where {T<:Q} where Q
86+
87+
all(x->x isa T, xs) || throw(ArgumentError("cannot convert $(eltype(xs)) to $T"))
88+
if isbitstype(T) &&
89+
reinterpret(T, xs)
90+
else
91+
T[x for x in xs]
92+
end
93+
end
94+
==#
95+
end
96+
# Resolve ambiguity
97+
#Base.convert(::Type{Array{Any,1}}, xs::Array{Any,1}) = xs
98+
7299

73100
# Comparison operators
74101
==(::Missing, ::Missing) = missing

‎test/missing.jl

+42
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,48 @@ end
2323
@test_throws MethodError convert(Union{Int, Missing}, "a")
2424
end
2525

26+
mutable struct NotABitsType a end
27+
==(n1::NotABitsType,n2::NotABitsType) = n1.a == n2.a
28+
29+
@testset "convert/reinterpret Arrays of Unions" begin
30+
unions(T) = (Union{T, Missing}, Union{T, Nothing}, Union{T, Missing, Nothing})
31+
corrupters(::Type{T}) where T = (x for x in (missing, nothing) if x isa T)
32+
33+
@testset "bits type ($U)" for U in unions(Int)
34+
xo = U[1, 2, 3]
35+
xr = reinterpret(Int, xo)
36+
#xc = convert(Vector{Int}, xo)
37+
#@test xc isa Vector{Int}
38+
@test xr isa Vector{Int}
39+
@test xr == xo
40+
#@test xc == xo
41+
# should not have allocated new memory
42+
#@test pointer(xc) == pointer(xo)
43+
@test pointer(xr) == pointer(xo)
44+
45+
@test_throws ArgumentError reinterpret(Integer, xo)
46+
47+
@testset "With non-target type values" begin
48+
yo = U[1, 2, 3, corrupters(U)...]
49+
#@test_throws ArgumentError convert(Vector{Int}, yo)
50+
yr = reinterpret(Int, yo)
51+
@test yr isa Vector{Int}
52+
# No comment on what happens for the nonInt values, unspecified behavour
53+
@test yr[1:3] == yo[1:3]
54+
@test length(yo) == length(yr)
55+
end
56+
end
57+
@testset "non-bits type ($U)" for U in unions(NotABitsType)
58+
xo = U[NotABitsType(1), NotABitsType(2), NotABitsType(3)]
59+
@test_throws ArgumentError reinterpret(Int, xo)
60+
#xc = convert(Vector{NotABitsType}, x)
61+
#@test xc isa Vector{Int}
62+
#@test xc == xo
63+
# should have allocated new memory
64+
#@test pointer(xc) != pointer(xo)
65+
end
66+
end
67+
2668
@testset "promote rules" begin
2769
@test promote_type(Missing, Missing) == Missing
2870
@test promote_type(Missing, Int) == Union{Missing, Int}

0 commit comments

Comments
 (0)
Please sign in to comment.