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

Side by Side Diff: runtime/vm/aot_optimizer.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/aot_optimizer.h" 5 #include "vm/aot_optimizer.h"
6 6
7 #include "vm/bit_vector.h" 7 #include "vm/bit_vector.h"
8 #include "vm/branch_optimizer.h" 8 #include "vm/branch_optimizer.h"
9 #include "vm/cha.h" 9 #include "vm/cha.h"
10 #include "vm/compiler.h" 10 #include "vm/compiler.h"
(...skipping 263 matching lines...) Expand 10 before | Expand all | Expand 10 after
274 if (receiver_cid == kDynamicCid) { 274 if (receiver_cid == kDynamicCid) {
275 return; // No information about receiver was infered. 275 return; // No information about receiver was infered.
276 } 276 }
277 277
278 const ICData& ic_data = TrySpecializeICData(call->ic_data(), receiver_cid); 278 const ICData& ic_data = TrySpecializeICData(call->ic_data(), receiver_cid);
279 if (ic_data.raw() == call->ic_data().raw()) { 279 if (ic_data.raw() == call->ic_data().raw()) {
280 // No specialization. 280 // No specialization.
281 return; 281 return;
282 } 282 }
283 283
284 const bool with_checks = false;
285 PolymorphicInstanceCallInstr* specialized = 284 PolymorphicInstanceCallInstr* specialized =
286 new(Z) PolymorphicInstanceCallInstr(call->instance_call(), 285 new(Z) PolymorphicInstanceCallInstr(call->instance_call(),
287 ic_data, 286 ic_data,
288 with_checks); 287 /* with_checks = */ false,
288 /* complete = */ false);
289 call->ReplaceWith(specialized, current_iterator()); 289 call->ReplaceWith(specialized, current_iterator());
290 } 290 }
291 291
292 292
293 static BinarySmiOpInstr* AsSmiShiftLeftInstruction(Definition* d) { 293 static BinarySmiOpInstr* AsSmiShiftLeftInstruction(Definition* d) {
294 BinarySmiOpInstr* instr = d->AsBinarySmiOp(); 294 BinarySmiOpInstr* instr = d->AsBinarySmiOp();
295 if ((instr != NULL) && (instr->op_kind() == Token::kSHL)) { 295 if ((instr != NULL) && (instr->op_kind() == Token::kSHL)) {
296 return instr; 296 return instr;
297 } 297 }
298 return NULL; 298 return NULL;
(...skipping 2135 matching lines...) Expand 10 before | Expand all | Expand 10 after
2434 has_one_target = !polymorphic_target; 2434 has_one_target = !polymorphic_target;
2435 } 2435 }
2436 2436
2437 if (has_one_target) { 2437 if (has_one_target) {
2438 RawFunction::Kind function_kind = 2438 RawFunction::Kind function_kind =
2439 Function::Handle(Z, unary_checks.GetTargetAt(0)).kind(); 2439 Function::Handle(Z, unary_checks.GetTargetAt(0)).kind();
2440 if (!flow_graph()->InstanceCallNeedsClassCheck( 2440 if (!flow_graph()->InstanceCallNeedsClassCheck(
2441 instr, function_kind)) { 2441 instr, function_kind)) {
2442 PolymorphicInstanceCallInstr* call = 2442 PolymorphicInstanceCallInstr* call =
2443 new(Z) PolymorphicInstanceCallInstr(instr, unary_checks, 2443 new(Z) PolymorphicInstanceCallInstr(instr, unary_checks,
2444 /* with_checks = */ false); 2444 /* with_checks = */ false,
2445 /* complete = */ true);
2445 instr->ReplaceWith(call, current_iterator()); 2446 instr->ReplaceWith(call, current_iterator());
2446 return; 2447 return;
2447 } 2448 }
2448 } 2449 }
2449 switch (instr->token_kind()) { 2450 switch (instr->token_kind()) {
2450 case Token::kEQ: 2451 case Token::kEQ:
2451 case Token::kLT: 2452 case Token::kLT:
2452 case Token::kLTE: 2453 case Token::kLTE:
2453 case Token::kGT: 2454 case Token::kGT:
2454 case Token::kGTE: 2455 case Token::kGTE:
(...skipping 52 matching lines...) Expand 10 before | Expand all | Expand 10 after
2507 // Create fake IC data with the resolved target. 2508 // Create fake IC data with the resolved target.
2508 const ICData& ic_data = ICData::Handle( 2509 const ICData& ic_data = ICData::Handle(
2509 ICData::New(flow_graph_->function(), 2510 ICData::New(flow_graph_->function(),
2510 instr->function_name(), 2511 instr->function_name(),
2511 args_desc_array, 2512 args_desc_array,
2512 Thread::kNoDeoptId, 2513 Thread::kNoDeoptId,
2513 /* args_tested = */ 1)); 2514 /* args_tested = */ 1));
2514 ic_data.AddReceiverCheck(receiver_class.id(), function); 2515 ic_data.AddReceiverCheck(receiver_class.id(), function);
2515 PolymorphicInstanceCallInstr* call = 2516 PolymorphicInstanceCallInstr* call =
2516 new(Z) PolymorphicInstanceCallInstr(instr, ic_data, 2517 new(Z) PolymorphicInstanceCallInstr(instr, ic_data,
2517 /* with_checks = */ false); 2518 /* with_checks = */ false,
2519 /* complete = */ true);
2518 instr->ReplaceWith(call, current_iterator()); 2520 instr->ReplaceWith(call, current_iterator());
2519 return; 2521 return;
2520 } 2522 }
2521 } 2523 }
2522 } 2524 }
2523 2525
2526 Definition* callee_receiver = instr->ArgumentAt(0);
2527 const Function& function = flow_graph_->function();
2528 if (function.IsDynamicFunction() &&
2529 flow_graph_->IsReceiver(callee_receiver)) {
2530 // Call receiver is method receiver.
2531 Class& receiver_class = Class::Handle(Z, function.Owner());
2532 GrowableArray<intptr_t> class_ids(6);
2533 if (thread()->cha()->ConcreteSubclasses(receiver_class, &class_ids)) {
2534 if (class_ids.length() <= 5) {
srdjan 2016/04/11 18:10:42 How about parametrizing 5 (with a flag), or at lea
rmacnak 2016/04/11 19:33:42 Added FLAG_max_exhaustive_polymorphic_checks
2535 if (FLAG_trace_cha) {
2536 THR_Print(" **(CHA) Only %" Pd " concrete subclasses of %s for %s\n",
2537 class_ids.length(),
2538 receiver_class.ToCString(),
2539 instr->function_name().ToCString());
2540 }
2541
2542 const Array& args_desc_array = Array::Handle(Z,
2543 ArgumentsDescriptor::New(instr->ArgumentCount(),
2544 instr->argument_names()));
2545 ArgumentsDescriptor args_desc(args_desc_array);
2546
2547 const ICData& ic_data = ICData::Handle(
2548 ICData::New(function,
2549 instr->function_name(),
2550 args_desc_array,
2551 Thread::kNoDeoptId,
2552 /* args_tested = */ 1));
2553
2554 Function& target = Function::Handle(Z);
2555 Class& cls = Class::Handle(Z);
2556 bool includes_dispatcher_case = false;
2557 for (intptr_t i = 0; i < class_ids.length(); i++) {
2558 intptr_t cid = class_ids[i];
2559 cls = isolate()->class_table()->At(cid);
2560 target = Resolver::ResolveDynamicForReceiverClass(
2561 cls,
2562 instr->function_name(),
2563 args_desc);
2564 if (target.IsNull()) {
2565 // noSuchMethod, call through getter or closurization
2566 includes_dispatcher_case = true;
2567 } else {
2568 ic_data.AddReceiverCheck(cid, target);
2569 }
2570 }
2571 if (!includes_dispatcher_case && (ic_data.NumberOfChecks() > 0)) {
2572 PolymorphicInstanceCallInstr* call =
Florian Schneider 2016/04/11 17:42:25 Could you change IL printing for PolymorphicInstan
rmacnak 2016/04/11 19:33:42 Done.
2573 new(Z) PolymorphicInstanceCallInstr(instr, ic_data,
2574 /* with_checks = */ true,
2575 /* complete = */ true);
2576 instr->ReplaceWith(call, current_iterator());
2577 return;
2578 }
2579 }
2580 }
2581 }
2582
2524 // More than one targets. Generate generic polymorphic call without 2583 // More than one targets. Generate generic polymorphic call without
2525 // deoptimization. 2584 // deoptimization.
2526 if (instr->ic_data()->NumberOfUsedChecks() > 0) { 2585 if (instr->ic_data()->NumberOfUsedChecks() > 0) {
2527 ASSERT(!FLAG_polymorphic_with_deopt); 2586 ASSERT(!FLAG_polymorphic_with_deopt);
2528 // OK to use checks with PolymorphicInstanceCallInstr since no 2587 // OK to use checks with PolymorphicInstanceCallInstr since no
2529 // deoptimization is allowed. 2588 // deoptimization is allowed.
2530 PolymorphicInstanceCallInstr* call = 2589 PolymorphicInstanceCallInstr* call =
2531 new(Z) PolymorphicInstanceCallInstr(instr, unary_checks, 2590 new(Z) PolymorphicInstanceCallInstr(instr, unary_checks,
2532 /* with_checks = */ true); 2591 /* with_checks = */ true,
2592 /* complete = */ false);
2533 instr->ReplaceWith(call, current_iterator()); 2593 instr->ReplaceWith(call, current_iterator());
2534 return; 2594 return;
2535 } 2595 }
2536 } 2596 }
2537 2597
2538 2598
2539 void AotOptimizer::VisitStaticCall(StaticCallInstr* call) { 2599 void AotOptimizer::VisitStaticCall(StaticCallInstr* call) {
2540 if (!CanUnboxDouble()) { 2600 if (!CanUnboxDouble()) {
2541 return; 2601 return;
2542 } 2602 }
(...skipping 214 matching lines...) Expand 10 before | Expand all | Expand 10 after
2757 2817
2758 // Discard the environment from the original instruction because the store 2818 // Discard the environment from the original instruction because the store
2759 // can't deoptimize. 2819 // can't deoptimize.
2760 instr->RemoveEnvironment(); 2820 instr->RemoveEnvironment();
2761 ReplaceCall(instr, store); 2821 ReplaceCall(instr, store);
2762 return true; 2822 return true;
2763 } 2823 }
2764 2824
2765 2825
2766 } // namespace dart 2826 } // namespace dart
OLDNEW
« no previous file with comments | « runtime/lib/string_patch.dart ('k') | runtime/vm/cha.h » ('j') | runtime/vm/cha.h » ('J')

Powered by Google App Engine
This is Rietveld 408576698