| OLD | NEW |
| 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/globals.h" // Needed here to get TARGET_ARCH_X64. | 5 #include "vm/globals.h" // Needed here to get TARGET_ARCH_X64. |
| 6 #if defined(TARGET_ARCH_X64) | 6 #if defined(TARGET_ARCH_X64) |
| 7 | 7 |
| 8 #include "vm/flow_graph_compiler.h" | 8 #include "vm/flow_graph_compiler.h" |
| 9 | 9 |
| 10 #include "vm/ast_printer.h" | 10 #include "vm/ast_printer.h" |
| (...skipping 1199 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1210 LocationSummary* locs) { | 1210 LocationSummary* locs) { |
| 1211 ASSERT(Array::Handle(zone(), ic_data.arguments_descriptor()).Length() > 0); | 1211 ASSERT(Array::Handle(zone(), ic_data.arguments_descriptor()).Length() > 0); |
| 1212 __ LoadUniqueObject(RBX, ic_data); | 1212 __ LoadUniqueObject(RBX, ic_data); |
| 1213 GenerateDartCall(deopt_id, token_pos, stub_entry, RawPcDescriptors::kIcCall, | 1213 GenerateDartCall(deopt_id, token_pos, stub_entry, RawPcDescriptors::kIcCall, |
| 1214 locs); | 1214 locs); |
| 1215 __ Drop(argument_count, RCX); | 1215 __ Drop(argument_count, RCX); |
| 1216 } | 1216 } |
| 1217 | 1217 |
| 1218 | 1218 |
| 1219 void FlowGraphCompiler::EmitMegamorphicInstanceCall( | 1219 void FlowGraphCompiler::EmitMegamorphicInstanceCall( |
| 1220 const ICData& ic_data, | 1220 const String& name, |
| 1221 const Array& arguments_descriptor, |
| 1221 intptr_t argument_count, | 1222 intptr_t argument_count, |
| 1222 intptr_t deopt_id, | 1223 intptr_t deopt_id, |
| 1223 TokenPosition token_pos, | 1224 TokenPosition token_pos, |
| 1224 LocationSummary* locs, | 1225 LocationSummary* locs, |
| 1225 intptr_t try_index, | 1226 intptr_t try_index, |
| 1226 intptr_t slow_path_argument_count) { | 1227 intptr_t slow_path_argument_count) { |
| 1227 const String& name = String::Handle(zone(), ic_data.target_name()); | |
| 1228 const Array& arguments_descriptor = | |
| 1229 Array::ZoneHandle(zone(), ic_data.arguments_descriptor()); | |
| 1230 ASSERT(!arguments_descriptor.IsNull() && (arguments_descriptor.Length() > 0)); | 1228 ASSERT(!arguments_descriptor.IsNull() && (arguments_descriptor.Length() > 0)); |
| 1231 const MegamorphicCache& cache = MegamorphicCache::ZoneHandle( | 1229 const MegamorphicCache& cache = MegamorphicCache::ZoneHandle( |
| 1232 zone(), | 1230 zone(), |
| 1233 MegamorphicCacheTable::Lookup(isolate(), name, arguments_descriptor)); | 1231 MegamorphicCacheTable::Lookup(isolate(), name, arguments_descriptor)); |
| 1234 __ Comment("MegamorphicCall"); | 1232 __ Comment("MegamorphicCall"); |
| 1235 // Load receiver into RDI. | 1233 // Load receiver into RDI. |
| 1236 __ movq(RDI, Address(RSP, (argument_count - 1) * kWordSize)); | 1234 __ movq(RDI, Address(RSP, (argument_count - 1) * kWordSize)); |
| 1237 __ LoadObject(RBX, cache); | 1235 __ LoadObject(RBX, cache); |
| 1238 __ call(Address(THR, Thread::megamorphic_call_checked_entry_offset())); | 1236 __ call(Address(THR, Thread::megamorphic_call_checked_entry_offset())); |
| 1239 | 1237 |
| (...skipping 166 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1406 // TODO(zerny): clobber non-live temporary FPU registers. | 1404 // TODO(zerny): clobber non-live temporary FPU registers. |
| 1407 if (tmp.IsRegister() && | 1405 if (tmp.IsRegister() && |
| 1408 !locs->live_registers()->ContainsRegister(tmp.reg())) { | 1406 !locs->live_registers()->ContainsRegister(tmp.reg())) { |
| 1409 __ movq(tmp.reg(), Immediate(0xf7)); | 1407 __ movq(tmp.reg(), Immediate(0xf7)); |
| 1410 } | 1408 } |
| 1411 } | 1409 } |
| 1412 } | 1410 } |
| 1413 #endif | 1411 #endif |
| 1414 | 1412 |
| 1415 | 1413 |
| 1416 void FlowGraphCompiler::EmitTestAndCall(const ICData& ic_data, | 1414 void FlowGraphCompiler::EmitTestAndCallLoadReceiver( |
| 1417 intptr_t argument_count, | 1415 intptr_t argument_count, |
| 1418 const Array& argument_names, | 1416 const Array& arguments_descriptor) { |
| 1419 Label* failed, | |
| 1420 Label* match_found, | |
| 1421 intptr_t deopt_id, | |
| 1422 TokenPosition token_index, | |
| 1423 LocationSummary* locs, | |
| 1424 bool complete, | |
| 1425 intptr_t total_ic_calls) { | |
| 1426 ASSERT(is_optimizing()); | |
| 1427 | |
| 1428 __ Comment("EmitTestAndCall"); | 1417 __ Comment("EmitTestAndCall"); |
| 1429 const Array& arguments_descriptor = Array::ZoneHandle( | |
| 1430 zone(), ArgumentsDescriptor::New(argument_count, argument_names)); | |
| 1431 // Load receiver into RAX. | 1418 // Load receiver into RAX. |
| 1432 __ movq(RAX, Address(RSP, (argument_count - 1) * kWordSize)); | 1419 __ movq(RAX, Address(RSP, (argument_count - 1) * kWordSize)); |
| 1433 __ LoadObject(R10, arguments_descriptor); | 1420 __ LoadObject(R10, arguments_descriptor); |
| 1434 | |
| 1435 const bool kFirstCheckIsSmi = ic_data.GetReceiverClassIdAt(0) == kSmiCid; | |
| 1436 const intptr_t num_checks = ic_data.NumberOfChecks(); | |
| 1437 | |
| 1438 ASSERT(!ic_data.IsNull() && (num_checks > 0)); | |
| 1439 | |
| 1440 Label after_smi_test; | |
| 1441 if (kFirstCheckIsSmi) { | |
| 1442 __ testq(RAX, Immediate(kSmiTagMask)); | |
| 1443 // Jump if receiver is not Smi. | |
| 1444 if (num_checks == 1) { | |
| 1445 __ j(NOT_ZERO, failed); | |
| 1446 } else { | |
| 1447 __ j(NOT_ZERO, &after_smi_test); | |
| 1448 } | |
| 1449 // Do not use the code from the function, but let the code be patched so | |
| 1450 // that we can record the outgoing edges to other code. | |
| 1451 const Function& function = | |
| 1452 Function::ZoneHandle(zone(), ic_data.GetTargetAt(0)); | |
| 1453 GenerateStaticDartCall(deopt_id, token_index, | |
| 1454 *StubCode::CallStaticFunction_entry(), | |
| 1455 RawPcDescriptors::kOther, locs, function); | |
| 1456 __ Drop(argument_count, RCX); | |
| 1457 if (num_checks > 1) { | |
| 1458 __ jmp(match_found); | |
| 1459 } | |
| 1460 } else { | |
| 1461 // Receiver is Smi, but Smi is not a valid class therefore fail. | |
| 1462 // (Smi class must be first in the list). | |
| 1463 if (!complete) { | |
| 1464 __ testq(RAX, Immediate(kSmiTagMask)); | |
| 1465 __ j(ZERO, failed); | |
| 1466 } | |
| 1467 } | |
| 1468 __ Bind(&after_smi_test); | |
| 1469 | |
| 1470 ASSERT(!ic_data.IsNull() && (num_checks > 0)); | |
| 1471 GrowableArray<CidRangeTarget> sorted(num_checks); | |
| 1472 SortICDataByCount(ic_data, &sorted, /* drop_smi = */ true); | |
| 1473 | |
| 1474 const intptr_t sorted_len = sorted.length(); | |
| 1475 // If sorted_len is 0 then only a Smi check was needed; the Smi check above | |
| 1476 // will fail if there was only one check and receiver is not Smi. | |
| 1477 if (sorted_len == 0) return; | |
| 1478 | |
| 1479 // Value is not Smi, | |
| 1480 __ LoadClassId(RDI, RAX); | |
| 1481 | |
| 1482 bool add_megamorphic_call = false; | |
| 1483 const int kMaxImmediateInInstruction = 127; | |
| 1484 int bias = | |
| 1485 ComputeGoodBiasForCidComparison(sorted, kMaxImmediateInInstruction); | |
| 1486 if (bias != 0) __ addl(RDI, Immediate(-bias)); | |
| 1487 | |
| 1488 for (intptr_t i = 0; i < sorted_len; i++) { | |
| 1489 const bool is_last_check = (i == (sorted_len - 1)); | |
| 1490 int cid_start = sorted[i].cid_start; | |
| 1491 int cid_end = sorted[i].cid_end; | |
| 1492 int count = sorted[i].count; | |
| 1493 if (!is_last_check && !complete && count < (total_ic_calls >> 5)) { | |
| 1494 // This case is hit too rarely to be worth writing class-id checks inline | |
| 1495 // for. | |
| 1496 add_megamorphic_call = true; | |
| 1497 break; | |
| 1498 } | |
| 1499 ASSERT(cid_start > kSmiCid || cid_end < kSmiCid); | |
| 1500 Label next_test; | |
| 1501 if (!complete || !is_last_check) { | |
| 1502 Label* next_label = is_last_check ? failed : &next_test; | |
| 1503 bool near = is_last_check ? Assembler::kFarJump : Assembler::kNearJump; | |
| 1504 if (cid_start == cid_end) { | |
| 1505 __ cmpl(RDI, Immediate(cid_start - bias)); | |
| 1506 __ j(NOT_EQUAL, next_label, near); | |
| 1507 } else { | |
| 1508 __ addl(RDI, Immediate(bias - cid_start)); | |
| 1509 bias = cid_start; | |
| 1510 __ cmpl(RDI, Immediate(cid_end - cid_start)); | |
| 1511 __ j(ABOVE, next_label, near); // Unsigned higher. | |
| 1512 } | |
| 1513 } | |
| 1514 // Do not use the code from the function, but let the code be patched so | |
| 1515 // that we can record the outgoing edges to other code. | |
| 1516 const Function& function = *sorted[i].target; | |
| 1517 GenerateStaticDartCall(deopt_id, token_index, | |
| 1518 *StubCode::CallStaticFunction_entry(), | |
| 1519 RawPcDescriptors::kOther, locs, function); | |
| 1520 __ Drop(argument_count, RCX); | |
| 1521 if (!is_last_check) { | |
| 1522 __ jmp(match_found); | |
| 1523 } | |
| 1524 __ Bind(&next_test); | |
| 1525 } | |
| 1526 if (add_megamorphic_call) { | |
| 1527 int try_index = CatchClauseNode::kInvalidTryIndex; | |
| 1528 EmitMegamorphicInstanceCall(ic_data, argument_count, deopt_id, token_index, | |
| 1529 locs, try_index, argument_count); | |
| 1530 } | |
| 1531 } | 1421 } |
| 1532 | 1422 |
| 1533 | 1423 |
| 1424 void FlowGraphCompiler::EmitTestAndCallSmiBranch(Label* label, bool if_smi) { |
| 1425 __ testq(RAX, Immediate(kSmiTagMask)); |
| 1426 // Jump if receiver is (not) Smi. |
| 1427 __ j(if_smi ? ZERO : NOT_ZERO, label); |
| 1428 } |
| 1429 |
| 1430 |
| 1431 void FlowGraphCompiler::EmitTestAndCallLoadCid() { |
| 1432 __ LoadClassId(RDI, RAX); |
| 1433 } |
| 1434 |
| 1435 |
| 1436 int FlowGraphCompiler::EmitTestAndCallCheckCid(Label* next_label, |
| 1437 const CidRangeTarget& target, |
| 1438 int bias) { |
| 1439 intptr_t cid_start = target.cid_start; |
| 1440 intptr_t cid_end = target.cid_end; |
| 1441 if (cid_start == cid_end) { |
| 1442 __ cmpl(RDI, Immediate(cid_start - bias)); |
| 1443 __ j(NOT_EQUAL, next_label); |
| 1444 } else { |
| 1445 __ addl(RDI, Immediate(bias - cid_start)); |
| 1446 bias = cid_start; |
| 1447 __ cmpl(RDI, Immediate(cid_end - cid_start)); |
| 1448 __ j(ABOVE, next_label); // Unsigned higher. |
| 1449 } |
| 1450 return bias; |
| 1451 } |
| 1452 |
| 1453 |
| 1534 #undef __ | 1454 #undef __ |
| 1535 #define __ compiler_->assembler()-> | 1455 #define __ compiler_->assembler()-> |
| 1536 | 1456 |
| 1537 | 1457 |
| 1538 void ParallelMoveResolver::EmitMove(int index) { | 1458 void ParallelMoveResolver::EmitMove(int index) { |
| 1539 MoveOperands* move = moves_[index]; | 1459 MoveOperands* move = moves_[index]; |
| 1540 const Location source = move->src(); | 1460 const Location source = move->src(); |
| 1541 const Location destination = move->dest(); | 1461 const Location destination = move->dest(); |
| 1542 | 1462 |
| 1543 if (source.IsRegister()) { | 1463 if (source.IsRegister()) { |
| (...skipping 217 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1761 __ movups(reg, Address(RSP, 0)); | 1681 __ movups(reg, Address(RSP, 0)); |
| 1762 __ AddImmediate(RSP, Immediate(kFpuRegisterSize)); | 1682 __ AddImmediate(RSP, Immediate(kFpuRegisterSize)); |
| 1763 } | 1683 } |
| 1764 | 1684 |
| 1765 | 1685 |
| 1766 #undef __ | 1686 #undef __ |
| 1767 | 1687 |
| 1768 } // namespace dart | 1688 } // namespace dart |
| 1769 | 1689 |
| 1770 #endif // defined TARGET_ARCH_X64 | 1690 #endif // defined TARGET_ARCH_X64 |
| OLD | NEW |