From 9352745102e0c22d62aabc882f836c2bc69886e9 Mon Sep 17 00:00:00 2001 From: Keno Fischer Date: Fri, 9 Sep 2022 22:57:10 +0000 Subject: [PATCH] Improve inference for ismutable with missing sparam We could already infer `ismutable(RefValue{T})` if we knew what `T` was at inference time. However, the mutable does of course not change depending on what `T` is, so fix that up by adding an appropriate special case in `_getfield_tfunc`. --- base/compiler/tfuncs.jl | 95 +++++++++++++++++++++++++------------- test/compiler/inference.jl | 1 + test/compiler/inline.jl | 4 ++ 3 files changed, 67 insertions(+), 33 deletions(-) diff --git a/base/compiler/tfuncs.jl b/base/compiler/tfuncs.jl index f84f4c767180b..a5b55c6fc3a8e 100644 --- a/base/compiler/tfuncs.jl +++ b/base/compiler/tfuncs.jl @@ -25,6 +25,7 @@ function find_tfunc(@nospecialize f) end const DATATYPE_TYPES_FIELDINDEX = fieldindex(DataType, :types) +const DATATYPE_NAME_FIELDINDEX = fieldindex(DataType, :name) ########## # tfuncs # @@ -823,7 +824,11 @@ function getfield_nothrow(@nospecialize(s00), @nospecialize(name), boundscheck:: if isa(s, Union) return getfield_nothrow(rewrap_unionall(s.a, s00), name, boundscheck) && getfield_nothrow(rewrap_unionall(s.b, s00), name, boundscheck) - elseif isa(s, DataType) + elseif isType(s) + sv = s.parameters[1] + s = s0 = typeof(sv) + end + if isa(s, DataType) # Can't say anything about abstract types isabstracttype(s) && return false s.name.atomicfields == C_NULL || return false # TODO: currently we're only testing for ordering === :not_atomic @@ -863,15 +868,40 @@ function getfield_tfunc(s00, name, order, boundscheck) return getfield_tfunc(s00, name) end getfield_tfunc(@nospecialize(s00), @nospecialize(name)) = _getfield_tfunc(s00, name, false) + +function _getfield_fieldindex(@nospecialize(s), name::Const) + nv = name.val + if isa(nv, Symbol) + nv = fieldindex(s, nv, false) + end + if isa(nv, Int) + return nv + end + return nothing +end + +function _getfield_tfunc_const(@nospecialize(sv), name::Const, setfield::Bool) + if isa(name, Const) + nv = _getfield_fieldindex(typeof(sv), name) + nv === nothing && return Bottom + if isa(sv, DataType) && nv == DATATYPE_TYPES_FIELDINDEX && isdefined(sv, nv) + return Const(getfield(sv, nv)) + end + if isconst(typeof(sv), nv) + if isdefined(sv, nv) + return Const(getfield(sv, nv)) + end + return Union{} + end + end + return nothing +end + function _getfield_tfunc(@nospecialize(s00), @nospecialize(name), setfield::Bool) if isa(s00, Conditional) return Bottom # Bool has no fields - elseif isa(s00, Const) || isconstType(s00) - if !isa(s00, Const) - sv = s00.parameters[1] - else - sv = s00.val - end + elseif isa(s00, Const) + sv = s00.val if isa(name, Const) nv = name.val if isa(sv, Module) @@ -881,31 +911,15 @@ function _getfield_tfunc(@nospecialize(s00), @nospecialize(name), setfield::Bool end return Bottom end - if isa(nv, Symbol) - nv = fieldindex(typeof(sv), nv, false) - end - if !isa(nv, Int) - return Bottom - end - if isa(sv, DataType) && nv == DATATYPE_TYPES_FIELDINDEX && isdefined(sv, nv) - return Const(getfield(sv, nv)) - end - if isconst(typeof(sv), nv) - if isdefined(sv, nv) - return Const(getfield(sv, nv)) - end - return Union{} - end + r = _getfield_tfunc_const(sv, name, setfield) + r !== nothing && return r end s = typeof(sv) elseif isa(s00, PartialStruct) s = widenconst(s00) sty = unwrap_unionall(s)::DataType if isa(name, Const) - nv = name.val - if isa(nv, Symbol) - nv = fieldindex(sty, nv, false) - end + nv = _getfield_fieldindex(sty, name) if isa(nv, Int) && 1 <= nv <= length(s00.fields) return unwrapva(s00.fields[nv]) end @@ -917,6 +931,26 @@ function _getfield_tfunc(@nospecialize(s00), @nospecialize(name), setfield::Bool return tmerge(_getfield_tfunc(rewrap_unionall(s.a, s00), name, setfield), _getfield_tfunc(rewrap_unionall(s.b, s00), name, setfield)) end + if isType(s) + if isconstType(s) + sv = s00.parameters[1] + if isa(name, Const) + r = _getfield_tfunc_const(sv, name, setfield) + r !== nothing && return r + end + s = typeof(sv) + else + sv = s.parameters[1] + if isa(sv, DataType) && isa(name, Const) && (!isType(sv) && sv !== Core.TypeofBottom) + nv = _getfield_fieldindex(DataType, name) + if nv == DATATYPE_NAME_FIELDINDEX + # N.B. This doesn't work in general, because + return Const(sv.name) + end + s = DataType + end + end + end isa(s, DataType) || return Any isabstracttype(s) && return Any if s <: Tuple && !hasintersect(widenconst(name), Int) @@ -972,13 +1006,8 @@ function _getfield_tfunc(@nospecialize(s00), @nospecialize(name), setfield::Bool end return t end - fld = name.val - if isa(fld, Symbol) - fld = fieldindex(s, fld, false) - end - if !isa(fld, Int) - return Bottom - end + fld = _getfield_fieldindex(s, name) + fld === nothing && return Bottom if s <: Tuple && fld >= nf && isvarargtype(ftypes[nf]) return rewrap_unionall(unwrapva(ftypes[nf]), s00) end diff --git a/test/compiler/inference.jl b/test/compiler/inference.jl index be0ba7c094bbe..65d9989a6428c 100644 --- a/test/compiler/inference.jl +++ b/test/compiler/inference.jl @@ -1588,6 +1588,7 @@ g23024(TT::Tuple{DataType}) = f23024(TT[1], v23024) @test g23024((UInt8,)) === 2 @test !Core.Compiler.isconstType(Type{typeof(Union{})}) # could be Core.TypeofBottom or Type{Union{}} at runtime +@test !isa(Core.Compiler.getfield_tfunc(Type{Core.TypeofBottom}, Core.Compiler.Const(:name)), Core.Compiler.Const) @test Base.return_types(supertype, (Type{typeof(Union{})},)) == Any[Any] # issue #23685 diff --git a/test/compiler/inline.jl b/test/compiler/inline.jl index 63e7efc2623ca..7feb628a7f789 100644 --- a/test/compiler/inline.jl +++ b/test/compiler/inline.jl @@ -1496,3 +1496,7 @@ call_twice_sitofp(x::Int) = twice_sitofp(x, 2) let src = code_typed1(call_twice_sitofp, (Int,)) @test count(iscall((src, Base.sitofp)), src.code) == 1 end + +# Test getfield modeling of Type{Ref{_A}} where _A +@test Core.Compiler.getfield_tfunc(Type, Core.Compiler.Const(:parameters)) !== Union{} +@test fully_eliminated(Base.ismutable, Tuple{Base.RefValue})