Skip to content

Commit

Permalink
Fix getfield_tfunc for constant TypeName (#37443)
Browse files Browse the repository at this point in the history
The field should be either `Symbol` or `Int`. Ref #37423

Unlike #37423, in additional to worse type info in inference,
the missing type check here can actually cause type inference error
due to errors in user code.
  • Loading branch information
yuyichao authored Sep 8, 2020
1 parent 1e6d771 commit 1378bb6
Show file tree
Hide file tree
Showing 2 changed files with 25 additions and 3 deletions.
6 changes: 3 additions & 3 deletions base/compiler/tfuncs.jl
Original file line number Diff line number Diff line change
Expand Up @@ -745,6 +745,9 @@ function getfield_tfunc(@nospecialize(s00), @nospecialize(name))
end
if isa(name, Const)
nv = name.val
if !(isa(nv,Symbol) || isa(nv,Int))
return Bottom
end
if isa(sv, UnionAll)
if nv === :var || nv === 1
return Const(sv.var)
Expand All @@ -771,9 +774,6 @@ function getfield_tfunc(@nospecialize(s00), @nospecialize(name))
if isa(sv, Module) && isa(nv, Symbol)
return abstract_eval_global(sv, nv)
end
if !(isa(nv,Symbol) || isa(nv,Int))
return Bottom
end
if (isa(sv, SimpleVector) || !ismutable(sv)) && isdefined(sv, nv)
return Const(getfield(sv, nv))
end
Expand Down
22 changes: 22 additions & 0 deletions test/compiler/inference.jl
Original file line number Diff line number Diff line change
Expand Up @@ -2751,3 +2751,25 @@ f_generator_splat(t::Tuple) = tuple((identity(l) for l in t)...)
@test !Core.Compiler.sizeof_nothrow(UnionAll)

@test Base.return_types(Expr) == Any[Expr]

# Use a global constant to rely less on unrelated constant propagation
const const_int32_typename = Int32.name
# Check constant propagation for field of constant `TypeName`
# works for both valid and invalid field names. (Ref #37443)
getfield_const_typename_good1() = getfield(const_int32_typename, 1)
getfield_const_typename_good2() = getfield(const_int32_typename, :name)
getfield_const_typename_bad1() = getfield(const_int32_typename, 0x1)
@eval getfield_const_typename_bad2() = getfield(const_int32_typename, $(()))
for goodf in [getfield_const_typename_good1, getfield_const_typename_good2]
local goodf
local code = code_typed(goodf, Tuple{})[1].first.code
@test code[1] === Core.ReturnNode(QuoteNode(:Int32))
@test goodf() === :Int32
end
for badf in [getfield_const_typename_bad1, getfield_const_typename_bad2]
local badf
local code = code_typed(badf, Tuple{})[1].first.code
@test Meta.isexpr(code[1], :call)
@test code[end] === Core.ReturnNode()
@test_throws TypeError badf()
end

0 comments on commit 1378bb6

Please sign in to comment.