Index: runtime/vm/compiler/backend/inliner.cc |
diff --git a/runtime/vm/compiler/backend/inliner.cc b/runtime/vm/compiler/backend/inliner.cc |
index 50cf3a1428f83f6a1ddcac6a597cb5e3fe1f6a1e..2df2dc009b719d110733082a6208427402de9390 100644 |
--- a/runtime/vm/compiler/backend/inliner.cc |
+++ b/runtime/vm/compiler/backend/inliner.cc |
@@ -456,7 +456,7 @@ class PolymorphicInliner : public ValueObject { |
const Function& caller_function, |
intptr_t caller_inlining_id); |
- void Inline(); |
+ bool Inline(); |
private: |
bool CheckInlinedDuplicate(const Function& target); |
@@ -620,37 +620,59 @@ class CallSiteInliner : public ValueObject { |
bool trace_inlining() const { return inliner_->trace_inlining(); } |
+ int inlining_depth() { return inlining_depth_; } |
+ |
+ struct InliningDecision { |
+ InliningDecision(bool b, const char* r) : value(b), reason(r) {} |
+ bool value; |
+ const char* reason; |
+ static InliningDecision Yes(const char* reason) { |
+ return InliningDecision(true, reason); |
+ } |
+ static InliningDecision No(const char* reason) { |
+ return InliningDecision(false, reason); |
+ } |
+ }; |
+ |
// Inlining heuristics based on Cooper et al. 2008. |
- bool ShouldWeInline(const Function& callee, |
- intptr_t instr_count, |
- intptr_t call_site_count, |
- intptr_t const_arg_count) { |
+ InliningDecision ShouldWeInline(const Function& callee, |
+ intptr_t instr_count, |
+ intptr_t call_site_count, |
+ intptr_t const_arg_count) { |
if (inliner_->AlwaysInline(callee)) { |
- return true; |
+ return InliningDecision::Yes("AlwaysInline"); |
} |
if (inlined_size_ > FLAG_inlining_caller_size_threshold) { |
// Prevent methods becoming humongous and thus slow to compile. |
- return false; |
+ return InliningDecision::No("--inlining-caller-size-threshold"); |
} |
if (const_arg_count > 0) { |
if (instr_count > FLAG_inlining_constant_arguments_max_size_threshold) { |
- return false; |
+ return InliningDecision( |
+ false, "--inlining-constant-arguments-max-size-threshold"); |
} |
} else if (instr_count > FLAG_inlining_callee_size_threshold) { |
- return false; |
+ return InliningDecision::No("--inlining-callee-size-threshold"); |
+ } |
+ int callee_inlining_depth = callee.inlining_depth(); |
+ if (callee_inlining_depth > 0 && callee_inlining_depth + inlining_depth_ > |
+ FLAG_inlining_depth_threshold) { |
+ return InliningDecision::No("--inlining-depth-threshold"); |
} |
// 'instr_count' can be 0 if it was not computed yet. |
if ((instr_count != 0) && (instr_count <= FLAG_inlining_size_threshold)) { |
- return true; |
+ return InliningDecision::Yes("--inlining-size-threshold"); |
} |
if (call_site_count <= FLAG_inlining_callee_call_sites_threshold) { |
- return true; |
+ return InliningDecision::Yes("--inlining-callee-call-sites-threshold"); |
} |
if ((const_arg_count >= FLAG_inlining_constant_arguments_count) && |
(instr_count <= FLAG_inlining_constant_arguments_min_size_threshold)) { |
- return true; |
+ return InliningDecision(true, |
+ "--inlining-constant-arguments-count and " |
+ "inlining-constant-arguments-min-size-threshold"); |
} |
- return false; |
+ return InliningDecision::No("default"); |
} |
void InlineCalls() { |
@@ -685,16 +707,18 @@ class CallSiteInliner : public ValueObject { |
inlining_call_sites_ = call_sites_temp; |
collected_call_sites_->Clear(); |
// Inline call sites at the current depth. |
- InlineInstanceCalls(); |
- InlineStaticCalls(); |
- InlineClosureCalls(); |
- // Increment the inlining depths. Checked before subsequent inlining. |
- ++inlining_depth_; |
- if (inlined_recursive_call_) { |
- ++inlining_recursion_depth_; |
- inlined_recursive_call_ = false; |
+ bool inlined_instance = InlineInstanceCalls(); |
+ bool inlined_statics = InlineStaticCalls(); |
+ bool inlined_closures = InlineClosureCalls(); |
+ if (inlined_instance || inlined_statics || inlined_closures) { |
+ // Increment the inlining depths. Checked before subsequent inlining. |
+ ++inlining_depth_; |
+ if (inlined_recursive_call_) { |
+ ++inlining_recursion_depth_; |
+ inlined_recursive_call_ = false; |
+ } |
+ thread()->CheckForSafepoint(); |
} |
- thread()->CheckForSafepoint(); |
} |
collected_call_sites_ = NULL; |
@@ -751,7 +775,7 @@ class CallSiteInliner : public ValueObject { |
// Do not rely on function type feedback or presence of code to determine |
// if a function was compiled. |
- if (!FLAG_precompiled_mode && !function.was_compiled()) { |
+ if (!FLAG_precompiled_mode && !function.WasCompiled()) { |
TRACE_INLINING(THR_Print(" Bailout: not compiled yet\n")); |
PRINT_INLINING_TREE("Not compiled", &call_data->caller, &function, |
call_data->call); |
@@ -795,16 +819,19 @@ class CallSiteInliner : public ValueObject { |
GrowableArray<Value*>* arguments = call_data->arguments; |
const intptr_t constant_arguments = CountConstants(*arguments); |
- if (!ShouldWeInline(function, function.optimized_instruction_count(), |
- function.optimized_call_site_count(), |
- constant_arguments)) { |
+ InliningDecision decision = ShouldWeInline( |
+ function, function.optimized_instruction_count(), |
+ function.optimized_call_site_count(), constant_arguments); |
+ if (!decision.value) { |
TRACE_INLINING( |
- THR_Print(" Bailout: early heuristics with " |
+ THR_Print(" Bailout: early heuristics (%s) with " |
"code size: %" Pd ", " |
"call sites: %" Pd ", " |
+ "inlining depth of callee: %d, " |
"const args: %" Pd "\n", |
- function.optimized_instruction_count(), |
- function.optimized_call_site_count(), constant_arguments)); |
+ decision.reason, function.optimized_instruction_count(), |
+ function.optimized_call_site_count(), |
+ function.inlining_depth(), constant_arguments)); |
PRINT_INLINING_TREE("Early heuristic", &call_data->caller, &function, |
call_data->call); |
return false; |
@@ -1045,11 +1072,10 @@ class CallSiteInliner : public ValueObject { |
const intptr_t size = function.optimized_instruction_count(); |
const intptr_t call_site_count = function.optimized_call_site_count(); |
- function.set_optimized_instruction_count(size); |
- function.set_optimized_call_site_count(call_site_count); |
- |
// Use heuristics do decide if this call should be inlined. |
- if (!ShouldWeInline(function, size, call_site_count, constants_count)) { |
+ InliningDecision decision = |
+ ShouldWeInline(function, size, call_site_count, constants_count); |
+ if (!decision.value) { |
// If size is larger than all thresholds, don't consider it again. |
if ((size > FLAG_inlining_size_threshold) && |
(call_site_count > FLAG_inlining_callee_call_sites_threshold) && |
@@ -1059,11 +1085,13 @@ class CallSiteInliner : public ValueObject { |
} |
thread()->set_deopt_id(prev_deopt_id); |
TRACE_INLINING( |
- THR_Print(" Bailout: heuristics with " |
+ THR_Print(" Bailout: heuristics (%s) with " |
"code size: %" Pd ", " |
"call sites: %" Pd ", " |
+ "inlining depth of callee: %d, " |
"const args: %" Pd "\n", |
- size, call_site_count, constants_count)); |
+ decision.reason, size, call_site_count, |
+ function.inlining_depth(), constants_count)); |
PRINT_INLINING_TREE("Heuristic fail", &call_data->caller, &function, |
call_data->call); |
return false; |
@@ -1259,7 +1287,8 @@ class CallSiteInliner : public ValueObject { |
return parsed_function; |
} |
- void InlineStaticCalls() { |
+ bool InlineStaticCalls() { |
+ bool inlined = false; |
const GrowableArray<CallSites::StaticCallInfo>& call_info = |
inlining_call_sites_->static_calls(); |
TRACE_INLINING(THR_Print(" Static Calls (%" Pd ")\n", call_info.length())); |
@@ -1287,11 +1316,14 @@ class CallSiteInliner : public ValueObject { |
call_info[call_idx].caller_graph->inlining_id()); |
if (TryInlining(call->function(), call->argument_names(), &call_data)) { |
InlineCall(&call_data); |
+ inlined = true; |
} |
} |
+ return inlined; |
} |
- void InlineClosureCalls() { |
+ bool InlineClosureCalls() { |
+ bool inlined = false; |
const GrowableArray<CallSites::ClosureCallInfo>& call_info = |
inlining_call_sites_->closure_calls(); |
TRACE_INLINING( |
@@ -1333,11 +1365,14 @@ class CallSiteInliner : public ValueObject { |
call_info[call_idx].caller_graph->inlining_id()); |
if (TryInlining(target, call->argument_names(), &call_data)) { |
InlineCall(&call_data); |
+ inlined = true; |
} |
} |
+ return inlined; |
} |
- void InlineInstanceCalls() { |
+ bool InlineInstanceCalls() { |
+ bool inlined = false; |
const GrowableArray<CallSites::InstanceCallInfo>& call_info = |
inlining_call_sites_->instance_calls(); |
TRACE_INLINING(THR_Print(" Polymorphic Instance Calls (%" Pd ")\n", |
@@ -1355,8 +1390,9 @@ class CallSiteInliner : public ValueObject { |
intptr_t caller_inlining_id = |
call_info[call_idx].caller_graph->inlining_id(); |
PolymorphicInliner inliner(this, call, cl, caller_inlining_id); |
- inliner.Inline(); |
+ if (inliner.Inline()) inlined = true; |
} |
+ return inlined; |
} |
bool AdjustForOptionalParameters(const ParsedFunction& parsed_function, |
@@ -1911,7 +1947,7 @@ bool PolymorphicInliner::trace_inlining() const { |
return owner_->trace_inlining(); |
} |
-void PolymorphicInliner::Inline() { |
+bool PolymorphicInliner::Inline() { |
ASSERT(&variants_ == &call_->targets_); |
intptr_t total = call_->total_call_count(); |
@@ -1984,16 +2020,13 @@ void PolymorphicInliner::Inline() { |
} |
// If there are no inlined variants, leave the call in place. |
- if (inlined_variants_.is_empty()) return; |
+ if (inlined_variants_.is_empty()) return false; |
// Now build a decision tree (a DAG because of shared inline variants) and |
// inline it at the call site. |
TargetEntryInstr* entry = BuildDecisionGraph(); |
exit_collector_->ReplaceCall(entry); |
-} |
- |
-static uint16_t ClampUint16(intptr_t v) { |
- return (v > 0xFFFF) ? 0xFFFF : static_cast<uint16_t>(v); |
+ return true; |
} |
static bool ShouldTraceInlining(FlowGraph* flow_graph) { |
@@ -2026,9 +2059,8 @@ void FlowGraphInliner::CollectGraphInfo(FlowGraph* flow_graph, bool force) { |
GraphInfoCollector info; |
info.Collect(*flow_graph); |
- function.set_optimized_instruction_count( |
- ClampUint16(info.instruction_count())); |
- function.set_optimized_call_site_count(ClampUint16(info.call_site_count())); |
+ function.SetOptimizedInstructionCountClamped(info.instruction_count()); |
+ function.SetOptimizedCallSiteCountClamped(info.call_site_count()); |
} |
} |
@@ -2090,7 +2122,7 @@ bool FlowGraphInliner::AlwaysInline(const Function& function) { |
return MethodRecognizer::AlwaysInline(function); |
} |
-void FlowGraphInliner::Inline() { |
+int FlowGraphInliner::Inline() { |
// Collect graph info and store it on the function. |
// We might later use it for an early bailout from the inlining. |
CollectGraphInfo(flow_graph_); |
@@ -2098,7 +2130,7 @@ void FlowGraphInliner::Inline() { |
const Function& top = flow_graph_->function(); |
if ((FLAG_inlining_filter != NULL) && |
(strstr(top.ToFullyQualifiedCString(), FLAG_inlining_filter) == NULL)) { |
- return; |
+ return 0; |
} |
if (trace_inlining()) { |
@@ -2139,6 +2171,7 @@ void FlowGraphInliner::Inline() { |
} |
} |
} |
+ return inliner.inlining_depth(); |
} |
intptr_t FlowGraphInliner::NextInlineId(const Function& function, |