| 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_IA32 | 7 #if V8_TARGET_ARCH_IA32 |
| 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 720 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 731 StubRuntimeCallHelper call_helper; | 731 StubRuntimeCallHelper call_helper; |
| 732 char_at_generator.GenerateSlow(masm, call_helper); | 732 char_at_generator.GenerateSlow(masm, call_helper); |
| 733 | 733 |
| 734 __ bind(&miss); | 734 __ bind(&miss); |
| 735 PropertyAccessCompiler::TailCallBuiltin( | 735 PropertyAccessCompiler::TailCallBuiltin( |
| 736 masm, PropertyAccessCompiler::MissBuiltin(Code::KEYED_LOAD_IC)); | 736 masm, PropertyAccessCompiler::MissBuiltin(Code::KEYED_LOAD_IC)); |
| 737 } | 737 } |
| 738 | 738 |
| 739 | 739 |
| 740 void ArgumentsAccessStub::GenerateReadElement(MacroAssembler* masm) { | 740 void ArgumentsAccessStub::GenerateReadElement(MacroAssembler* masm) { |
| 741 CHECK(!has_new_target()); |
| 741 // The key is in edx and the parameter count is in eax. | 742 // The key is in edx and the parameter count is in eax. |
| 742 DCHECK(edx.is(ArgumentsAccessReadDescriptor::index())); | 743 DCHECK(edx.is(ArgumentsAccessReadDescriptor::index())); |
| 743 DCHECK(eax.is(ArgumentsAccessReadDescriptor::parameter_count())); | 744 DCHECK(eax.is(ArgumentsAccessReadDescriptor::parameter_count())); |
| 744 | 745 |
| 745 // The displacement is used for skipping the frame pointer on the | 746 // The displacement is used for skipping the frame pointer on the |
| 746 // stack. It is the offset of the last parameter (if any) relative | 747 // stack. It is the offset of the last parameter (if any) relative |
| 747 // to the frame pointer. | 748 // to the frame pointer. |
| 748 static const int kDisplacement = 1 * kPointerSize; | 749 static const int kDisplacement = 1 * kPointerSize; |
| 749 | 750 |
| 750 // Check that the key is a smi. | 751 // Check that the key is a smi. |
| (...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 797 __ TailCallRuntime(Runtime::kGetArgumentsProperty, 1, 1); | 798 __ TailCallRuntime(Runtime::kGetArgumentsProperty, 1, 1); |
| 798 } | 799 } |
| 799 | 800 |
| 800 | 801 |
| 801 void ArgumentsAccessStub::GenerateNewSloppySlow(MacroAssembler* masm) { | 802 void ArgumentsAccessStub::GenerateNewSloppySlow(MacroAssembler* masm) { |
| 802 // esp[0] : return address | 803 // esp[0] : return address |
| 803 // esp[4] : number of parameters | 804 // esp[4] : number of parameters |
| 804 // esp[8] : receiver displacement | 805 // esp[8] : receiver displacement |
| 805 // esp[12] : function | 806 // esp[12] : function |
| 806 | 807 |
| 808 CHECK(!has_new_target()); |
| 809 |
| 807 // Check if the calling frame is an arguments adaptor frame. | 810 // Check if the calling frame is an arguments adaptor frame. |
| 808 Label runtime; | 811 Label runtime; |
| 809 __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); | 812 __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
| 810 __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); | 813 __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); |
| 811 __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); | 814 __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
| 812 __ j(not_equal, &runtime, Label::kNear); | 815 __ j(not_equal, &runtime, Label::kNear); |
| 813 | 816 |
| 814 // Patch the arguments.length and the parameters pointer. | 817 // Patch the arguments.length and the parameters pointer. |
| 815 __ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); | 818 __ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
| 816 __ mov(Operand(esp, 1 * kPointerSize), ecx); | 819 __ mov(Operand(esp, 1 * kPointerSize), ecx); |
| 817 __ lea(edx, Operand(edx, ecx, times_2, | 820 __ lea(edx, Operand(edx, ecx, times_2, |
| 818 StandardFrameConstants::kCallerSPOffset)); | 821 StandardFrameConstants::kCallerSPOffset)); |
| 819 __ mov(Operand(esp, 2 * kPointerSize), edx); | 822 __ mov(Operand(esp, 2 * kPointerSize), edx); |
| 820 | 823 |
| 821 __ bind(&runtime); | 824 __ bind(&runtime); |
| 822 __ TailCallRuntime(Runtime::kNewSloppyArguments, 3, 1); | 825 __ TailCallRuntime(Runtime::kNewSloppyArguments, 3, 1); |
| 823 } | 826 } |
| 824 | 827 |
| 825 | 828 |
| 826 void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { | 829 void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
| 827 // esp[0] : return address | 830 // esp[0] : return address |
| 828 // esp[4] : number of parameters (tagged) | 831 // esp[4] : number of parameters (tagged) |
| 829 // esp[8] : receiver displacement | 832 // esp[8] : receiver displacement |
| 830 // esp[12] : function | 833 // esp[12] : function |
| 831 | 834 |
| 832 // ebx = parameter count (tagged) | 835 // ebx = parameter count (tagged) |
| 833 __ mov(ebx, Operand(esp, 1 * kPointerSize)); | 836 __ mov(ebx, Operand(esp, 1 * kPointerSize)); |
| 834 | 837 |
| 838 CHECK(!has_new_target()); |
| 839 |
| 835 // Check if the calling frame is an arguments adaptor frame. | 840 // Check if the calling frame is an arguments adaptor frame. |
| 836 // TODO(rossberg): Factor out some of the bits that are shared with the other | 841 // TODO(rossberg): Factor out some of the bits that are shared with the other |
| 837 // Generate* functions. | 842 // Generate* functions. |
| 838 Label runtime; | 843 Label runtime; |
| 839 Label adaptor_frame, try_allocate; | 844 Label adaptor_frame, try_allocate; |
| 840 __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); | 845 __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
| 841 __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); | 846 __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); |
| 842 __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); | 847 __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
| 843 __ j(equal, &adaptor_frame, Label::kNear); | 848 __ j(equal, &adaptor_frame, Label::kNear); |
| 844 | 849 |
| (...skipping 219 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1064 __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); | 1069 __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
| 1065 __ j(equal, &adaptor_frame, Label::kNear); | 1070 __ j(equal, &adaptor_frame, Label::kNear); |
| 1066 | 1071 |
| 1067 // Get the length from the frame. | 1072 // Get the length from the frame. |
| 1068 __ mov(ecx, Operand(esp, 1 * kPointerSize)); | 1073 __ mov(ecx, Operand(esp, 1 * kPointerSize)); |
| 1069 __ jmp(&try_allocate, Label::kNear); | 1074 __ jmp(&try_allocate, Label::kNear); |
| 1070 | 1075 |
| 1071 // Patch the arguments.length and the parameters pointer. | 1076 // Patch the arguments.length and the parameters pointer. |
| 1072 __ bind(&adaptor_frame); | 1077 __ bind(&adaptor_frame); |
| 1073 __ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); | 1078 __ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
| 1074 __ mov(Operand(esp, 1 * kPointerSize), ecx); | 1079 |
| 1080 if (has_new_target()) { |
| 1081 // Subtract 1 from smi-tagged arguments count. |
| 1082 __ sub(ecx, Immediate(2)); |
| 1083 } |
| 1084 |
| 1075 __ lea(edx, Operand(edx, ecx, times_2, | 1085 __ lea(edx, Operand(edx, ecx, times_2, |
| 1076 StandardFrameConstants::kCallerSPOffset)); | 1086 StandardFrameConstants::kCallerSPOffset)); |
| 1087 __ mov(Operand(esp, 1 * kPointerSize), ecx); |
| 1077 __ mov(Operand(esp, 2 * kPointerSize), edx); | 1088 __ mov(Operand(esp, 2 * kPointerSize), edx); |
| 1078 | 1089 |
| 1079 // Try the new space allocation. Start out with computing the size of | 1090 // Try the new space allocation. Start out with computing the size of |
| 1080 // the arguments object and the elements array. | 1091 // the arguments object and the elements array. |
| 1081 Label add_arguments_object; | 1092 Label add_arguments_object; |
| 1082 __ bind(&try_allocate); | 1093 __ bind(&try_allocate); |
| 1083 __ test(ecx, ecx); | 1094 __ test(ecx, ecx); |
| 1084 __ j(zero, &add_arguments_object, Label::kNear); | 1095 __ j(zero, &add_arguments_object, Label::kNear); |
| 1085 __ lea(ecx, Operand(ecx, times_2, FixedArray::kHeaderSize)); | 1096 __ lea(ecx, Operand(ecx, times_2, FixedArray::kHeaderSize)); |
| 1086 __ bind(&add_arguments_object); | 1097 __ bind(&add_arguments_object); |
| (...skipping 1055 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2142 isolate()->factory()->allocation_site_map(); | 2153 isolate()->factory()->allocation_site_map(); |
| 2143 __ cmp(FieldOperand(ebx, 0), Immediate(allocation_site_map)); | 2154 __ cmp(FieldOperand(ebx, 0), Immediate(allocation_site_map)); |
| 2144 __ j(equal, &feedback_register_initialized); | 2155 __ j(equal, &feedback_register_initialized); |
| 2145 __ mov(ebx, isolate()->factory()->undefined_value()); | 2156 __ mov(ebx, isolate()->factory()->undefined_value()); |
| 2146 __ bind(&feedback_register_initialized); | 2157 __ bind(&feedback_register_initialized); |
| 2147 } | 2158 } |
| 2148 | 2159 |
| 2149 __ AssertUndefinedOrAllocationSite(ebx); | 2160 __ AssertUndefinedOrAllocationSite(ebx); |
| 2150 } | 2161 } |
| 2151 | 2162 |
| 2152 // Pass original constructor to construct stub. | 2163 if (IsSuperConstructorCall()) { |
| 2153 __ mov(edx, edi); | 2164 __ mov(edx, Operand(esp, eax, times_pointer_size, 2 * kPointerSize)); |
| 2165 } else { |
| 2166 // Pass original constructor to construct stub. |
| 2167 __ mov(edx, edi); |
| 2168 } |
| 2154 | 2169 |
| 2155 // Jump to the function-specific construct stub. | 2170 // Jump to the function-specific construct stub. |
| 2156 Register jmp_reg = ecx; | 2171 Register jmp_reg = ecx; |
| 2157 __ mov(jmp_reg, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset)); | 2172 __ mov(jmp_reg, FieldOperand(edi, JSFunction::kSharedFunctionInfoOffset)); |
| 2158 __ mov(jmp_reg, FieldOperand(jmp_reg, | 2173 __ mov(jmp_reg, FieldOperand(jmp_reg, |
| 2159 SharedFunctionInfo::kConstructStubOffset)); | 2174 SharedFunctionInfo::kConstructStubOffset)); |
| 2160 __ lea(jmp_reg, FieldOperand(jmp_reg, Code::kHeaderSize)); | 2175 __ lea(jmp_reg, FieldOperand(jmp_reg, Code::kHeaderSize)); |
| 2161 __ jmp(jmp_reg); | 2176 __ jmp(jmp_reg); |
| 2162 | 2177 |
| 2163 // edi: called object | 2178 // edi: called object |
| (...skipping 2934 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 5098 ApiParameterOperand(2), kStackSpace, nullptr, | 5113 ApiParameterOperand(2), kStackSpace, nullptr, |
| 5099 Operand(ebp, 7 * kPointerSize), NULL); | 5114 Operand(ebp, 7 * kPointerSize), NULL); |
| 5100 } | 5115 } |
| 5101 | 5116 |
| 5102 | 5117 |
| 5103 #undef __ | 5118 #undef __ |
| 5104 | 5119 |
| 5105 } } // namespace v8::internal | 5120 } } // namespace v8::internal |
| 5106 | 5121 |
| 5107 #endif // V8_TARGET_ARCH_IA32 | 5122 #endif // V8_TARGET_ARCH_IA32 |
| OLD | NEW |