Skip to content

Commit

Permalink
fix #22592, specificity of union compared to a subtype of it (#29139)
Browse files Browse the repository at this point in the history
We considered Union{A,B} more specific than B if A was more specific
than B (but not a subtype of it). Clearly, it should not be.

(cherry picked from commit 3143d89)
  • Loading branch information
JeffBezanson authored and KristofferC committed Feb 11, 2019
1 parent 646d745 commit 1fb5c00
Show file tree
Hide file tree
Showing 2 changed files with 22 additions and 14 deletions.
25 changes: 11 additions & 14 deletions src/subtype.c
Original file line number Diff line number Diff line change
Expand Up @@ -2619,18 +2619,6 @@ static int args_morespecific_fix1(jl_value_t *a, jl_value_t *b, int swap, jl_typ
return ret;
}

static int partially_morespecific(jl_value_t *a, jl_value_t *b, int invariant, jl_typeenv_t *env)
{
if (jl_is_uniontype(b)) {
jl_uniontype_t *u = (jl_uniontype_t*)b;
if (type_morespecific_(a, u->a, invariant, env) ||
type_morespecific_(a, u->b, invariant, env))
return 1;
return 0;
}
return type_morespecific_(a, b, invariant, env);
}

static int count_occurs(jl_value_t *t, jl_tvar_t *v)
{
if (t == (jl_value_t*)v)
Expand Down Expand Up @@ -2702,9 +2690,18 @@ static int type_morespecific_(jl_value_t *a, jl_value_t *b, int invariant, jl_ty
if (jl_is_uniontype(a)) {
// Union a is more specific than b if some element of a is more specific than b, but
// not vice-versa.
if (sub_msp(b, a, env))
return 0;
jl_uniontype_t *u = (jl_uniontype_t*)a;
return ((partially_morespecific(u->a, b, invariant, env) || partially_morespecific(u->b, b, invariant, env)) &&
!partially_morespecific(b, a, invariant, env));
if (type_morespecific_(u->a, b, invariant, env) || type_morespecific_(u->b, b, invariant, env)) {
if (jl_is_uniontype(b)) {
jl_uniontype_t *v = (jl_uniontype_t*)b;
if (type_morespecific_(v->a, a, invariant, env) || type_morespecific_(v->b, a, invariant, env))
return 0;
}
return 1;
}
return 0;
}

if (jl_is_type_type(a) && !invariant) {
Expand Down
11 changes: 11 additions & 0 deletions test/specificity.jl
Original file line number Diff line number Diff line change
Expand Up @@ -214,3 +214,14 @@ f27361(::M) where M <: Tuple{3} = nothing

@test args_morespecific(Tuple{Type{Any}, Type}, Tuple{Type{T}, Type{T}} where T)
@test !args_morespecific(Tuple{Type{Any}, Type}, Tuple{Type{T}, Type{T}} where T<:Union{})

# issue #22592
abstract type Colorant22592{T,N} end
abstract type Color22592{T, N} <: Colorant22592{T,N} end
abstract type AbstractRGB22592{T} <: Color22592{T,3} end
AbstractGray22592{T} = Color22592{T,1}
MathTypes22592{T,C} = Union{AbstractRGB22592{T},AbstractGray22592{T}}
@test !args_morespecific(Tuple{MathTypes22592}, Tuple{AbstractGray22592})
@test !args_morespecific(Tuple{MathTypes22592, MathTypes22592}, Tuple{AbstractGray22592})

@test args_morespecific(Union{Set,Dict,Vector}, Union{Vector,AbstractSet})

0 comments on commit 1fb5c00

Please sign in to comment.