OLD | NEW |
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/v8.h" | 5 #include "src/v8.h" |
6 | 6 |
7 #if V8_TARGET_ARCH_X87 | 7 #if V8_TARGET_ARCH_X87 |
8 | 8 |
9 #include "src/base/bits.h" | 9 #include "src/base/bits.h" |
10 #include "src/bootstrapper.h" | 10 #include "src/bootstrapper.h" |
(...skipping 1582 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1593 | 1593 |
1594 // Call the native; it returns -1 (less), 0 (equal), or 1 (greater) | 1594 // Call the native; it returns -1 (less), 0 (equal), or 1 (greater) |
1595 // tagged as a small integer. | 1595 // tagged as a small integer. |
1596 __ InvokeBuiltin(builtin, JUMP_FUNCTION); | 1596 __ InvokeBuiltin(builtin, JUMP_FUNCTION); |
1597 | 1597 |
1598 __ bind(&miss); | 1598 __ bind(&miss); |
1599 GenerateMiss(masm); | 1599 GenerateMiss(masm); |
1600 } | 1600 } |
1601 | 1601 |
1602 | 1602 |
| 1603 static void CallStubInRecordCallTarget(MacroAssembler* masm, CodeStub* stub) { |
| 1604 // eax : number of arguments to the construct function |
| 1605 // ebx : Feedback vector |
| 1606 // edx : slot in feedback vector (Smi) |
| 1607 // edi : the function to call |
| 1608 FrameScope scope(masm, StackFrame::INTERNAL); |
| 1609 |
| 1610 // Arguments register must be smi-tagged to call out. |
| 1611 __ SmiTag(eax); |
| 1612 __ push(eax); |
| 1613 __ push(edi); |
| 1614 __ push(edx); |
| 1615 __ push(ebx); |
| 1616 |
| 1617 __ CallStub(stub); |
| 1618 |
| 1619 __ pop(ebx); |
| 1620 __ pop(edx); |
| 1621 __ pop(edi); |
| 1622 __ pop(eax); |
| 1623 __ SmiUntag(eax); |
| 1624 } |
| 1625 |
| 1626 |
1603 static void GenerateRecordCallTarget(MacroAssembler* masm) { | 1627 static void GenerateRecordCallTarget(MacroAssembler* masm) { |
1604 // Cache the called function in a feedback vector slot. Cache states | 1628 // Cache the called function in a feedback vector slot. Cache states |
1605 // are uninitialized, monomorphic (indicated by a JSFunction), and | 1629 // are uninitialized, monomorphic (indicated by a JSFunction), and |
1606 // megamorphic. | 1630 // megamorphic. |
1607 // eax : number of arguments to the construct function | 1631 // eax : number of arguments to the construct function |
1608 // ebx : Feedback vector | 1632 // ebx : Feedback vector |
1609 // edx : slot in feedback vector (Smi) | 1633 // edx : slot in feedback vector (Smi) |
1610 // edi : the function to call | 1634 // edi : the function to call |
1611 Isolate* isolate = masm->isolate(); | 1635 Isolate* isolate = masm->isolate(); |
1612 Label initialize, done, miss, megamorphic, not_array_function; | 1636 Label initialize, done, miss, megamorphic, not_array_function; |
1613 | 1637 |
1614 // Load the cache state into ecx. | 1638 // Load the cache state into ecx. |
1615 __ mov(ecx, FieldOperand(ebx, edx, times_half_pointer_size, | 1639 __ mov(ecx, FieldOperand(ebx, edx, times_half_pointer_size, |
1616 FixedArray::kHeaderSize)); | 1640 FixedArray::kHeaderSize)); |
1617 | 1641 |
1618 // A monomorphic cache hit or an already megamorphic state: invoke the | 1642 // A monomorphic cache hit or an already megamorphic state: invoke the |
1619 // function without changing the state. | 1643 // function without changing the state. |
1620 __ cmp(ecx, edi); | 1644 Label check_allocation_site; |
| 1645 __ cmp(edi, FieldOperand(ecx, WeakCell::kValueOffset)); |
1621 __ j(equal, &done, Label::kFar); | 1646 __ j(equal, &done, Label::kFar); |
1622 __ cmp(ecx, Immediate(TypeFeedbackVector::MegamorphicSentinel(isolate))); | 1647 __ CompareRoot(ecx, Heap::kmegamorphic_symbolRootIndex); |
1623 __ j(equal, &done, Label::kFar); | 1648 __ j(equal, &done, Label::kFar); |
| 1649 __ CompareRoot(FieldOperand(ecx, 0), Heap::kWeakCellMapRootIndex); |
| 1650 __ j(not_equal, FLAG_pretenuring_call_new ? &miss : &check_allocation_site); |
| 1651 |
| 1652 // If edi is not equal to the weak cell value, and the weak cell value is |
| 1653 // cleared, we have a new chance to become monomorphic. |
| 1654 __ JumpIfSmi(FieldOperand(ecx, WeakCell::kValueOffset), &initialize); |
| 1655 __ jmp(&megamorphic); |
1624 | 1656 |
1625 if (!FLAG_pretenuring_call_new) { | 1657 if (!FLAG_pretenuring_call_new) { |
| 1658 __ bind(&check_allocation_site); |
1626 // If we came here, we need to see if we are the array function. | 1659 // If we came here, we need to see if we are the array function. |
1627 // If we didn't have a matching function, and we didn't find the megamorph | 1660 // If we didn't have a matching function, and we didn't find the megamorph |
1628 // sentinel, then we have in the slot either some other function or an | 1661 // sentinel, then we have in the slot either some other function or an |
1629 // AllocationSite. Do a map check on the object in ecx. | 1662 // AllocationSite. Do a map check on the object in ecx. |
1630 Handle<Map> allocation_site_map = isolate->factory()->allocation_site_map(); | 1663 __ CompareRoot(FieldOperand(ecx, 0), Heap::kAllocationSiteMapRootIndex); |
1631 __ cmp(FieldOperand(ecx, 0), Immediate(allocation_site_map)); | |
1632 __ j(not_equal, &miss); | 1664 __ j(not_equal, &miss); |
1633 | 1665 |
1634 // Make sure the function is the Array() function | 1666 // Make sure the function is the Array() function |
1635 __ LoadGlobalFunction(Context::ARRAY_FUNCTION_INDEX, ecx); | 1667 __ LoadGlobalFunction(Context::ARRAY_FUNCTION_INDEX, ecx); |
1636 __ cmp(edi, ecx); | 1668 __ cmp(edi, ecx); |
1637 __ j(not_equal, &megamorphic); | 1669 __ j(not_equal, &megamorphic); |
1638 __ jmp(&done, Label::kFar); | 1670 __ jmp(&done, Label::kFar); |
1639 } | 1671 } |
1640 | 1672 |
1641 __ bind(&miss); | 1673 __ bind(&miss); |
1642 | 1674 |
1643 // A monomorphic miss (i.e, here the cache is not uninitialized) goes | 1675 // A monomorphic miss (i.e, here the cache is not uninitialized) goes |
1644 // megamorphic. | 1676 // megamorphic. |
1645 __ cmp(ecx, Immediate(TypeFeedbackVector::UninitializedSentinel(isolate))); | 1677 __ CompareRoot(ecx, Heap::kuninitialized_symbolRootIndex); |
1646 __ j(equal, &initialize); | 1678 __ j(equal, &initialize); |
1647 // MegamorphicSentinel is an immortal immovable object (undefined) so no | 1679 // MegamorphicSentinel is an immortal immovable object (undefined) so no |
1648 // write-barrier is needed. | 1680 // write-barrier is needed. |
1649 __ bind(&megamorphic); | 1681 __ bind(&megamorphic); |
1650 __ mov( | 1682 __ mov( |
1651 FieldOperand(ebx, edx, times_half_pointer_size, FixedArray::kHeaderSize), | 1683 FieldOperand(ebx, edx, times_half_pointer_size, FixedArray::kHeaderSize), |
1652 Immediate(TypeFeedbackVector::MegamorphicSentinel(isolate))); | 1684 Immediate(TypeFeedbackVector::MegamorphicSentinel(isolate))); |
1653 __ jmp(&done, Label::kFar); | 1685 __ jmp(&done, Label::kFar); |
1654 | 1686 |
1655 // An uninitialized cache is patched with the function or sentinel to | 1687 // An uninitialized cache is patched with the function or sentinel to |
1656 // indicate the ElementsKind if function is the Array constructor. | 1688 // indicate the ElementsKind if function is the Array constructor. |
1657 __ bind(&initialize); | 1689 __ bind(&initialize); |
1658 if (!FLAG_pretenuring_call_new) { | 1690 if (!FLAG_pretenuring_call_new) { |
1659 // Make sure the function is the Array() function | 1691 // Make sure the function is the Array() function |
1660 __ LoadGlobalFunction(Context::ARRAY_FUNCTION_INDEX, ecx); | 1692 __ LoadGlobalFunction(Context::ARRAY_FUNCTION_INDEX, ecx); |
1661 __ cmp(edi, ecx); | 1693 __ cmp(edi, ecx); |
1662 __ j(not_equal, ¬_array_function); | 1694 __ j(not_equal, ¬_array_function); |
1663 | 1695 |
1664 // The target function is the Array constructor, | 1696 // The target function is the Array constructor, |
1665 // Create an AllocationSite if we don't already have it, store it in the | 1697 // Create an AllocationSite if we don't already have it, store it in the |
1666 // slot. | 1698 // slot. |
1667 { | 1699 CreateAllocationSiteStub create_stub(isolate); |
1668 FrameScope scope(masm, StackFrame::INTERNAL); | 1700 CallStubInRecordCallTarget(masm, &create_stub); |
1669 | |
1670 // Arguments register must be smi-tagged to call out. | |
1671 __ SmiTag(eax); | |
1672 __ push(eax); | |
1673 __ push(edi); | |
1674 __ push(edx); | |
1675 __ push(ebx); | |
1676 | |
1677 CreateAllocationSiteStub create_stub(isolate); | |
1678 __ CallStub(&create_stub); | |
1679 | |
1680 __ pop(ebx); | |
1681 __ pop(edx); | |
1682 __ pop(edi); | |
1683 __ pop(eax); | |
1684 __ SmiUntag(eax); | |
1685 } | |
1686 __ jmp(&done); | 1701 __ jmp(&done); |
1687 | 1702 |
1688 __ bind(¬_array_function); | 1703 __ bind(¬_array_function); |
1689 } | 1704 } |
1690 | 1705 |
1691 __ mov(FieldOperand(ebx, edx, times_half_pointer_size, | 1706 CreateWeakCellStub create_stub(isolate); |
1692 FixedArray::kHeaderSize), | 1707 CallStubInRecordCallTarget(masm, &create_stub); |
1693 edi); | |
1694 // We won't need edx or ebx anymore, just save edi | |
1695 __ push(edi); | |
1696 __ push(ebx); | |
1697 __ push(edx); | |
1698 __ RecordWriteArray(ebx, edi, edx, kDontSaveFPRegs, EMIT_REMEMBERED_SET, | |
1699 OMIT_SMI_CHECK); | |
1700 __ pop(edx); | |
1701 __ pop(ebx); | |
1702 __ pop(edi); | |
1703 | |
1704 __ bind(&done); | 1708 __ bind(&done); |
1705 } | 1709 } |
1706 | 1710 |
1707 | 1711 |
1708 static void EmitContinueIfStrictOrNative(MacroAssembler* masm, Label* cont) { | 1712 static void EmitContinueIfStrictOrNative(MacroAssembler* masm, Label* cont) { |
1709 // Do not transform the receiver for strict mode functions. | 1713 // Do not transform the receiver for strict mode functions. |
1710 __ mov(ecx, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset)); | 1714 __ mov(ecx, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset)); |
1711 __ test_b(FieldOperand(ecx, SharedFunctionInfo::kStrictModeByteOffset), | 1715 __ test_b(FieldOperand(ecx, SharedFunctionInfo::kStrictModeByteOffset), |
1712 1 << SharedFunctionInfo::kStrictModeBitWithinByte); | 1716 1 << SharedFunctionInfo::kStrictModeBitWithinByte); |
1713 __ j(not_equal, cont); | 1717 __ j(not_equal, cont); |
(...skipping 3349 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
5063 ApiParameterOperand(2), kStackSpace, nullptr, | 5067 ApiParameterOperand(2), kStackSpace, nullptr, |
5064 Operand(ebp, 7 * kPointerSize), NULL); | 5068 Operand(ebp, 7 * kPointerSize), NULL); |
5065 } | 5069 } |
5066 | 5070 |
5067 | 5071 |
5068 #undef __ | 5072 #undef __ |
5069 | 5073 |
5070 } } // namespace v8::internal | 5074 } } // namespace v8::internal |
5071 | 5075 |
5072 #endif // V8_TARGET_ARCH_X87 | 5076 #endif // V8_TARGET_ARCH_X87 |
OLD | NEW |