| OLD | NEW |
| 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, 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" | 5 #include "vm/globals.h" |
| 6 #if defined(TARGET_ARCH_X64) | 6 #if defined(TARGET_ARCH_X64) |
| 7 | 7 |
| 8 #include "vm/assembler.h" | 8 #include "vm/assembler.h" |
| 9 #include "vm/assembler_macros.h" | 9 #include "vm/assembler_macros.h" |
| 10 #include "vm/compiler.h" | 10 #include "vm/compiler.h" |
| (...skipping 679 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 690 __ cmpq(RBX, Immediate(RawObject::SizeTag::kMaxSizeTag)); | 690 __ cmpq(RBX, Immediate(RawObject::SizeTag::kMaxSizeTag)); |
| 691 __ j(ABOVE, &size_tag_overflow, Assembler::kNearJump); | 691 __ j(ABOVE, &size_tag_overflow, Assembler::kNearJump); |
| 692 __ shlq(RBX, Immediate(RawObject::kSizeTagBit - kObjectAlignmentLog2)); | 692 __ shlq(RBX, Immediate(RawObject::kSizeTagBit - kObjectAlignmentLog2)); |
| 693 __ jmp(&done); | 693 __ jmp(&done); |
| 694 | 694 |
| 695 __ Bind(&size_tag_overflow); | 695 __ Bind(&size_tag_overflow); |
| 696 __ movq(RBX, Immediate(0)); | 696 __ movq(RBX, Immediate(0)); |
| 697 __ Bind(&done); | 697 __ Bind(&done); |
| 698 | 698 |
| 699 // Get the class index and insert it into the tags. | 699 // Get the class index and insert it into the tags. |
| 700 __ orq(RBX, Immediate(RawObject::ClassIdTag::encode(kArray))); | 700 __ orq(RBX, Immediate(RawObject::ClassIdTag::encode(kArrayCid))); |
| 701 __ movq(FieldAddress(RAX, Array::tags_offset()), RBX); | 701 __ movq(FieldAddress(RAX, Array::tags_offset()), RBX); |
| 702 } | 702 } |
| 703 | 703 |
| 704 // Initialize all array elements to raw_null. | 704 // Initialize all array elements to raw_null. |
| 705 // RAX: new object start as a tagged pointer. | 705 // RAX: new object start as a tagged pointer. |
| 706 // R12: new object end address. | 706 // R12: new object end address. |
| 707 __ leaq(RBX, FieldAddress(RAX, Array::data_offset())); | 707 __ leaq(RBX, FieldAddress(RAX, Array::data_offset())); |
| 708 // RBX: iterator which initially points to the start of the variable | 708 // RBX: iterator which initially points to the start of the variable |
| 709 // data area to be initialized. | 709 // data area to be initialized. |
| 710 Label done; | 710 Label done; |
| (...skipping 512 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1223 // RDI: new object type arguments. | 1223 // RDI: new object type arguments. |
| 1224 } | 1224 } |
| 1225 | 1225 |
| 1226 // RAX: new object start. | 1226 // RAX: new object start. |
| 1227 // RBX: next object start. | 1227 // RBX: next object start. |
| 1228 // RDI: new object type arguments (if is_cls_parameterized). | 1228 // RDI: new object type arguments (if is_cls_parameterized). |
| 1229 __ LoadObject(RDX, cls); // Load class of object to be allocated. | 1229 __ LoadObject(RDX, cls); // Load class of object to be allocated. |
| 1230 // Set the tags. | 1230 // Set the tags. |
| 1231 uword tags = 0; | 1231 uword tags = 0; |
| 1232 tags = RawObject::SizeTag::update(instance_size, tags); | 1232 tags = RawObject::SizeTag::update(instance_size, tags); |
| 1233 ASSERT(cls.id() != kIllegalObjectKind); | 1233 ASSERT(cls.id() != kIllegalCid); |
| 1234 tags = RawObject::ClassIdTag::update(cls.id(), tags); | 1234 tags = RawObject::ClassIdTag::update(cls.id(), tags); |
| 1235 __ movq(Address(RAX, Instance::tags_offset()), Immediate(tags)); | 1235 __ movq(Address(RAX, Instance::tags_offset()), Immediate(tags)); |
| 1236 | 1236 |
| 1237 // Initialize the remaining words of the object. | 1237 // Initialize the remaining words of the object. |
| 1238 const Immediate raw_null = | 1238 const Immediate raw_null = |
| 1239 Immediate(reinterpret_cast<intptr_t>(Object::null())); | 1239 Immediate(reinterpret_cast<intptr_t>(Object::null())); |
| 1240 | 1240 |
| 1241 // RAX: new object start. | 1241 // RAX: new object start. |
| 1242 // RBX: next object start. | 1242 // RBX: next object start. |
| 1243 // RDX: class of the object to be allocated. | 1243 // RDX: class of the object to be allocated. |
| (...skipping 369 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1613 Immediate(reinterpret_cast<intptr_t>(Object::null())); | 1613 Immediate(reinterpret_cast<intptr_t>(Object::null())); |
| 1614 Label loop, found; | 1614 Label loop, found; |
| 1615 if (num_args == 1) { | 1615 if (num_args == 1) { |
| 1616 __ call(&get_class_id_as_smi); | 1616 __ call(&get_class_id_as_smi); |
| 1617 // RAX: receiver's class id as Smi. | 1617 // RAX: receiver's class id as Smi. |
| 1618 __ Bind(&loop); | 1618 __ Bind(&loop); |
| 1619 __ movq(R13, Address(R12, 0)); // Get class if (Smi) to check. | 1619 __ movq(R13, Address(R12, 0)); // Get class if (Smi) to check. |
| 1620 __ cmpq(RAX, R13); // Match? | 1620 __ cmpq(RAX, R13); // Match? |
| 1621 __ j(EQUAL, &found, Assembler::kNearJump); | 1621 __ j(EQUAL, &found, Assembler::kNearJump); |
| 1622 __ addq(R12, Immediate(kWordSize * 2)); // Next element (class + target). | 1622 __ addq(R12, Immediate(kWordSize * 2)); // Next element (class + target). |
| 1623 __ cmpq(R13, Immediate(Smi::RawValue(kIllegalObjectKind))); // Done? | 1623 __ cmpq(R13, Immediate(Smi::RawValue(kIllegalCid))); // Done? |
| 1624 __ j(NOT_EQUAL, &loop, Assembler::kNearJump); | 1624 __ j(NOT_EQUAL, &loop, Assembler::kNearJump); |
| 1625 } else if (num_args == 2) { | 1625 } else if (num_args == 2) { |
| 1626 Label no_match; | 1626 Label no_match; |
| 1627 __ Bind(&loop); | 1627 __ Bind(&loop); |
| 1628 // Get receiver. | 1628 // Get receiver. |
| 1629 __ movq(RAX, FieldAddress(R10, Array::data_offset())); | 1629 __ movq(RAX, FieldAddress(R10, Array::data_offset())); |
| 1630 __ movq(RAX, Address(RSP, RAX, TIMES_4, 0)); // RAX is Smi. | 1630 __ movq(RAX, Address(RSP, RAX, TIMES_4, 0)); // RAX is Smi. |
| 1631 __ call(&get_class_id_as_smi); | 1631 __ call(&get_class_id_as_smi); |
| 1632 __ movq(R13, Address(R12, 0)); // Get class id from IC data to check. | 1632 __ movq(R13, Address(R12, 0)); // Get class id from IC data to check. |
| 1633 __ cmpq(RAX, R13); // Class id match? | 1633 __ cmpq(RAX, R13); // Class id match? |
| 1634 __ j(NOT_EQUAL, &no_match, Assembler::kNearJump); | 1634 __ j(NOT_EQUAL, &no_match, Assembler::kNearJump); |
| 1635 // Check second. | 1635 // Check second. |
| 1636 // Get next argument. | 1636 // Get next argument. |
| 1637 __ movq(RAX, FieldAddress(R10, Array::data_offset())); | 1637 __ movq(RAX, FieldAddress(R10, Array::data_offset())); |
| 1638 __ movq(RAX, Address(RSP, RAX, TIMES_4, -kWordSize)); // RAX is Smi. | 1638 __ movq(RAX, Address(RSP, RAX, TIMES_4, -kWordSize)); // RAX is Smi. |
| 1639 __ call(&get_class_id_as_smi); | 1639 __ call(&get_class_id_as_smi); |
| 1640 __ movq(R13, Address(R12, kWordSize)); // Get class id from IC data. | 1640 __ movq(R13, Address(R12, kWordSize)); // Get class id from IC data. |
| 1641 __ cmpq(RAX, R13); // Class id match? | 1641 __ cmpq(RAX, R13); // Class id match? |
| 1642 __ j(EQUAL, &found); | 1642 __ j(EQUAL, &found); |
| 1643 __ Bind(&no_match); | 1643 __ Bind(&no_match); |
| 1644 __ addq(R12, Immediate(kWordSize * (1 + num_args))); // Next element. | 1644 __ addq(R12, Immediate(kWordSize * (1 + num_args))); // Next element. |
| 1645 __ cmpq(R13, Immediate(Smi::RawValue(kIllegalObjectKind))); // Done? | 1645 __ cmpq(R13, Immediate(Smi::RawValue(kIllegalCid))); // Done? |
| 1646 __ j(NOT_EQUAL, &loop, Assembler::kNearJump); | 1646 __ j(NOT_EQUAL, &loop, Assembler::kNearJump); |
| 1647 } | 1647 } |
| 1648 | 1648 |
| 1649 __ Bind(&ic_miss); | 1649 __ Bind(&ic_miss); |
| 1650 // Compute address of arguments (first read number of arguments from argument | 1650 // Compute address of arguments (first read number of arguments from argument |
| 1651 // descriptor array and then compute address on the stack). | 1651 // descriptor array and then compute address on the stack). |
| 1652 __ movq(RAX, FieldAddress(R10, Array::data_offset())); | 1652 __ movq(RAX, FieldAddress(R10, Array::data_offset())); |
| 1653 __ leaq(RAX, Address(RSP, RAX, TIMES_4, 0)); // RAX is Smi. | 1653 __ leaq(RAX, Address(RSP, RAX, TIMES_4, 0)); // RAX is Smi. |
| 1654 AssemblerMacros::EnterStubFrame(assembler); | 1654 AssemblerMacros::EnterStubFrame(assembler); |
| 1655 __ pushq(R10); // Preserve arguments array. | 1655 __ pushq(R10); // Preserve arguments array. |
| (...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1690 __ movq(RAX, FieldAddress(RAX, Function::code_offset())); | 1690 __ movq(RAX, FieldAddress(RAX, Function::code_offset())); |
| 1691 __ movq(RAX, FieldAddress(RAX, Code::instructions_offset())); | 1691 __ movq(RAX, FieldAddress(RAX, Code::instructions_offset())); |
| 1692 __ addq(RAX, Immediate(Instructions::HeaderSize() - kHeapObjectTag)); | 1692 __ addq(RAX, Immediate(Instructions::HeaderSize() - kHeapObjectTag)); |
| 1693 __ jmp(RAX); | 1693 __ jmp(RAX); |
| 1694 | 1694 |
| 1695 __ Bind(&get_class_id_as_smi); | 1695 __ Bind(&get_class_id_as_smi); |
| 1696 Label not_smi; | 1696 Label not_smi; |
| 1697 // Test if Smi -> load Smi class for comparison. | 1697 // Test if Smi -> load Smi class for comparison. |
| 1698 __ testq(RAX, Immediate(kSmiTagMask)); | 1698 __ testq(RAX, Immediate(kSmiTagMask)); |
| 1699 __ j(NOT_ZERO, ¬_smi, Assembler::kNearJump); | 1699 __ j(NOT_ZERO, ¬_smi, Assembler::kNearJump); |
| 1700 __ movq(RAX, Immediate(Smi::RawValue(kSmi))); | 1700 __ movq(RAX, Immediate(Smi::RawValue(kSmiCid))); |
| 1701 __ ret(); | 1701 __ ret(); |
| 1702 | 1702 |
| 1703 __ Bind(¬_smi); | 1703 __ Bind(¬_smi); |
| 1704 __ LoadClassId(RAX, RAX); | 1704 __ LoadClassId(RAX, RAX); |
| 1705 __ SmiTag(RAX); | 1705 __ SmiTag(RAX); |
| 1706 __ ret(); | 1706 __ ret(); |
| 1707 } | 1707 } |
| 1708 | 1708 |
| 1709 | 1709 |
| 1710 // Use inline cache data array to invoke the target or continue in inline | 1710 // Use inline cache data array to invoke the target or continue in inline |
| (...skipping 223 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1934 void StubCode::GenerateJumpToErrorHandlerStub(Assembler* assembler) { | 1934 void StubCode::GenerateJumpToErrorHandlerStub(Assembler* assembler) { |
| 1935 __ movq(RAX, RCX); // error object. | 1935 __ movq(RAX, RCX); // error object. |
| 1936 __ movq(RBP, RDX); // target frame_pointer. | 1936 __ movq(RBP, RDX); // target frame_pointer. |
| 1937 __ movq(RSP, RSI); // target stack_pointer. | 1937 __ movq(RSP, RSI); // target stack_pointer. |
| 1938 __ jmp(RDI); // Jump to the exception handler code. | 1938 __ jmp(RDI); // Jump to the exception handler code. |
| 1939 } | 1939 } |
| 1940 | 1940 |
| 1941 } // namespace dart | 1941 } // namespace dart |
| 1942 | 1942 |
| 1943 #endif // defined TARGET_ARCH_X64 | 1943 #endif // defined TARGET_ARCH_X64 |
| OLD | NEW |