Index: runtime/vm/flow_graph_inliner.cc |
diff --git a/runtime/vm/flow_graph_inliner.cc b/runtime/vm/flow_graph_inliner.cc |
index f2d94b3307d3bb7ae9fc9d9233d321b6f049fd13..47201026280c8c2e399a3ec3a5043df40effa8d2 100644 |
--- a/runtime/vm/flow_graph_inliner.cc |
+++ b/runtime/vm/flow_graph_inliner.cc |
@@ -1214,7 +1214,7 @@ class CallSiteInliner : public ValueObject { |
PolymorphicInstanceCallInstr* call = call_info[call_idx].call; |
if (call->with_checks()) { |
// PolymorphicInliner introduces deoptimization paths. |
- if (!FLAG_polymorphic_with_deopt) { |
+ if (!call->complete() && !FLAG_polymorphic_with_deopt) { |
TRACE_INLINING(THR_Print( |
" => %s\n Bailout: call with checks\n", |
call->instance_call()->function_name().ToCString())); |
@@ -1631,18 +1631,21 @@ TargetEntryInstr* PolymorphicInliner::BuildDecisionGraph() { |
if ((i == (inlined_variants_.length() - 1)) && |
non_inlined_variants_.is_empty()) { |
// If it is the last variant use a check class id instruction which can |
- // deoptimize, followed unconditionally by the body. |
- RedefinitionInstr* cid_redefinition = |
- new RedefinitionInstr(new(Z) Value(load_cid)); |
- cid_redefinition->set_ssa_temp_index( |
- owner_->caller_graph()->alloc_ssa_temp_index()); |
- cursor = AppendInstruction(cursor, cid_redefinition); |
- CheckClassIdInstr* check_class_id = new(Z) CheckClassIdInstr( |
- new(Z) Value(cid_redefinition), |
- inlined_variants_[i].cid, |
- call_->deopt_id()); |
- check_class_id->InheritDeoptTarget(zone(), call_); |
- cursor = AppendInstruction(cursor, check_class_id); |
+ // deoptimize, followed unconditionally by the body. Omit the check if |
+ // we know that we have covered all possible classes. |
+ if (!call_->complete()) { |
+ RedefinitionInstr* cid_redefinition = |
+ new RedefinitionInstr(new(Z) Value(load_cid)); |
+ cid_redefinition->set_ssa_temp_index( |
+ owner_->caller_graph()->alloc_ssa_temp_index()); |
+ cursor = AppendInstruction(cursor, cid_redefinition); |
+ CheckClassIdInstr* check_class_id = new(Z) CheckClassIdInstr( |
+ new(Z) Value(cid_redefinition), |
+ inlined_variants_[i].cid, |
+ call_->deopt_id()); |
+ check_class_id->InheritDeoptTarget(zone(), call_); |
+ cursor = AppendInstruction(cursor, check_class_id); |
+ } |
// The next instruction is the first instruction of the inlined body. |
// Handle the two possible cases (unshared and shared subsequent |
@@ -1775,7 +1778,8 @@ TargetEntryInstr* PolymorphicInliner::BuildDecisionGraph() { |
PolymorphicInstanceCallInstr* fallback_call = |
new PolymorphicInstanceCallInstr(call_->instance_call(), |
new_checks, |
- true); // With checks. |
+ true, // With checks. |
+ call_->complete()); // Complete. |
Florian Schneider
2016/04/08 22:39:38
This call could avoid the slow path code if you ch
|
fallback_call->set_ssa_temp_index( |
owner_->caller_graph()->alloc_ssa_temp_index()); |
fallback_call->InheritDeoptTarget(zone(), call_); |
@@ -2870,6 +2874,11 @@ bool FlowGraphInliner::TryInlineRecognizedMethod(FlowGraph* flow_graph, |
const ICData& ic_data, |
TargetEntryInstr** entry, |
Definition** last) { |
+ if (FLAG_precompiled_mode) { |
+ // The graphs generated below include deopts. |
+ return false; |
+ } |
+ |
ICData& value_check = ICData::ZoneHandle(Z); |
MethodRecognizer::Kind kind = MethodRecognizer::RecognizeKind(target); |
switch (kind) { |