@@ -827,8 +827,8 @@ function effect_free(@nospecialize(e), src::CodeInfo, mod::Module, allow_volatil
827
827
return true
828
828
end
829
829
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) )
832
832
end
833
833
if e. typ === Bottom
834
834
return false
@@ -841,17 +841,17 @@ function effect_free(@nospecialize(e), src::CodeInfo, mod::Module, allow_volatil
841
841
return false
842
842
elseif is_known_call (e, getfield, src, mod)
843
843
nargs = length (ea)
844
- (2 < nargs < 5 ) || return false
844
+ (3 < nargs < 4 ) || return false
845
845
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)
847
848
# first argument must be immutable to ensure e is affect_free
848
849
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 )
855
855
return false
856
856
end
857
857
end
@@ -874,7 +874,8 @@ function effect_free(@nospecialize(e), src::CodeInfo, mod::Module, allow_volatil
874
874
# `Expr(:new)` of unknown type could raise arbitrary TypeError.
875
875
typ, isexact = instanceof_tfunc (typ)
876
876
isexact || return false
877
- (isleaftype (typ) && ! iskindtype (typ)) || return false
877
+ isconcretetype (typ) || return false
878
+ ! iskindtype (typ) || return false
878
879
typ = typ:: DataType
879
880
if ! allow_volatile && typ. mutable
880
881
return false
@@ -1086,11 +1087,11 @@ function inlineable(@nospecialize(f), @nospecialize(ft), e::Expr, atypes::Vector
1086
1087
sv:: OptimizationState )
1087
1088
argexprs = e. args
1088
1089
1089
- if (f === typeassert || ft ⊑ typeof (typeassert)) && length (atypes)== 3
1090
+ if (f === typeassert || ft ⊑ typeof (typeassert)) && length (atypes) == 3
1090
1091
# typeassert(x::S, T) => x, when S<:T
1091
1092
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)
1094
1095
return (argexprs[2 ], ())
1095
1096
end
1096
1097
end
@@ -1119,7 +1120,9 @@ function inlineable(@nospecialize(f), @nospecialize(ft), e::Expr, atypes::Vector
1119
1120
if f === Core. invoke && length (atypes) >= 3
1120
1121
ft = widenconst (atypes[2 ])
1121
1122
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?
1123
1126
return NOT_FOUND
1124
1127
end
1125
1128
if ! (isa (invoke_tt. parameters[1 ], Type) &&
@@ -1282,12 +1285,10 @@ function inlineable(@nospecialize(f), @nospecialize(ft), e::Expr, atypes::Vector
1282
1285
haveconst = false
1283
1286
for i in 1 : length (atypes)
1284
1287
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
1291
1292
end
1292
1293
end
1293
1294
if haveconst
@@ -1549,8 +1550,9 @@ end
1549
1550
1550
1551
# saturating sum (inputs are nonnegative), prevents overflow with typemax(Int) below
1551
1552
plus_saturate (x, y) = max (x, y, x+ y)
1553
+
1552
1554
# known return type
1553
- isknowntype (T) = (T == Union{}) || isleaftype (T)
1555
+ isknowntype (@nospecialize T) = (T == Union{}) || isconcretetype (T)
1554
1556
1555
1557
function statement_cost (ex:: Expr , line:: Int , src:: CodeInfo , mod:: Module , params:: Params )
1556
1558
head = ex. head
@@ -1781,7 +1783,8 @@ function inline_call(e::Expr, sv::OptimizationState, stmts::Vector{Any}, boundsc
1781
1783
ft = Bool
1782
1784
else
1783
1785
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?
1785
1788
return e
1786
1789
end
1787
1790
end
@@ -1938,7 +1941,8 @@ function inline_call(e::Expr, sv::OptimizationState, stmts::Vector{Any}, boundsc
1938
1941
ft = Bool
1939
1942
else
1940
1943
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?
1942
1946
return e
1943
1947
end
1944
1948
end
@@ -2754,15 +2758,15 @@ end
2754
2758
2755
2759
function split_disjoint_assign! (ctx:: AllocOptContext , info, key)
2756
2760
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
2758
2762
alltypes = IdDict ()
2759
2763
ndefs = length (info. defs)
2760
2764
deftypes = Vector {Any} (uninitialized, ndefs)
2761
2765
for i in 1 : ndefs
2762
2766
def = info. defs[i]
2763
2767
defex = (def. assign:: Expr ). args[2 ]
2764
2768
rhstyp = widenconst (exprtype (defex, ctx. sv. src, ctx. sv. mod))
2765
- isleaftype (rhstyp) || return false
2769
+ isdispatchelem (rhstyp) || return false
2766
2770
alltypes[rhstyp] = nothing
2767
2771
deftypes[i] = rhstyp
2768
2772
end
@@ -2772,7 +2776,7 @@ function split_disjoint_assign!(ctx::AllocOptContext, info, key)
2772
2776
slot = usex. args[use. exidx]
2773
2777
if isa (slot, TypedSlot)
2774
2778
usetyp = widenconst (slot. typ)
2775
- if isleaftype (usetyp)
2779
+ if isdispatchelem (usetyp)
2776
2780
alltypes[usetyp] = nothing
2777
2781
continue
2778
2782
end
@@ -2827,7 +2831,7 @@ function split_disjoint_assign!(ctx::AllocOptContext, info, key)
2827
2831
slot = usex. args[use. exidx]
2828
2832
if isa (slot, TypedSlot)
2829
2833
usetyp = widenconst (slot. typ)
2830
- if isleaftype (usetyp)
2834
+ if isdispatchelem (usetyp)
2831
2835
usetyp = widenconst (slot. typ)
2832
2836
new_slot = alltypes[usetyp]
2833
2837
if ! isa (new_slot, SlotNumber)
@@ -2997,7 +3001,7 @@ function split_struct_alloc!(ctx::AllocOptContext, info, key)
2997
3001
elseif defex. head === :new
2998
3002
typ = widenconst (exprtype (defex, ctx. sv. src, ctx. sv. mod))
2999
3003
# 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
3001
3005
return false
3002
3006
end
3003
3007
si = structinfo_new (ctx, defex, typ)
0 commit comments