Index: src/ia32/code-stubs-ia32.cc |
diff --git a/src/ia32/code-stubs-ia32.cc b/src/ia32/code-stubs-ia32.cc |
index 71fcc8d0c3b7ef71bab256ba11b3f2cda0fde70e..0bec1b571852f85b373358e736db72db0f7b4213 100644 |
--- a/src/ia32/code-stubs-ia32.cc |
+++ b/src/ia32/code-stubs-ia32.cc |
@@ -792,72 +792,78 @@ void ArgumentsAccessStub::GenerateReadElement(MacroAssembler* masm) { |
void ArgumentsAccessStub::GenerateNewSloppySlow(MacroAssembler* masm) { |
+ // ecx : number of parameters (tagged) |
+ // edx : parameters pointer |
+ // edi : function |
// esp[0] : return address |
- // esp[4] : number of parameters |
- // esp[8] : receiver displacement |
- // esp[12] : function |
+ |
+ DCHECK(edi.is(ArgumentsAccessNewDescriptor::function())); |
+ DCHECK(ecx.is(ArgumentsAccessNewDescriptor::parameter_count())); |
+ DCHECK(edx.is(ArgumentsAccessNewDescriptor::parameter_pointer())); |
// Check if the calling frame is an arguments adaptor frame. |
Label runtime; |
- __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
- __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); |
- __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
+ __ mov(ebx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
+ __ mov(eax, Operand(ebx, StandardFrameConstants::kContextOffset)); |
+ __ cmp(eax, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
__ j(not_equal, &runtime, Label::kNear); |
// Patch the arguments.length and the parameters pointer. |
- __ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
- __ mov(Operand(esp, 1 * kPointerSize), ecx); |
- __ lea(edx, Operand(edx, ecx, times_2, |
- StandardFrameConstants::kCallerSPOffset)); |
- __ mov(Operand(esp, 2 * kPointerSize), edx); |
+ __ mov(ecx, Operand(ebx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
+ __ lea(edx, |
+ Operand(ebx, ecx, times_2, StandardFrameConstants::kCallerSPOffset)); |
__ bind(&runtime); |
+ __ pop(eax); // Pop return address. |
+ __ push(edi); // Push function. |
+ __ push(edx); // Push parameters pointer. |
+ __ push(ecx); // Push parameter count. |
+ __ push(eax); // Push return address. |
__ TailCallRuntime(Runtime::kNewSloppyArguments, 3, 1); |
} |
void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
+ // ecx : number of parameters (tagged) |
+ // edx : parameters pointer |
+ // edi : function |
// esp[0] : return address |
- // esp[4] : number of parameters (tagged) |
- // esp[8] : receiver displacement |
- // esp[12] : function |
- // ebx = parameter count (tagged) |
- __ mov(ebx, Operand(esp, 1 * kPointerSize)); |
+ DCHECK(edi.is(ArgumentsAccessNewDescriptor::function())); |
+ DCHECK(ecx.is(ArgumentsAccessNewDescriptor::parameter_count())); |
+ DCHECK(edx.is(ArgumentsAccessNewDescriptor::parameter_pointer())); |
// Check if the calling frame is an arguments adaptor frame. |
- // TODO(rossberg): Factor out some of the bits that are shared with the other |
- // Generate* functions. |
- Label runtime; |
- Label adaptor_frame, try_allocate; |
- __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
- __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); |
- __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
+ Label adaptor_frame, try_allocate, runtime; |
+ __ mov(ebx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
+ __ mov(eax, Operand(ebx, StandardFrameConstants::kContextOffset)); |
+ __ cmp(eax, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
__ j(equal, &adaptor_frame, Label::kNear); |
// No adaptor, parameter count = argument count. |
- __ mov(ecx, ebx); |
+ __ mov(ebx, ecx); |
+ __ push(ecx); |
__ jmp(&try_allocate, Label::kNear); |
// We have an adaptor frame. Patch the parameters pointer. |
__ bind(&adaptor_frame); |
+ __ mov(ebx, ecx); |
+ __ push(ecx); |
+ __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
__ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
__ lea(edx, Operand(edx, ecx, times_2, |
StandardFrameConstants::kCallerSPOffset)); |
- __ mov(Operand(esp, 2 * kPointerSize), edx); |
// ebx = parameter count (tagged) |
// ecx = argument count (smi-tagged) |
- // esp[4] = parameter count (tagged) |
- // esp[8] = address of receiver argument |
// Compute the mapped parameter count = min(ebx, ecx) in ebx. |
__ cmp(ebx, ecx); |
__ j(less_equal, &try_allocate, Label::kNear); |
__ mov(ebx, ecx); |
+ // Save mapped parameter count and function. |
__ bind(&try_allocate); |
- |
- // Save mapped parameter count. |
+ __ push(edi); |
__ push(ebx); |
// Compute the sizes of backing store, parameter map, and arguments object. |
@@ -877,13 +883,13 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
__ add(ebx, Immediate(Heap::kSloppyArgumentsObjectSize)); |
// Do the allocation of all three objects in one go. |
- __ Allocate(ebx, eax, edx, edi, &runtime, TAG_OBJECT); |
+ __ Allocate(ebx, eax, edi, no_reg, &runtime, TAG_OBJECT); |
// eax = address of new object(s) (tagged) |
// ecx = argument count (smi-tagged) |
// esp[0] = mapped parameter count (tagged) |
+ // esp[4] = function |
// esp[8] = parameter count (tagged) |
- // esp[12] = address of receiver argument |
// Get the arguments map from the current native context into edi. |
Label has_mapped_parameters, instantiate; |
__ mov(edi, Operand(esi, Context::SlotOffset(Context::GLOBAL_OBJECT_INDEX))); |
@@ -906,8 +912,8 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// ecx = argument count (smi-tagged) |
// edi = address of arguments map (tagged) |
// esp[0] = mapped parameter count (tagged) |
+ // esp[4] = function |
// esp[8] = parameter count (tagged) |
- // esp[12] = address of receiver argument |
// Copy the JS object part. |
__ mov(FieldOperand(eax, JSObject::kMapOffset), edi); |
__ mov(FieldOperand(eax, JSObject::kPropertiesOffset), |
@@ -917,11 +923,11 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// Set up the callee in-object property. |
STATIC_ASSERT(Heap::kArgumentsCalleeIndex == 1); |
- __ mov(edx, Operand(esp, 4 * kPointerSize)); |
- __ AssertNotSmi(edx); |
+ __ mov(edi, Operand(esp, 1 * kPointerSize)); |
+ __ AssertNotSmi(edi); |
__ mov(FieldOperand(eax, JSObject::kHeaderSize + |
- Heap::kArgumentsCalleeIndex * kPointerSize), |
- edx); |
+ Heap::kArgumentsCalleeIndex * kPointerSize), |
+ edi); |
// Use the length (smi tagged) and set that as an in-object property too. |
__ AssertSmi(ecx); |
@@ -939,11 +945,13 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// eax = address of new object (tagged) |
// ebx = mapped parameter count (tagged) |
// ecx = argument count (tagged) |
+ // edx = address of receiver argument |
// edi = address of parameter map or backing store (tagged) |
// esp[0] = mapped parameter count (tagged) |
+ // esp[4] = function |
// esp[8] = parameter count (tagged) |
- // esp[12] = address of receiver argument |
- // Free a register. |
+ // Free two registers. |
+ __ push(edx); |
__ push(eax); |
// Initialize parameter map. If there are no mapped arguments, we're done. |
@@ -969,9 +977,9 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// We loop from right to left. |
Label parameters_loop, parameters_test; |
__ push(ecx); |
- __ mov(eax, Operand(esp, 2 * kPointerSize)); |
+ __ mov(eax, Operand(esp, 3 * kPointerSize)); |
__ mov(ebx, Immediate(Smi::FromInt(Context::MIN_CONTEXT_SLOTS))); |
- __ add(ebx, Operand(esp, 4 * kPointerSize)); |
+ __ add(ebx, Operand(esp, 5 * kPointerSize)); |
__ sub(ebx, eax); |
__ mov(ecx, isolate()->factory()->the_hole_value()); |
__ mov(edx, edi); |
@@ -983,9 +991,10 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// edi = address of backing store (tagged) |
// esp[0] = argument count (tagged) |
// esp[4] = address of new object (tagged) |
- // esp[8] = mapped parameter count (tagged) |
- // esp[16] = parameter count (tagged) |
- // esp[20] = address of receiver argument |
+ // esp[8] = address of receiver argument |
+ // esp[12] = mapped parameter count (tagged) |
+ // esp[16] = function |
+ // esp[20] = parameter count (tagged) |
__ jmp(¶meters_test, Label::kNear); |
__ bind(¶meters_loop); |
@@ -1003,17 +1012,18 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// ecx = argument count (tagged) |
// edi = address of backing store (tagged) |
// esp[0] = address of new object (tagged) |
- // esp[4] = mapped parameter count (tagged) |
- // esp[12] = parameter count (tagged) |
- // esp[16] = address of receiver argument |
+ // esp[4] = address of receiver argument |
+ // esp[8] = mapped parameter count (tagged) |
+ // esp[12] = function |
+ // esp[16] = parameter count (tagged) |
// Copy arguments header and remaining slots (if there are any). |
__ mov(FieldOperand(edi, FixedArray::kMapOffset), |
Immediate(isolate()->factory()->fixed_array_map())); |
__ mov(FieldOperand(edi, FixedArray::kLengthOffset), ecx); |
Label arguments_loop, arguments_test; |
- __ mov(ebx, Operand(esp, 1 * kPointerSize)); |
- __ mov(edx, Operand(esp, 4 * kPointerSize)); |
+ __ mov(ebx, Operand(esp, 2 * kPointerSize)); |
+ __ mov(edx, Operand(esp, 1 * kPointerSize)); |
__ sub(edx, ebx); // Is there a smarter way to do negative scaling? |
__ sub(edx, ebx); |
__ jmp(&arguments_test, Label::kNear); |
@@ -1030,57 +1040,60 @@ void ArgumentsAccessStub::GenerateNewSloppyFast(MacroAssembler* masm) { |
// Restore. |
__ pop(eax); // Address of arguments object. |
- __ pop(ebx); // Parameter count. |
+ __ Drop(4); |
- // Return and remove the on-stack parameters. |
- __ ret(3 * kPointerSize); |
+ // Return. |
+ __ ret(0); |
// Do the runtime call to allocate the arguments object. |
__ bind(&runtime); |
- __ pop(eax); // Remove saved parameter count. |
- __ mov(Operand(esp, 1 * kPointerSize), ecx); // Patch argument count. |
+ __ pop(eax); // Remove saved mapped parameter count. |
+ __ pop(edi); // Pop saved function. |
+ __ pop(eax); // Remove saved parameter count. |
+ __ pop(eax); // Pop return address. |
+ __ push(edi); // Push function. |
+ __ push(edx); // Push parameters pointer. |
+ __ push(ecx); // Push parameter count. |
+ __ push(eax); // Push return address. |
__ TailCallRuntime(Runtime::kNewSloppyArguments, 3, 1); |
} |
void ArgumentsAccessStub::GenerateNewStrict(MacroAssembler* masm) { |
+ // ecx : number of parameters (tagged) |
+ // edx : parameters pointer |
+ // edi : function |
// esp[0] : return address |
- // esp[4] : number of parameters |
- // esp[8] : receiver displacement |
- // esp[12] : function |
- // Check if the calling frame is an arguments adaptor frame. |
- Label adaptor_frame, try_allocate, runtime; |
- __ mov(edx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
- __ mov(ecx, Operand(edx, StandardFrameConstants::kContextOffset)); |
- __ cmp(ecx, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
- __ j(equal, &adaptor_frame, Label::kNear); |
+ DCHECK(edi.is(ArgumentsAccessNewDescriptor::function())); |
+ DCHECK(ecx.is(ArgumentsAccessNewDescriptor::parameter_count())); |
+ DCHECK(edx.is(ArgumentsAccessNewDescriptor::parameter_pointer())); |
- // Get the length from the frame. |
- __ mov(ecx, Operand(esp, 1 * kPointerSize)); |
- __ jmp(&try_allocate, Label::kNear); |
+ // Check if the calling frame is an arguments adaptor frame. |
+ Label try_allocate, runtime; |
+ __ mov(ebx, Operand(ebp, StandardFrameConstants::kCallerFPOffset)); |
+ __ mov(eax, Operand(ebx, StandardFrameConstants::kContextOffset)); |
+ __ cmp(eax, Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR))); |
+ __ j(not_equal, &try_allocate, Label::kNear); |
// Patch the arguments.length and the parameters pointer. |
- __ bind(&adaptor_frame); |
- __ mov(ecx, Operand(edx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
- |
- __ lea(edx, Operand(edx, ecx, times_2, |
- StandardFrameConstants::kCallerSPOffset)); |
- __ mov(Operand(esp, 1 * kPointerSize), ecx); |
- __ mov(Operand(esp, 2 * kPointerSize), edx); |
+ __ mov(ecx, Operand(ebx, ArgumentsAdaptorFrameConstants::kLengthOffset)); |
+ __ lea(edx, |
+ Operand(ebx, ecx, times_2, StandardFrameConstants::kCallerSPOffset)); |
// Try the new space allocation. Start out with computing the size of |
// the arguments object and the elements array. |
Label add_arguments_object; |
__ bind(&try_allocate); |
- __ test(ecx, ecx); |
+ __ mov(eax, ecx); |
+ __ test(eax, eax); |
__ j(zero, &add_arguments_object, Label::kNear); |
- __ lea(ecx, Operand(ecx, times_2, FixedArray::kHeaderSize)); |
+ __ lea(eax, Operand(eax, times_2, FixedArray::kHeaderSize)); |
__ bind(&add_arguments_object); |
- __ add(ecx, Immediate(Heap::kStrictArgumentsObjectSize)); |
+ __ add(eax, Immediate(Heap::kStrictArgumentsObjectSize)); |
// Do the allocation of both objects in one go. |
- __ Allocate(ecx, eax, edx, ebx, &runtime, TAG_OBJECT); |
+ __ Allocate(eax, eax, ebx, no_reg, &runtime, TAG_OBJECT); |
// Get the arguments map from the current native context. |
__ mov(edi, Operand(esi, Context::SlotOffset(Context::GLOBAL_OBJECT_INDEX))); |
@@ -1096,7 +1109,6 @@ void ArgumentsAccessStub::GenerateNewStrict(MacroAssembler* masm) { |
// Get the length (smi tagged) and set that as an in-object property too. |
STATIC_ASSERT(Heap::kArgumentsLengthIndex == 0); |
- __ mov(ecx, Operand(esp, 1 * kPointerSize)); |
__ AssertSmi(ecx); |
__ mov(FieldOperand(eax, JSObject::kHeaderSize + |
Heap::kArgumentsLengthIndex * kPointerSize), |
@@ -1107,17 +1119,14 @@ void ArgumentsAccessStub::GenerateNewStrict(MacroAssembler* masm) { |
__ test(ecx, ecx); |
__ j(zero, &done, Label::kNear); |
- // Get the parameters pointer from the stack. |
- __ mov(edx, Operand(esp, 2 * kPointerSize)); |
- |
// Set up the elements pointer in the allocated arguments object and |
// initialize the header in the elements fixed array. |
__ lea(edi, Operand(eax, Heap::kStrictArgumentsObjectSize)); |
__ mov(FieldOperand(eax, JSObject::kElementsOffset), edi); |
__ mov(FieldOperand(edi, FixedArray::kMapOffset), |
Immediate(isolate()->factory()->fixed_array_map())); |
- |
__ mov(FieldOperand(edi, FixedArray::kLengthOffset), ecx); |
+ |
// Untag the length for the loop below. |
__ SmiUntag(ecx); |
@@ -1131,12 +1140,17 @@ void ArgumentsAccessStub::GenerateNewStrict(MacroAssembler* masm) { |
__ dec(ecx); |
__ j(not_zero, &loop); |
- // Return and remove the on-stack parameters. |
+ // Return. |
__ bind(&done); |
- __ ret(3 * kPointerSize); |
+ __ ret(0); |
// Do the runtime call to allocate the arguments object. |
__ bind(&runtime); |
+ __ pop(eax); // Pop return address. |
+ __ push(edi); // Push function. |
+ __ push(edx); // Push parameters pointer. |
+ __ push(ecx); // Push parameter count. |
+ __ push(eax); // Push return address. |
__ TailCallRuntime(Runtime::kNewStrictArguments, 3, 1); |
} |