| 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 493 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 504 // - preserved instance in RAX and optional instantiator type arguments in RDX. | 504 // - preserved instance in RAX and optional instantiator type arguments in RDX. |
| 505 // Note that this inlined code must be followed by the runtime_call code, as it | 505 // Note that this inlined code must be followed by the runtime_call code, as it |
| 506 // may fall through to it. Otherwise, this inline code will jump to the label | 506 // may fall through to it. Otherwise, this inline code will jump to the label |
| 507 // is_instance or to the label is_not_instance. | 507 // is_instance or to the label is_not_instance. |
| 508 RawSubtypeTestCache* FlowGraphCompiler::GenerateInlineInstanceof( | 508 RawSubtypeTestCache* FlowGraphCompiler::GenerateInlineInstanceof( |
| 509 TokenPosition token_pos, | 509 TokenPosition token_pos, |
| 510 const AbstractType& type, | 510 const AbstractType& type, |
| 511 Label* is_instance_lbl, | 511 Label* is_instance_lbl, |
| 512 Label* is_not_instance_lbl) { | 512 Label* is_not_instance_lbl) { |
| 513 __ Comment("InlineInstanceof"); | 513 __ Comment("InlineInstanceof"); |
| 514 if (type.IsVoidType()) { | |
| 515 // A non-null value is returned from a void function, which will result in a | |
| 516 // type error. A null value is handled prior to executing this inline code. | |
| 517 return SubtypeTestCache::null(); | |
| 518 } | |
| 519 if (type.IsInstantiated()) { | 514 if (type.IsInstantiated()) { |
| 520 const Class& type_class = Class::ZoneHandle(zone(), type.type_class()); | 515 const Class& type_class = Class::ZoneHandle(zone(), type.type_class()); |
| 521 // A class equality check is only applicable with a dst type (not a | 516 // A class equality check is only applicable with a dst type (not a |
| 522 // function type) of a non-parameterized class or with a raw dst type of | 517 // function type) of a non-parameterized class or with a raw dst type of |
| 523 // a parameterized class. | 518 // a parameterized class. |
| 524 if (type.IsFunctionType() || (type_class.NumTypeArguments() > 0)) { | 519 if (type.IsFunctionType() || (type_class.NumTypeArguments() > 0)) { |
| 525 return GenerateInstantiatedTypeWithArgumentsTest( | 520 return GenerateInstantiatedTypeWithArgumentsTest( |
| 526 token_pos, type, is_instance_lbl, is_not_instance_lbl); | 521 token_pos, type, is_instance_lbl, is_not_instance_lbl); |
| 527 // Fall through to runtime call. | 522 // Fall through to runtime call. |
| 528 } | 523 } |
| (...skipping 23 matching lines...) Expand all Loading... |
| 552 // - RDX: instantiator type arguments or raw_null. | 547 // - RDX: instantiator type arguments or raw_null. |
| 553 // Clobbers RDX. | 548 // Clobbers RDX. |
| 554 // Returns: | 549 // Returns: |
| 555 // - true or false in RAX. | 550 // - true or false in RAX. |
| 556 void FlowGraphCompiler::GenerateInstanceOf(TokenPosition token_pos, | 551 void FlowGraphCompiler::GenerateInstanceOf(TokenPosition token_pos, |
| 557 intptr_t deopt_id, | 552 intptr_t deopt_id, |
| 558 const AbstractType& type, | 553 const AbstractType& type, |
| 559 bool negate_result, | 554 bool negate_result, |
| 560 LocationSummary* locs) { | 555 LocationSummary* locs) { |
| 561 ASSERT(type.IsFinalized() && !type.IsMalformedOrMalbounded()); | 556 ASSERT(type.IsFinalized() && !type.IsMalformedOrMalbounded()); |
| 562 ASSERT(!type.IsObjectType() && !type.IsDynamicType()); | 557 ASSERT(!type.IsObjectType() && !type.IsDynamicType() && !type.IsVoidType()); |
| 563 | 558 |
| 564 Label is_instance, is_not_instance; | 559 Label is_instance, is_not_instance; |
| 565 __ pushq(RDX); // Store instantiator type arguments. | 560 __ pushq(RDX); // Store instantiator type arguments. |
| 566 // If type is instantiated and non-parameterized, we can inline code | 561 // If type is instantiated and non-parameterized, we can inline code |
| 567 // checking whether the tested instance is a Smi. | 562 // checking whether the tested instance is a Smi. |
| 568 if (type.IsInstantiated()) { | 563 if (type.IsInstantiated()) { |
| 569 // A null object is only an instance of Null, Object, and dynamic. | 564 // A null object is only an instance of Null, Object, void and dynamic. |
| 570 // Object and dynamic have already been checked above (if the type is | 565 // Object void and dynamic have already been checked above (if the type is |
| 571 // instantiated). So we can return false here if the instance is null, | 566 // instantiated). So we can return false here if the instance is null, |
| 572 // unless the type is Null (and if the type is instantiated). | 567 // unless the type is Null (and if the type is instantiated). |
| 573 // We can only inline this null check if the type is instantiated at compile | 568 // We can only inline this null check if the type is instantiated at compile |
| 574 // time, since an uninstantiated type at compile time could be Null, Object, | 569 // time, since an uninstantiated type at compile time could be Null, Object, |
| 575 // or dynamic at run time. | 570 // or dynamic at run time. |
| 576 __ CompareObject(RAX, Object::null_object()); | 571 __ CompareObject(RAX, Object::null_object()); |
| 577 __ j(EQUAL, type.IsNullType() ? &is_instance : &is_not_instance); | 572 __ j(EQUAL, type.IsNullType() ? &is_instance : &is_not_instance); |
| 578 } | 573 } |
| 579 | 574 |
| 580 // Generate inline instanceof test. | 575 // Generate inline instanceof test. |
| (...skipping 52 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 633 void FlowGraphCompiler::GenerateAssertAssignable(TokenPosition token_pos, | 628 void FlowGraphCompiler::GenerateAssertAssignable(TokenPosition token_pos, |
| 634 intptr_t deopt_id, | 629 intptr_t deopt_id, |
| 635 const AbstractType& dst_type, | 630 const AbstractType& dst_type, |
| 636 const String& dst_name, | 631 const String& dst_name, |
| 637 LocationSummary* locs) { | 632 LocationSummary* locs) { |
| 638 ASSERT(!token_pos.IsClassifying()); | 633 ASSERT(!token_pos.IsClassifying()); |
| 639 ASSERT(!dst_type.IsNull()); | 634 ASSERT(!dst_type.IsNull()); |
| 640 ASSERT(dst_type.IsFinalized()); | 635 ASSERT(dst_type.IsFinalized()); |
| 641 // Assignable check is skipped in FlowGraphBuilder, not here. | 636 // Assignable check is skipped in FlowGraphBuilder, not here. |
| 642 ASSERT(dst_type.IsMalformedOrMalbounded() || | 637 ASSERT(dst_type.IsMalformedOrMalbounded() || |
| 643 (!dst_type.IsDynamicType() && !dst_type.IsObjectType())); | 638 (!dst_type.IsDynamicType() && !dst_type.IsObjectType() && |
| 639 !dst_type.IsVoidType())); |
| 644 __ pushq(RDX); // Store instantiator type arguments. | 640 __ pushq(RDX); // Store instantiator type arguments. |
| 645 // A null object is always assignable and is returned as result. | 641 // A null object is always assignable and is returned as result. |
| 646 Label is_assignable, runtime_call; | 642 Label is_assignable, runtime_call; |
| 647 __ CompareObject(RAX, Object::null_object()); | 643 __ CompareObject(RAX, Object::null_object()); |
| 648 __ j(EQUAL, &is_assignable); | 644 __ j(EQUAL, &is_assignable); |
| 649 | 645 |
| 650 // Generate throw new TypeError() if the type is malformed or malbounded. | 646 // Generate throw new TypeError() if the type is malformed or malbounded. |
| 651 if (dst_type.IsMalformedOrMalbounded()) { | 647 if (dst_type.IsMalformedOrMalbounded()) { |
| 652 __ PushObject(Object::null_object()); // Make room for the result. | 648 __ PushObject(Object::null_object()); // Make room for the result. |
| 653 __ pushq(RAX); // Push the source object. | 649 __ pushq(RAX); // Push the source object. |
| (...skipping 1118 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1772 __ movups(reg, Address(RSP, 0)); | 1768 __ movups(reg, Address(RSP, 0)); |
| 1773 __ AddImmediate(RSP, Immediate(kFpuRegisterSize)); | 1769 __ AddImmediate(RSP, Immediate(kFpuRegisterSize)); |
| 1774 } | 1770 } |
| 1775 | 1771 |
| 1776 | 1772 |
| 1777 #undef __ | 1773 #undef __ |
| 1778 | 1774 |
| 1779 } // namespace dart | 1775 } // namespace dart |
| 1780 | 1776 |
| 1781 #endif // defined TARGET_ARCH_X64 | 1777 #endif // defined TARGET_ARCH_X64 |
| OLD | NEW |