Skip to content

Commit 5cc69d1

Browse files
committed
fix type-predicate queries
expand the set of them to be more "complete", and document them more fully
1 parent d0fb6d4 commit 5cc69d1

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

46 files changed

+830
-878
lines changed

NEWS.md

+4-4
Original file line numberDiff line numberDiff line change
@@ -827,10 +827,10 @@ Deprecated or removed
827827
been deprecated due to inconsistency with linear algebra. Use `.+` and `.-` for these operations
828828
instead ([#22880], [#22932]).
829829

830-
* `isleaftype` is deprecated in favor of a simpler predicate `isconcrete`. Concrete types are
831-
those that might equal `typeof(x)` for some `x`; `isleaftype` includes some types for which
832-
this is not true. If you are certain you need the old behavior, it is temporarily available
833-
as `Base._isleaftype` ([#17086]).
830+
* `isleaftype` is deprecated in favor of the simpler predicates `isconcretetype` and `isdispatchtuple`.
831+
Concrete types are those that might equal `typeof(x)` for some `x`;
832+
`isleaftype` included some types for which this is not true. Those are now categorized more precisely
833+
as "dispatch tuple types" and "!has_free_typevars" (not exported). ([#17086], [#25496])
834834

835835
* `contains(eq, itr, item)` is deprecated in favor of `any` with a predicate ([#23716]).
836836

base/abstractarray.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -1985,7 +1985,7 @@ function hash(a::AbstractArray{T}, h::UInt) where T
19851985
# to a range with more than two elements because more extreme values
19861986
# cannot be represented. We must still hash the two first values as a
19871987
# range since they can always be considered as such (in a wider type)
1988-
if isconcrete(T)
1988+
if isconcretetype(T)
19891989
try
19901990
step = x2 - x1
19911991
catch err

base/broadcast.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -622,7 +622,7 @@ julia> string.(("one","two","three","four"), ": ", 1:4)
622622
const NonleafHandlingTypes = Union{DefaultArrayStyle,ArrayConflict,VectorStyle,MatrixStyle}
623623

624624
@inline function broadcast(f, s::NonleafHandlingTypes, ::Type{ElType}, inds::Indices, As...) where ElType
625-
if !Base._isleaftype(ElType)
625+
if !Base.isconcretetype(ElType)
626626
return broadcast_nonleaf(f, s, ElType, inds, As...)
627627
end
628628
dest = broadcast_similar(f, s, ElType, inds, As...)

base/compiler/abstractinterpretation.jl

+22-23
Original file line numberDiff line numberDiff line change
@@ -118,12 +118,10 @@ function abstract_call_method_with_const_args(@nospecialize(f), argtypes::Vector
118118
haveconst = false
119119
for i in 1:nargs
120120
a = argtypes[i]
121-
if isa(a, Const) && !isdefined(typeof(a.val), :instance)
122-
if !isleaftype(a.val) # alternately: !isa(a.val, DataType) || !isconstType(Type{a.val})
123-
# have new information from argtypes that wasn't available from the signature
124-
haveconst = true
125-
break
126-
end
121+
if isa(a, Const) && !isdefined(typeof(a.val), :instance) && !(isa(a.val, Type) && issingletontype(a.val))
122+
# have new information from argtypes that wasn't available from the signature
123+
haveconst = true
124+
break
127125
end
128126
end
129127
haveconst || return Any
@@ -378,11 +376,13 @@ end
378376

379377
# do apply(af, fargs...), where af is a function value
380378
function abstract_apply(@nospecialize(aft), fargs::Vector{Any}, aargtypes::Vector{Any}, vtypes::VarTable, sv::InferenceState)
381-
if !isa(aft, Const) && !isconstType(aft)
382-
if !(isleaftype(aft) || aft <: Type) || (aft <: Builtin) || (aft <: IntrinsicFunction)
379+
if !isa(aft, Const) && (!isType(aft) || has_free_typevars(aft))
380+
if !isconcretetype(aft) || (aft <: Builtin)
381+
# non-constant function of unknown type: bail now,
382+
# since it seems unlikely that abstract_call will be able to do any better after splitting
383+
# this also ensures we don't call abstract_call_gf_by_type below on an IntrinsicFunction or Builtin
383384
return Any
384385
end
385-
# non-constant function, but type is known
386386
end
387387
res = Union{}
388388
nargs = length(fargs)
@@ -394,7 +394,7 @@ function abstract_apply(@nospecialize(aft), fargs::Vector{Any}, aargtypes::Vecto
394394
for ti in (splitunions ? uniontypes(aargtypes[i]) : Any[aargtypes[i]])
395395
cti = precise_container_type(fargs[i], ti, vtypes, sv)
396396
for ct in ctypes
397-
if !isempty(ct) && isvarargtype(ct[end])
397+
if isvarargtype(ct[end])
398398
tail = tuple_tail_elem(unwrapva(ct[end]), cti)
399399
push!(ctypes´, push!(ct[1:(end - 1)], tail))
400400
else
@@ -672,23 +672,22 @@ function abstract_eval_call(e::Expr, vtypes::VarTable, sv::InferenceState)
672672
ft = argtypes[1]
673673
if isa(ft, Const)
674674
f = ft.val
675+
elseif isconstType(ft)
676+
f = ft.parameters[1]
677+
elseif isa(ft, DataType) && isdefined(ft, :instance)
678+
f = ft.instance
675679
else
676-
if isType(ft) && isleaftype(ft.parameters[1])
677-
f = ft.parameters[1]
678-
elseif isleaftype(ft) && isdefined(ft, :instance)
679-
f = ft.instance
680-
else
681-
for i = 2:(length(argtypes)-1)
682-
if isvarargtype(argtypes[i])
683-
return Any
684-
end
685-
end
686-
# non-constant function, but type is known
687-
if (isleaftype(ft) || ft <: Type) && !(ft <: Builtin) && !(ft <: IntrinsicFunction)
688-
return abstract_call_gf_by_type(nothing, argtypes, argtypes_to_type(argtypes), sv)
680+
for i = 2:(length(argtypes) - 1)
681+
if isvarargtype(argtypes[i])
682+
return Any
689683
end
684+
end
685+
# non-constant function, but the number of arguments is known
686+
# and the ft is not a Builtin or IntrinsicFunction
687+
if typeintersect(widenconst(ft), Builtin) != Union{}
690688
return Any
691689
end
690+
return abstract_call_gf_by_type(nothing, argtypes, argtypes_to_type(argtypes), sv)
692691
end
693692
return abstract_call(f, e.args, argtypes, vtypes, sv)
694693
end

base/compiler/compiler.jl

-2
Original file line numberDiff line numberDiff line change
@@ -76,8 +76,6 @@ include("docs/core.jl")
7676
inlining_enabled() = (JLOptions().can_inline == 1)
7777
coverage_enabled() = (JLOptions().code_coverage != 0)
7878

79-
const isleaftype = _isleaftype
80-
8179
include("compiler/utilities.jl")
8280
include("compiler/validation.jl")
8381

base/compiler/inferencestate.jl

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ mutable struct InferenceState
8484
at = (i > nargs) ? Bottom : argtypes[i]
8585
if !toplevel && linfo.def.isva && i == nargs
8686
if !(at == Tuple) # would just be a no-op
87-
vararg_type_container = limit_tuple_depth(params, unwrap_unionall(at)) # TODO: should be limiting tuple depth much earlier than here
87+
vararg_type_container = unwrap_unionall(at)
8888
vararg_type = tuple_tfunc(vararg_type_container) # returns a Const object, if applicable
8989
at = rewrap(vararg_type, linfo.specTypes)
9090
end

base/compiler/optimize.jl

+33-29
Original file line numberDiff line numberDiff line change
@@ -827,8 +827,8 @@ function effect_free(@nospecialize(e), src::CodeInfo, mod::Module, allow_volatil
827827
return true
828828
end
829829
if head === :static_parameter
830-
# if we aren't certain about the type, it might be an UndefVarError at runtime
831-
return (isa(e.typ, DataType) && isleaftype(e.typ)) || isa(e.typ, Const)
830+
# if we aren't certain enough about the type, it might be an UndefVarError at runtime
831+
return isa(e.typ, Const) || issingletontype(widenconst(e.typ))
832832
end
833833
if e.typ === Bottom
834834
return false
@@ -841,17 +841,17 @@ function effect_free(@nospecialize(e), src::CodeInfo, mod::Module, allow_volatil
841841
return false
842842
elseif is_known_call(e, getfield, src, mod)
843843
nargs = length(ea)
844-
(2 < nargs < 5) || return false
844+
(3 < nargs < 4) || return false
845845
et = exprtype(e, src, mod)
846-
if !isa(et, Const) && !(isType(et) && isleaftype(et))
846+
# TODO: check ninitialized
847+
if !isa(et, Const) && !isconstType(et)
847848
# first argument must be immutable to ensure e is affect_free
848849
a = ea[2]
849-
typ = widenconst(exprtype(a, src, mod))
850-
if isconstType(typ)
851-
if Const(:uid) exprtype(ea[3], src, mod)
852-
return false # DataType uid field can change
853-
end
854-
elseif typ !== SimpleVector && (!isa(typ, DataType) || typ.mutable || typ.abstract)
850+
typ = unwrap_unionall(widenconst(exprtype(a, src, mod)))
851+
if isType(typ)
852+
# all fields of subtypes of Type are effect-free
853+
# (including the non-inferrable uid field)
854+
elseif !isa(typ, DataType) || typ.abstract || (typ.mutable && length(typ.types) > 0)
855855
return false
856856
end
857857
end
@@ -874,7 +874,8 @@ function effect_free(@nospecialize(e), src::CodeInfo, mod::Module, allow_volatil
874874
# `Expr(:new)` of unknown type could raise arbitrary TypeError.
875875
typ, isexact = instanceof_tfunc(typ)
876876
isexact || return false
877-
(isleaftype(typ) && !iskindtype(typ)) || return false
877+
isconcretetype(typ) || return false
878+
!iskindtype(typ) || return false
878879
typ = typ::DataType
879880
if !allow_volatile && typ.mutable
880881
return false
@@ -1086,11 +1087,11 @@ function inlineable(@nospecialize(f), @nospecialize(ft), e::Expr, atypes::Vector
10861087
sv::OptimizationState)
10871088
argexprs = e.args
10881089

1089-
if (f === typeassert || ft typeof(typeassert)) && length(atypes)==3
1090+
if (f === typeassert || ft typeof(typeassert)) && length(atypes) == 3
10901091
# typeassert(x::S, T) => x, when S<:T
10911092
a3 = atypes[3]
1092-
if (isType(a3) && isleaftype(a3) && atypes[2] a3.parameters[1]) ||
1093-
(isa(a3,Const) && isa(a3.val,Type) && atypes[2] a3.val)
1093+
if (isType(a3) && !has_free_typevars(a3) && atypes[2] a3.parameters[1]) ||
1094+
(isa(a3, Const) && isa(a3.val, Type) && atypes[2] a3.val)
10941095
return (argexprs[2], ())
10951096
end
10961097
end
@@ -1119,7 +1120,9 @@ function inlineable(@nospecialize(f), @nospecialize(ft), e::Expr, atypes::Vector
11191120
if f === Core.invoke && length(atypes) >= 3
11201121
ft = widenconst(atypes[2])
11211122
invoke_tt = widenconst(atypes[3])
1122-
if !isleaftype(ft) || !isleaftype(invoke_tt) || !isType(invoke_tt)
1123+
if !(isconcretetype(ft) || ft <: Type) || !isType(invoke_tt) ||
1124+
has_free_typevars(invoke_tt) || has_free_typevars(ft) || (ft <: Builtin)
1125+
# TODO: this is really aggressive at preventing inlining of closures. maybe drop `isconcretetype` requirement?
11231126
return NOT_FOUND
11241127
end
11251128
if !(isa(invoke_tt.parameters[1], Type) &&
@@ -1282,12 +1285,10 @@ function inlineable(@nospecialize(f), @nospecialize(ft), e::Expr, atypes::Vector
12821285
haveconst = false
12831286
for i in 1:length(atypes)
12841287
a = atypes[i]
1285-
if isa(a, Const) && !isdefined(typeof(a.val), :instance)
1286-
if !isleaftype(a.val) # alternately: !isa(a.val, DataType) || !isconstType(Type{a.val})
1287-
# have new information from argtypes that wasn't available from the signature
1288-
haveconst = true
1289-
break
1290-
end
1288+
if isa(a, Const) && !isdefined(typeof(a.val), :instance) && !(isa(a.val, Type) && issingletontype(a.val))
1289+
# have new information from argtypes that wasn't available from the signature
1290+
haveconst = true
1291+
break
12911292
end
12921293
end
12931294
if haveconst
@@ -1549,8 +1550,9 @@ end
15491550

15501551
# saturating sum (inputs are nonnegative), prevents overflow with typemax(Int) below
15511552
plus_saturate(x, y) = max(x, y, x+y)
1553+
15521554
# known return type
1553-
isknowntype(T) = (T == Union{}) || isleaftype(T)
1555+
isknowntype(@nospecialize T) = (T == Union{}) || isconcretetype(T)
15541556

15551557
function statement_cost(ex::Expr, line::Int, src::CodeInfo, mod::Module, params::Params)
15561558
head = ex.head
@@ -1781,7 +1783,8 @@ function inline_call(e::Expr, sv::OptimizationState, stmts::Vector{Any}, boundsc
17811783
ft = Bool
17821784
else
17831785
f = nothing
1784-
if !( isleaftype(ft) || ft<:Type )
1786+
if !(isconcretetype(ft) || (widenconst(ft) <: Type)) || has_free_typevars(ft)
1787+
# TODO: this is really aggressive at preventing inlining of closures. maybe drop `isconcretetype` requirement?
17851788
return e
17861789
end
17871790
end
@@ -1938,7 +1941,8 @@ function inline_call(e::Expr, sv::OptimizationState, stmts::Vector{Any}, boundsc
19381941
ft = Bool
19391942
else
19401943
f = nothing
1941-
if !( isleaftype(ft) || ft<:Type )
1944+
if !(isconcretetype(ft) || (widenconst(ft) <: Type)) || has_free_typevars(ft)
1945+
# TODO: this is really aggressive at preventing inlining of closures. maybe drop `isconcretetype` requirement?
19421946
return e
19431947
end
19441948
end
@@ -2754,15 +2758,15 @@ end
27542758

27552759
function split_disjoint_assign!(ctx::AllocOptContext, info, key)
27562760
key.second && return false
2757-
isleaftype(ctx.sv.src.slottypes[key.first]) && return false
2761+
isdispatchelem(widenconst(ctx.sv.src.slottypes[key.first])) && return false # no splitting can be necessary
27582762
alltypes = IdDict()
27592763
ndefs = length(info.defs)
27602764
deftypes = Vector{Any}(uninitialized, ndefs)
27612765
for i in 1:ndefs
27622766
def = info.defs[i]
27632767
defex = (def.assign::Expr).args[2]
27642768
rhstyp = widenconst(exprtype(defex, ctx.sv.src, ctx.sv.mod))
2765-
isleaftype(rhstyp) || return false
2769+
isdispatchelem(rhstyp) || return false
27662770
alltypes[rhstyp] = nothing
27672771
deftypes[i] = rhstyp
27682772
end
@@ -2772,7 +2776,7 @@ function split_disjoint_assign!(ctx::AllocOptContext, info, key)
27722776
slot = usex.args[use.exidx]
27732777
if isa(slot, TypedSlot)
27742778
usetyp = widenconst(slot.typ)
2775-
if isleaftype(usetyp)
2779+
if isdispatchelem(usetyp)
27762780
alltypes[usetyp] = nothing
27772781
continue
27782782
end
@@ -2827,7 +2831,7 @@ function split_disjoint_assign!(ctx::AllocOptContext, info, key)
28272831
slot = usex.args[use.exidx]
28282832
if isa(slot, TypedSlot)
28292833
usetyp = widenconst(slot.typ)
2830-
if isleaftype(usetyp)
2834+
if isdispatchelem(usetyp)
28312835
usetyp = widenconst(slot.typ)
28322836
new_slot = alltypes[usetyp]
28332837
if !isa(new_slot, SlotNumber)
@@ -2997,7 +3001,7 @@ function split_struct_alloc!(ctx::AllocOptContext, info, key)
29973001
elseif defex.head === :new
29983002
typ = widenconst(exprtype(defex, ctx.sv.src, ctx.sv.mod))
29993003
# typ <: Tuple shouldn't happen but just in case someone generated invalid AST
3000-
if !isa(typ, DataType) || !isleaftype(typ) || typ <: Tuple
3004+
if !isa(typ, DataType) || !isdispatchelem(typ) || typ <: Tuple
30013005
return false
30023006
end
30033007
si = structinfo_new(ctx, defex, typ)

base/compiler/params.jl

+1-3
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,6 @@ struct Params
2828

2929
# parameters limiting large types
3030
MAX_TUPLETYPE_LEN::Int
31-
MAX_TUPLE_DEPTH::Int
3231

3332
# when attempting to inlining _apply, abort the optimization if the tuple
3433
# contains more than this many elements
@@ -42,13 +41,12 @@ struct Params
4241
inline_tupleret_bonus::Int = 400,
4342
max_methods::Int = 4,
4443
tupletype_len::Int = 15,
45-
tuple_depth::Int = 4,
4644
tuple_splat::Int = 16,
4745
union_splitting::Int = 4,
4846
apply_union_enum::Int = 8)
4947
return new(Vector{InferenceResult}(),
5048
world, inlining, true, false, inline_cost_threshold, inline_nonleaf_penalty,
5149
inline_tupleret_bonus, max_methods, union_splitting, apply_union_enum,
52-
tupletype_len, tuple_depth, tuple_splat)
50+
tupletype_len, tuple_splat)
5351
end
5452
end

0 commit comments

Comments
 (0)