Chromium Code Reviews| 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_X64 | 7 #if V8_TARGET_ARCH_X64 |
| 8 | 8 |
| 9 #include "src/code-factory.h" | 9 #include "src/code-factory.h" |
| 10 #include "src/code-stubs.h" | 10 #include "src/code-stubs.h" |
| (...skipping 95 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 106 | 106 |
| 107 #ifdef DEBUG | 107 #ifdef DEBUG |
| 108 if (strlen(FLAG_stop_at) > 0 && | 108 if (strlen(FLAG_stop_at) > 0 && |
| 109 info->function()->name()->IsUtf8EqualTo(CStrVector(FLAG_stop_at))) { | 109 info->function()->name()->IsUtf8EqualTo(CStrVector(FLAG_stop_at))) { |
| 110 __ int3(); | 110 __ int3(); |
| 111 } | 111 } |
| 112 #endif | 112 #endif |
| 113 | 113 |
| 114 // Sloppy mode functions and builtins need to replace the receiver with the | 114 // Sloppy mode functions and builtins need to replace the receiver with the |
| 115 // global proxy when called as functions (without an explicit receiver | 115 // global proxy when called as functions (without an explicit receiver |
| 116 // object). | 116 // object). Arrow functions need to replace the receiver with the receiver |
| 117 if (info->strict_mode() == SLOPPY && !info->is_native()) { | 117 // of the parent scope, so this does not need to be done for those. |
|
wingo
2015/01/15 10:09:06
It's not necessarily the receiver from the parent
aperez
2015/01/15 16:58:03
Done.
| |
| 118 if (info->strict_mode() == SLOPPY && !info->is_native() && | |
| 119 !info->function()->is_arrow()) { | |
| 118 Label ok; | 120 Label ok; |
| 119 // +1 for return address. | 121 // +1 for return address. |
| 120 StackArgumentsAccessor args(rsp, info->scope()->num_parameters()); | 122 StackArgumentsAccessor args(rsp, info->scope()->num_parameters()); |
| 121 __ movp(rcx, args.GetReceiverOperand()); | 123 __ movp(rcx, args.GetReceiverOperand()); |
| 122 | 124 |
| 123 __ CompareRoot(rcx, Heap::kUndefinedValueRootIndex); | 125 __ CompareRoot(rcx, Heap::kUndefinedValueRootIndex); |
| 124 __ j(not_equal, &ok, Label::kNear); | 126 __ j(not_equal, &ok, Label::kNear); |
| 125 | 127 |
| 126 __ movp(rcx, GlobalObjectOperand()); | 128 __ movp(rcx, GlobalObjectOperand()); |
| 127 __ movp(rcx, FieldOperand(rcx, GlobalObject::kGlobalProxyOffset)); | 129 __ movp(rcx, FieldOperand(rcx, GlobalObject::kGlobalProxyOffset)); |
| 128 | 130 |
| 129 __ movp(args.GetReceiverOperand(), rcx); | 131 __ movp(args.GetReceiverOperand(), rcx); |
| 130 | 132 |
| 131 __ bind(&ok); | 133 __ bind(&ok); |
| 132 } | 134 } |
| 133 | 135 |
| 134 // Open a frame scope to indicate that there is a frame on the stack. The | 136 // Open a frame scope to indicate that there is a frame on the stack. The |
| 135 // MANUAL indicates that the scope shouldn't actually generate code to set up | 137 // MANUAL indicates that the scope shouldn't actually generate code to set up |
| 136 // the frame (that is done below). | 138 // the frame (that is done below). |
| 137 FrameScope frame_scope(masm_, StackFrame::MANUAL); | 139 FrameScope frame_scope(masm_, StackFrame::MANUAL); |
| 138 | 140 |
| 139 info->set_prologue_offset(masm_->pc_offset()); | 141 info->set_prologue_offset(masm_->pc_offset()); |
| 140 __ Prologue(info->IsCodePreAgingActive()); | 142 __ Prologue(info->IsCodePreAgingActive()); |
| 141 info->AddNoFrameRange(0, masm_->pc_offset()); | 143 info->AddNoFrameRange(0, masm_->pc_offset()); |
| 142 | 144 |
| 145 // For arrow functions the value of the receiver is stored in the context, | |
| 146 // so in the case of a function that actually uses "this", the receiver is | |
| 147 // replaced by the one resulting from looking it up in the context. | |
| 148 if (info->function()->is_arrow() && info->scope()->uses_this()) { | |
| 149 DCHECK(info->scope()->scope_type() == ARROW_SCOPE); | |
| 150 Comment cmnt(masm_, "[ Patch receiver for arrow function"); | |
| 151 StackArgumentsAccessor args(rbp, info->scope()->num_parameters()); | |
| 152 __ Push(rsi); // Context | |
| 153 __ Push(isolate()->factory()->this_string()); | |
| 154 __ CallRuntime(Runtime::kLoadLookupSlot, 2); | |
| 155 __ movp(args.GetReceiverOperand(), rax); | |
| 156 } | |
| 157 | |
| 143 { Comment cmnt(masm_, "[ Allocate locals"); | 158 { Comment cmnt(masm_, "[ Allocate locals"); |
| 144 int locals_count = info->scope()->num_stack_slots(); | 159 int locals_count = info->scope()->num_stack_slots(); |
| 145 // Generators allocate locals, if any, in context slots. | 160 // Generators allocate locals, if any, in context slots. |
| 146 DCHECK(!info->function()->is_generator() || locals_count == 0); | 161 DCHECK(!info->function()->is_generator() || locals_count == 0); |
| 147 if (locals_count == 1) { | 162 if (locals_count == 1) { |
| 148 __ PushRoot(Heap::kUndefinedValueRootIndex); | 163 __ PushRoot(Heap::kUndefinedValueRootIndex); |
| 149 } else if (locals_count > 1) { | 164 } else if (locals_count > 1) { |
| 150 if (locals_count >= 128) { | 165 if (locals_count >= 128) { |
| 151 Label ok; | 166 Label ok; |
| 152 __ movp(rcx, rsp); | 167 __ movp(rcx, rsp); |
| (...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 200 __ Push(rdi); | 215 __ Push(rdi); |
| 201 __ CallRuntime(Runtime::kNewFunctionContext, 1); | 216 __ CallRuntime(Runtime::kNewFunctionContext, 1); |
| 202 } | 217 } |
| 203 function_in_register = false; | 218 function_in_register = false; |
| 204 // Context is returned in rax. It replaces the context passed to us. | 219 // Context is returned in rax. It replaces the context passed to us. |
| 205 // It's saved in the stack and kept live in rsi. | 220 // It's saved in the stack and kept live in rsi. |
| 206 __ movp(rsi, rax); | 221 __ movp(rsi, rax); |
| 207 __ movp(Operand(rbp, StandardFrameConstants::kContextOffset), rax); | 222 __ movp(Operand(rbp, StandardFrameConstants::kContextOffset), rax); |
| 208 | 223 |
| 209 // Copy any necessary parameters into the context. | 224 // Copy any necessary parameters into the context. |
| 225 if (info->scope()->inner_uses_this()) { | |
|
wingo
2015/01/15 10:09:06
Let's factor this into the loop below
aperez
2015/01/15 16:58:03
Done.
| |
| 226 Comment cmnt(masm_, "[ Save receiver in context for arrow function"); | |
| 227 DCHECK(info->scope()->receiver()->IsContextSlot()); | |
| 228 // Put the "this" in the context after the rest of the function | |
| 229 // parameters, to avoid having to change the case that handles those. | |
| 230 int parameter_offset = StandardFrameConstants::kCallerSPOffset + | |
| 231 info->scope()->num_parameters() * kPointerSize; | |
| 232 __ movp(rax, Operand(rbp, parameter_offset)); | |
| 233 | |
| 234 // Store it in the context. | |
| 235 int context_offset = Context::SlotOffset(scope()->receiver()->index()); | |
| 236 __ movp(Operand(rsi, context_offset), rax); | |
| 237 | |
| 238 if (need_write_barrier) { | |
| 239 __ RecordWriteContextSlot( | |
| 240 rsi, context_offset, rax, rbx, kDontSaveFPRegs); | |
| 241 } else if (FLAG_debug_code) { | |
| 242 Label done; | |
| 243 __ JumpIfInNewSpace(rsi, rax, &done, Label::kNear); | |
| 244 __ Abort(kExpectedNewSpaceObject); | |
| 245 __ bind(&done); | |
| 246 } | |
| 247 } | |
| 248 | |
| 210 int num_parameters = info->scope()->num_parameters(); | 249 int num_parameters = info->scope()->num_parameters(); |
| 211 for (int i = 0; i < num_parameters; i++) { | 250 for (int i = 0; i < num_parameters; i++) { |
| 212 Variable* var = scope()->parameter(i); | 251 Variable* var = scope()->parameter(i); |
| 213 if (var->IsContextSlot()) { | 252 if (var->IsContextSlot()) { |
| 214 int parameter_offset = StandardFrameConstants::kCallerSPOffset + | 253 int parameter_offset = StandardFrameConstants::kCallerSPOffset + |
| 215 (num_parameters - 1 - i) * kPointerSize; | 254 (num_parameters - 1 - i) * kPointerSize; |
| 216 // Load parameter from stack. | 255 // Load parameter from stack. |
| 217 __ movp(rax, Operand(rbp, parameter_offset)); | 256 __ movp(rax, Operand(rbp, parameter_offset)); |
| 218 // Store it in the context. | 257 // Store it in the context. |
| 219 int context_offset = Context::SlotOffset(var->index()); | 258 int context_offset = Context::SlotOffset(var->index()); |
| (...skipping 2494 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 2714 | 2753 |
| 2715 CallICState::CallType call_type = | 2754 CallICState::CallType call_type = |
| 2716 callee->IsVariableProxy() ? CallICState::FUNCTION : CallICState::METHOD; | 2755 callee->IsVariableProxy() ? CallICState::FUNCTION : CallICState::METHOD; |
| 2717 // Get the target function. | 2756 // Get the target function. |
| 2718 if (call_type == CallICState::FUNCTION) { | 2757 if (call_type == CallICState::FUNCTION) { |
| 2719 { StackValueContext context(this); | 2758 { StackValueContext context(this); |
| 2720 EmitVariableLoad(callee->AsVariableProxy()); | 2759 EmitVariableLoad(callee->AsVariableProxy()); |
| 2721 PrepareForBailout(callee, NO_REGISTERS); | 2760 PrepareForBailout(callee, NO_REGISTERS); |
| 2722 } | 2761 } |
| 2723 // Push undefined as receiver. This is patched in the method prologue if it | 2762 // Push undefined as receiver. This is patched in the method prologue if it |
| 2724 // is a sloppy mode method. | 2763 // is a sloppy mode method, or by the function prologue it is an arrow |
|
wingo
2015/01/15 10:09:06
"if it is"
aperez
2015/01/15 16:58:03
Done.
| |
| 2764 // function. | |
| 2725 __ Push(isolate()->factory()->undefined_value()); | 2765 __ Push(isolate()->factory()->undefined_value()); |
| 2726 } else { | 2766 } else { |
| 2727 // Load the function from the receiver. | 2767 // Load the function from the receiver. |
| 2728 DCHECK(callee->IsProperty()); | 2768 DCHECK(callee->IsProperty()); |
| 2729 DCHECK(!callee->AsProperty()->IsSuperAccess()); | 2769 DCHECK(!callee->AsProperty()->IsSuperAccess()); |
| 2730 __ movp(LoadDescriptor::ReceiverRegister(), Operand(rsp, 0)); | 2770 __ movp(LoadDescriptor::ReceiverRegister(), Operand(rsp, 0)); |
| 2731 EmitNamedPropertyLoad(callee->AsProperty()); | 2771 EmitNamedPropertyLoad(callee->AsProperty()); |
| 2732 PrepareForBailoutForId(callee->AsProperty()->LoadId(), TOS_REG); | 2772 PrepareForBailoutForId(callee->AsProperty()->LoadId(), TOS_REG); |
| 2733 // Push the target function under the receiver. | 2773 // Push the target function under the receiver. |
| 2734 __ Push(Operand(rsp, 0)); | 2774 __ Push(Operand(rsp, 0)); |
| (...skipping 2458 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 5193 DCHECK_EQ(isolate->builtins()->OsrAfterStackCheck()->entry(), | 5233 DCHECK_EQ(isolate->builtins()->OsrAfterStackCheck()->entry(), |
| 5194 Assembler::target_address_at(call_target_address, | 5234 Assembler::target_address_at(call_target_address, |
| 5195 unoptimized_code)); | 5235 unoptimized_code)); |
| 5196 return OSR_AFTER_STACK_CHECK; | 5236 return OSR_AFTER_STACK_CHECK; |
| 5197 } | 5237 } |
| 5198 | 5238 |
| 5199 | 5239 |
| 5200 } } // namespace v8::internal | 5240 } } // namespace v8::internal |
| 5201 | 5241 |
| 5202 #endif // V8_TARGET_ARCH_X64 | 5242 #endif // V8_TARGET_ARCH_X64 |
| OLD | NEW |