Skip to content

Commit e3b681c

Browse files
authored
inference: always use const-prop'ed result (#44001)
Previously, for `invoke`/opaque closure callsite, we use constant-prop'ed method body only when the inferred return type gets strictly improved by const-prop. But since the inliner is now able to inline the method body from `InferenceResult`, I believe it is always better to use method body shaped up by const-prop' no matter if the return type is improved (as we already do for ordinal call sites). > use constant prop' result even when the return type doesn't get refined ```julia const Gx = Ref{Any}() Base.@constprop :aggressive function conditional_escape!(cnd, x) if cnd Gx[] = x end return nothing end @test fully_eliminated((String,)) do x Base.@invoke conditional_escape!(false::Any, x::Any) end ```
1 parent b405562 commit e3b681c

File tree

2 files changed

+20
-20
lines changed

2 files changed

+20
-20
lines changed

base/compiler/abstractinterpretation.jl

+8-20
Original file line numberDiff line numberDiff line change
@@ -85,10 +85,7 @@ function abstract_call_gf_by_type(interp::AbstractInterpreter, @nospecialize(f),
8585
this_arginfo = ArgInfo(fargs, this_argtypes)
8686
const_result = abstract_call_method_with_const_args(interp, result, f, this_arginfo, match, sv, false)
8787
if const_result !== nothing
88-
const_rt, const_result = const_result
89-
if const_rt !== rt && const_rt rt
90-
rt = const_rt
91-
end
88+
rt, const_result = const_result
9289
end
9390
push!(const_results, const_result)
9491
if const_result !== nothing
@@ -125,10 +122,7 @@ function abstract_call_gf_by_type(interp::AbstractInterpreter, @nospecialize(f),
125122
this_arginfo = ArgInfo(fargs, this_argtypes)
126123
const_result = abstract_call_method_with_const_args(interp, result, f, this_arginfo, match, sv, false)
127124
if const_result !== nothing
128-
const_this_rt, const_result = const_result
129-
if const_this_rt !== this_rt && const_this_rt this_rt
130-
this_rt = const_this_rt
131-
end
125+
this_rt, const_result = const_result
132126
end
133127
push!(const_results, const_result)
134128
if const_result !== nothing
@@ -622,7 +616,7 @@ function abstract_call_method_with_const_args(interp::AbstractInterpreter, resul
622616
# if constant inference hits a cycle, just bail out
623617
isa(result, InferenceState) && return nothing
624618
add_backedge!(mi, sv)
625-
return result, inf_result
619+
return Pair{Any,InferenceResult}(result, inf_result)
626620
end
627621

628622
# if there's a possibility we could get a better result (hopefully without doing too much work)
@@ -1364,12 +1358,9 @@ function abstract_invoke(interp::AbstractInterpreter, (; fargs, argtypes)::ArgIn
13641358
# end
13651359
const_result = abstract_call_method_with_const_args(interp, result, singleton_type(ft′), arginfo, match, sv, false)
13661360
if const_result !== nothing
1367-
const_rt, const_result = const_result
1368-
if const_rt !== rt && const_rt rt
1369-
rt, res = const_rt, const_result
1370-
end
1361+
rt, const_result = const_result
13711362
end
1372-
return CallMeta(from_interprocedural!(rt, sv, arginfo, sig), InvokeCallInfo(match, res))
1363+
return CallMeta(from_interprocedural!(rt, sv, arginfo, sig), InvokeCallInfo(match, const_result))
13731364
end
13741365

13751366
function invoke_rewrite(xs::Vector{Any})
@@ -1491,18 +1482,15 @@ function abstract_call_opaque_closure(interp::AbstractInterpreter, closure::Part
14911482
tt = closure.typ
14921483
sigT = (unwrap_unionall(tt)::DataType).parameters[1]
14931484
match = MethodMatch(sig, Core.svec(), closure.source, sig <: rewrap_unionall(sigT, tt))
1494-
res = nothing
1485+
const_result = nothing
14951486
if !result.edgecycle
14961487
const_result = abstract_call_method_with_const_args(interp, result, closure,
14971488
arginfo, match, sv, closure.isva)
14981489
if const_result !== nothing
1499-
const_rettype, const_result = const_result
1500-
if const_rettype rt
1501-
rt, res = const_rettype, const_result
1502-
end
1490+
rt, const_result = const_result
15031491
end
15041492
end
1505-
info = OpaqueClosureCallInfo(match, res)
1493+
info = OpaqueClosureCallInfo(match, const_result)
15061494
return CallMeta(from_interprocedural!(rt, sv, arginfo, match.spec_types), info)
15071495
end
15081496

test/compiler/inline.jl

+12
Original file line numberDiff line numberDiff line change
@@ -895,3 +895,15 @@ end
895895
# sqrt not considered volatile
896896
f_sqrt() = sqrt(2)
897897
@test fully_eliminated(f_sqrt, Tuple{})
898+
899+
# use constant prop' result even when the return type doesn't get refined
900+
const Gx = Ref{Any}()
901+
Base.@constprop :aggressive function conditional_escape!(cnd, x)
902+
if cnd
903+
Gx[] = x
904+
end
905+
return nothing
906+
end
907+
@test fully_eliminated((String,)) do x
908+
Base.@invoke conditional_escape!(false::Any, x::Any)
909+
end

0 commit comments

Comments
 (0)