Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(533)

Side by Side Diff: runtime/vm/flow_graph_inliner.cc

Issue 1867913004: Specialize instance calls when the call receiver is the method receiver and the method class has a … (Closed) Base URL: git@github.com:dart-lang/sdk.git@master
Patch Set: Created 4 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a 2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file. 3 // BSD-style license that can be found in the LICENSE file.
4 4
5 #include "vm/flow_graph_inliner.h" 5 #include "vm/flow_graph_inliner.h"
6 6
7 #include "vm/aot_optimizer.h" 7 #include "vm/aot_optimizer.h"
8 #include "vm/block_scheduler.h" 8 #include "vm/block_scheduler.h"
9 #include "vm/branch_optimizer.h" 9 #include "vm/branch_optimizer.h"
10 #include "vm/compiler.h" 10 #include "vm/compiler.h"
(...skipping 1196 matching lines...) Expand 10 before | Expand all | Expand 10 after
1207 1207
1208 void InlineInstanceCalls() { 1208 void InlineInstanceCalls() {
1209 const GrowableArray<CallSites::InstanceCallInfo>& call_info = 1209 const GrowableArray<CallSites::InstanceCallInfo>& call_info =
1210 inlining_call_sites_->instance_calls(); 1210 inlining_call_sites_->instance_calls();
1211 TRACE_INLINING(THR_Print(" Polymorphic Instance Calls (%" Pd ")\n", 1211 TRACE_INLINING(THR_Print(" Polymorphic Instance Calls (%" Pd ")\n",
1212 call_info.length())); 1212 call_info.length()));
1213 for (intptr_t call_idx = 0; call_idx < call_info.length(); ++call_idx) { 1213 for (intptr_t call_idx = 0; call_idx < call_info.length(); ++call_idx) {
1214 PolymorphicInstanceCallInstr* call = call_info[call_idx].call; 1214 PolymorphicInstanceCallInstr* call = call_info[call_idx].call;
1215 if (call->with_checks()) { 1215 if (call->with_checks()) {
1216 // PolymorphicInliner introduces deoptimization paths. 1216 // PolymorphicInliner introduces deoptimization paths.
1217 if (!FLAG_polymorphic_with_deopt) { 1217 if (!call->complete() && !FLAG_polymorphic_with_deopt) {
1218 TRACE_INLINING(THR_Print( 1218 TRACE_INLINING(THR_Print(
1219 " => %s\n Bailout: call with checks\n", 1219 " => %s\n Bailout: call with checks\n",
1220 call->instance_call()->function_name().ToCString())); 1220 call->instance_call()->function_name().ToCString()));
1221 continue; 1221 continue;
1222 } 1222 }
1223 const Function& cl = call_info[call_idx].caller(); 1223 const Function& cl = call_info[call_idx].caller();
1224 intptr_t caller_inlining_id = 1224 intptr_t caller_inlining_id =
1225 call_info[call_idx].caller_graph->inlining_id(); 1225 call_info[call_idx].caller_graph->inlining_id();
1226 PolymorphicInliner inliner(this, call, cl, caller_inlining_id); 1226 PolymorphicInliner inliner(this, call, cl, caller_inlining_id);
1227 inliner.Inline(); 1227 inliner.Inline();
(...skipping 396 matching lines...) Expand 10 before | Expand all | Expand 10 after
1624 // at least one branch on the class id. 1624 // at least one branch on the class id.
1625 LoadClassIdInstr* load_cid = 1625 LoadClassIdInstr* load_cid =
1626 new(Z) LoadClassIdInstr(new(Z) Value(receiver)); 1626 new(Z) LoadClassIdInstr(new(Z) Value(receiver));
1627 load_cid->set_ssa_temp_index(owner_->caller_graph()->alloc_ssa_temp_index()); 1627 load_cid->set_ssa_temp_index(owner_->caller_graph()->alloc_ssa_temp_index());
1628 cursor = AppendInstruction(cursor, load_cid); 1628 cursor = AppendInstruction(cursor, load_cid);
1629 for (intptr_t i = 0; i < inlined_variants_.length(); ++i) { 1629 for (intptr_t i = 0; i < inlined_variants_.length(); ++i) {
1630 // 1. Guard the body with a class id check. 1630 // 1. Guard the body with a class id check.
1631 if ((i == (inlined_variants_.length() - 1)) && 1631 if ((i == (inlined_variants_.length() - 1)) &&
1632 non_inlined_variants_.is_empty()) { 1632 non_inlined_variants_.is_empty()) {
1633 // If it is the last variant use a check class id instruction which can 1633 // If it is the last variant use a check class id instruction which can
1634 // deoptimize, followed unconditionally by the body. 1634 // deoptimize, followed unconditionally by the body. Omit the check if
1635 RedefinitionInstr* cid_redefinition = 1635 // we know that we have covered all possible classes.
1636 new RedefinitionInstr(new(Z) Value(load_cid)); 1636 if (!call_->complete()) {
1637 cid_redefinition->set_ssa_temp_index( 1637 RedefinitionInstr* cid_redefinition =
1638 owner_->caller_graph()->alloc_ssa_temp_index()); 1638 new RedefinitionInstr(new(Z) Value(load_cid));
1639 cursor = AppendInstruction(cursor, cid_redefinition); 1639 cid_redefinition->set_ssa_temp_index(
1640 CheckClassIdInstr* check_class_id = new(Z) CheckClassIdInstr( 1640 owner_->caller_graph()->alloc_ssa_temp_index());
1641 new(Z) Value(cid_redefinition), 1641 cursor = AppendInstruction(cursor, cid_redefinition);
1642 inlined_variants_[i].cid, 1642 CheckClassIdInstr* check_class_id = new(Z) CheckClassIdInstr(
1643 call_->deopt_id()); 1643 new(Z) Value(cid_redefinition),
1644 check_class_id->InheritDeoptTarget(zone(), call_); 1644 inlined_variants_[i].cid,
1645 cursor = AppendInstruction(cursor, check_class_id); 1645 call_->deopt_id());
1646 check_class_id->InheritDeoptTarget(zone(), call_);
1647 cursor = AppendInstruction(cursor, check_class_id);
1648 }
1646 1649
1647 // The next instruction is the first instruction of the inlined body. 1650 // The next instruction is the first instruction of the inlined body.
1648 // Handle the two possible cases (unshared and shared subsequent 1651 // Handle the two possible cases (unshared and shared subsequent
1649 // predecessors) separately. 1652 // predecessors) separately.
1650 BlockEntryInstr* callee_entry = inlined_entries_[i]; 1653 BlockEntryInstr* callee_entry = inlined_entries_[i];
1651 if (callee_entry->IsGraphEntry()) { 1654 if (callee_entry->IsGraphEntry()) {
1652 // Unshared. Graft the normal entry on after the check class 1655 // Unshared. Graft the normal entry on after the check class
1653 // instruction. 1656 // instruction.
1654 TargetEntryInstr* target = 1657 TargetEntryInstr* target =
1655 callee_entry->AsGraphEntry()->normal_entry(); 1658 callee_entry->AsGraphEntry()->normal_entry();
(...skipping 112 matching lines...) Expand 10 before | Expand all | Expand 10 after
1768 old_checks.deopt_id(), 1771 old_checks.deopt_id(),
1769 1)); // Number of args tested. 1772 1)); // Number of args tested.
1770 for (intptr_t i = 0; i < non_inlined_variants_.length(); ++i) { 1773 for (intptr_t i = 0; i < non_inlined_variants_.length(); ++i) {
1771 new_checks.AddReceiverCheck(non_inlined_variants_[i].cid, 1774 new_checks.AddReceiverCheck(non_inlined_variants_[i].cid,
1772 *non_inlined_variants_[i].target, 1775 *non_inlined_variants_[i].target,
1773 non_inlined_variants_[i].count); 1776 non_inlined_variants_[i].count);
1774 } 1777 }
1775 PolymorphicInstanceCallInstr* fallback_call = 1778 PolymorphicInstanceCallInstr* fallback_call =
1776 new PolymorphicInstanceCallInstr(call_->instance_call(), 1779 new PolymorphicInstanceCallInstr(call_->instance_call(),
1777 new_checks, 1780 new_checks,
1778 true); // With checks. 1781 true, // With checks.
1782 call_->complete()); // Complete.
Florian Schneider 2016/04/08 22:39:38 This call could avoid the slow path code if you ch
1779 fallback_call->set_ssa_temp_index( 1783 fallback_call->set_ssa_temp_index(
1780 owner_->caller_graph()->alloc_ssa_temp_index()); 1784 owner_->caller_graph()->alloc_ssa_temp_index());
1781 fallback_call->InheritDeoptTarget(zone(), call_); 1785 fallback_call->InheritDeoptTarget(zone(), call_);
1782 ReturnInstr* fallback_return = 1786 ReturnInstr* fallback_return =
1783 new ReturnInstr(call_->instance_call()->token_pos(), 1787 new ReturnInstr(call_->instance_call()->token_pos(),
1784 new Value(fallback_call)); 1788 new Value(fallback_call));
1785 fallback_return->InheritDeoptTargetAfter( 1789 fallback_return->InheritDeoptTargetAfter(
1786 owner_->caller_graph(), 1790 owner_->caller_graph(),
1787 call_, 1791 call_,
1788 fallback_call); 1792 fallback_call);
(...skipping 1074 matching lines...) Expand 10 before | Expand all | Expand 10 after
2863 2867
2864 bool FlowGraphInliner::TryInlineRecognizedMethod(FlowGraph* flow_graph, 2868 bool FlowGraphInliner::TryInlineRecognizedMethod(FlowGraph* flow_graph,
2865 intptr_t receiver_cid, 2869 intptr_t receiver_cid,
2866 const Function& target, 2870 const Function& target,
2867 Instruction* call, 2871 Instruction* call,
2868 Definition* receiver, 2872 Definition* receiver,
2869 TokenPosition token_pos, 2873 TokenPosition token_pos,
2870 const ICData& ic_data, 2874 const ICData& ic_data,
2871 TargetEntryInstr** entry, 2875 TargetEntryInstr** entry,
2872 Definition** last) { 2876 Definition** last) {
2877 if (FLAG_precompiled_mode) {
2878 // The graphs generated below include deopts.
2879 return false;
2880 }
2881
2873 ICData& value_check = ICData::ZoneHandle(Z); 2882 ICData& value_check = ICData::ZoneHandle(Z);
2874 MethodRecognizer::Kind kind = MethodRecognizer::RecognizeKind(target); 2883 MethodRecognizer::Kind kind = MethodRecognizer::RecognizeKind(target);
2875 switch (kind) { 2884 switch (kind) {
2876 // Recognized [] operators. 2885 // Recognized [] operators.
2877 case MethodRecognizer::kImmutableArrayGetIndexed: 2886 case MethodRecognizer::kImmutableArrayGetIndexed:
2878 case MethodRecognizer::kObjectArrayGetIndexed: 2887 case MethodRecognizer::kObjectArrayGetIndexed:
2879 case MethodRecognizer::kGrowableArrayGetIndexed: 2888 case MethodRecognizer::kGrowableArrayGetIndexed:
2880 case MethodRecognizer::kInt8ArrayGetIndexed: 2889 case MethodRecognizer::kInt8ArrayGetIndexed:
2881 case MethodRecognizer::kUint8ArrayGetIndexed: 2890 case MethodRecognizer::kUint8ArrayGetIndexed:
2882 case MethodRecognizer::kUint8ClampedArrayGetIndexed: 2891 case MethodRecognizer::kUint8ClampedArrayGetIndexed:
(...skipping 214 matching lines...) Expand 10 before | Expand all | Expand 10 after
3097 return InlineDoubleOp(flow_graph, Token::kMUL, call, entry, last); 3106 return InlineDoubleOp(flow_graph, Token::kMUL, call, entry, last);
3098 case MethodRecognizer::kDoubleDiv: 3107 case MethodRecognizer::kDoubleDiv:
3099 return InlineDoubleOp(flow_graph, Token::kDIV, call, entry, last); 3108 return InlineDoubleOp(flow_graph, Token::kDIV, call, entry, last);
3100 default: 3109 default:
3101 return false; 3110 return false;
3102 } 3111 }
3103 } 3112 }
3104 3113
3105 3114
3106 } // namespace dart 3115 } // namespace dart
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698