| OLD | NEW |
| 1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 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 #if V8_TARGET_ARCH_PPC | 5 #if V8_TARGET_ARCH_PPC |
| 6 | 6 |
| 7 #include "src/code-factory.h" | 7 #include "src/code-factory.h" |
| 8 #include "src/code-stubs.h" | 8 #include "src/code-stubs.h" |
| 9 #include "src/codegen.h" | 9 #include "src/codegen.h" |
| 10 #include "src/compiler.h" | 10 #include "src/compiler.h" |
| (...skipping 168 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 179 __ bdnz(&loop_header); | 179 __ bdnz(&loop_header); |
| 180 } | 180 } |
| 181 int remaining = locals_count % kMaxPushes; | 181 int remaining = locals_count % kMaxPushes; |
| 182 // Emit the remaining pushes. | 182 // Emit the remaining pushes. |
| 183 for (int i = 0; i < remaining; i++) { | 183 for (int i = 0; i < remaining; i++) { |
| 184 __ push(ip); | 184 __ push(ip); |
| 185 } | 185 } |
| 186 } | 186 } |
| 187 } | 187 } |
| 188 | 188 |
| 189 bool function_in_register = true; | 189 bool function_in_register_r4 = true; |
| 190 | 190 |
| 191 // Possibly allocate a local context. | 191 // Possibly allocate a local context. |
| 192 if (info->scope()->num_heap_slots() > 0) { | 192 if (info->scope()->num_heap_slots() > 0) { |
| 193 // Argument to NewContext is the function, which is still in r4. | 193 // Argument to NewContext is the function, which is still in r4. |
| 194 Comment cmnt(masm_, "[ Allocate context"); | 194 Comment cmnt(masm_, "[ Allocate context"); |
| 195 bool need_write_barrier = true; | 195 bool need_write_barrier = true; |
| 196 int slots = info->scope()->num_heap_slots() - Context::MIN_CONTEXT_SLOTS; | 196 int slots = info->scope()->num_heap_slots() - Context::MIN_CONTEXT_SLOTS; |
| 197 if (info->scope()->is_script_scope()) { | 197 if (info->scope()->is_script_scope()) { |
| 198 __ push(r4); | 198 __ push(r4); |
| 199 __ Push(info->scope()->GetScopeInfo(info->isolate())); | 199 __ Push(info->scope()->GetScopeInfo(info->isolate())); |
| 200 __ CallRuntime(Runtime::kNewScriptContext, 2); | 200 __ CallRuntime(Runtime::kNewScriptContext, 2); |
| 201 } else if (slots <= FastNewContextStub::kMaximumSlots) { | 201 } else if (slots <= FastNewContextStub::kMaximumSlots) { |
| 202 FastNewContextStub stub(isolate(), slots); | 202 FastNewContextStub stub(isolate(), slots); |
| 203 __ CallStub(&stub); | 203 __ CallStub(&stub); |
| 204 // Result of FastNewContextStub is always in new space. | 204 // Result of FastNewContextStub is always in new space. |
| 205 need_write_barrier = false; | 205 need_write_barrier = false; |
| 206 } else { | 206 } else { |
| 207 __ push(r4); | 207 __ push(r4); |
| 208 __ CallRuntime(Runtime::kNewFunctionContext, 1); | 208 __ CallRuntime(Runtime::kNewFunctionContext, 1); |
| 209 } | 209 } |
| 210 function_in_register = false; | 210 function_in_register_r4 = false; |
| 211 // Context is returned in r3. It replaces the context passed to us. | 211 // Context is returned in r3. It replaces the context passed to us. |
| 212 // It's saved in the stack and kept live in cp. | 212 // It's saved in the stack and kept live in cp. |
| 213 __ mr(cp, r3); | 213 __ mr(cp, r3); |
| 214 __ StoreP(r3, MemOperand(fp, StandardFrameConstants::kContextOffset)); | 214 __ StoreP(r3, MemOperand(fp, StandardFrameConstants::kContextOffset)); |
| 215 // Copy any necessary parameters into the context. | 215 // Copy any necessary parameters into the context. |
| 216 int num_parameters = info->scope()->num_parameters(); | 216 int num_parameters = info->scope()->num_parameters(); |
| 217 int first_parameter = info->scope()->has_this_declaration() ? -1 : 0; | 217 int first_parameter = info->scope()->has_this_declaration() ? -1 : 0; |
| 218 for (int i = first_parameter; i < num_parameters; i++) { | 218 for (int i = first_parameter; i < num_parameters; i++) { |
| 219 Variable* var = (i == -1) ? scope()->receiver() : scope()->parameter(i); | 219 Variable* var = (i == -1) ? scope()->receiver() : scope()->parameter(i); |
| 220 if (var->IsContextSlot()) { | 220 if (var->IsContextSlot()) { |
| (...skipping 12 matching lines...) Expand all Loading... |
| 233 } else if (FLAG_debug_code) { | 233 } else if (FLAG_debug_code) { |
| 234 Label done; | 234 Label done; |
| 235 __ JumpIfInNewSpace(cp, r3, &done); | 235 __ JumpIfInNewSpace(cp, r3, &done); |
| 236 __ Abort(kExpectedNewSpaceObject); | 236 __ Abort(kExpectedNewSpaceObject); |
| 237 __ bind(&done); | 237 __ bind(&done); |
| 238 } | 238 } |
| 239 } | 239 } |
| 240 } | 240 } |
| 241 } | 241 } |
| 242 | 242 |
| 243 PrepareForBailoutForId(BailoutId::Prologue(), NO_REGISTERS); |
| 244 // Function register is trashed in case we bailout here. But since that |
| 245 // could happen only when we allocate a context the value of |
| 246 // |function_in_register_r4| is correct. |
| 247 |
| 243 // Possibly set up a local binding to the this function which is used in | 248 // Possibly set up a local binding to the this function which is used in |
| 244 // derived constructors with super calls. | 249 // derived constructors with super calls. |
| 245 Variable* this_function_var = scope()->this_function_var(); | 250 Variable* this_function_var = scope()->this_function_var(); |
| 246 if (this_function_var != nullptr) { | 251 if (this_function_var != nullptr) { |
| 247 Comment cmnt(masm_, "[ This function"); | 252 Comment cmnt(masm_, "[ This function"); |
| 248 if (!function_in_register) { | 253 if (!function_in_register_r4) { |
| 249 __ LoadP(r4, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); | 254 __ LoadP(r4, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); |
| 250 // The write barrier clobbers register again, keep is marked as such. | 255 // The write barrier clobbers register again, keep it marked as such. |
| 251 } | 256 } |
| 252 SetVar(this_function_var, r4, r3, r5); | 257 SetVar(this_function_var, r4, r3, r5); |
| 253 } | 258 } |
| 254 | 259 |
| 255 Variable* new_target_var = scope()->new_target_var(); | 260 Variable* new_target_var = scope()->new_target_var(); |
| 256 if (new_target_var != nullptr) { | 261 if (new_target_var != nullptr) { |
| 257 Comment cmnt(masm_, "[ new.target"); | 262 Comment cmnt(masm_, "[ new.target"); |
| 258 | 263 |
| 259 // Get the frame pointer for the calling frame. | 264 // Get the frame pointer for the calling frame. |
| 260 __ LoadP(r5, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); | 265 __ LoadP(r5, MemOperand(fp, StandardFrameConstants::kCallerFPOffset)); |
| 261 | 266 |
| 262 // Skip the arguments adaptor frame if it exists. | 267 // Skip the arguments adaptor frame if it exists. |
| 263 __ LoadP(r4, MemOperand(r5, StandardFrameConstants::kContextOffset)); | 268 __ LoadP(r4, MemOperand(r5, StandardFrameConstants::kContextOffset)); |
| 264 __ CmpSmiLiteral(r4, Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR), r0); | 269 __ CmpSmiLiteral(r4, Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR), r0); |
| 265 Label skip; | 270 Label skip; |
| 266 __ bne(&skip); | 271 __ bne(&skip); |
| 267 __ LoadP(r5, MemOperand(r5, StandardFrameConstants::kCallerFPOffset)); | 272 __ LoadP(r5, MemOperand(r5, StandardFrameConstants::kCallerFPOffset)); |
| 268 __ bind(&skip); | 273 __ bind(&skip); |
| 269 | 274 |
| 270 // Check the marker in the calling frame. | 275 // Check the marker in the calling frame. |
| 271 __ LoadP(r4, MemOperand(r5, StandardFrameConstants::kMarkerOffset)); | 276 __ LoadP(r4, MemOperand(r5, StandardFrameConstants::kMarkerOffset)); |
| 272 __ CmpSmiLiteral(r4, Smi::FromInt(StackFrame::CONSTRUCT), r0); | 277 __ CmpSmiLiteral(r4, Smi::FromInt(StackFrame::CONSTRUCT), r0); |
| 273 Label non_construct_frame, done; | 278 Label non_construct_frame, done; |
| 279 function_in_register_r4 = false; |
| 274 | 280 |
| 275 __ bne(&non_construct_frame); | 281 __ bne(&non_construct_frame); |
| 276 __ LoadP(r3, MemOperand( | 282 __ LoadP(r3, MemOperand( |
| 277 r5, ConstructFrameConstants::kOriginalConstructorOffset)); | 283 r5, ConstructFrameConstants::kOriginalConstructorOffset)); |
| 278 __ b(&done); | 284 __ b(&done); |
| 279 | 285 |
| 280 __ bind(&non_construct_frame); | 286 __ bind(&non_construct_frame); |
| 281 __ LoadRoot(r3, Heap::kUndefinedValueRootIndex); | 287 __ LoadRoot(r3, Heap::kUndefinedValueRootIndex); |
| 282 __ bind(&done); | 288 __ bind(&done); |
| 283 | 289 |
| 284 SetVar(new_target_var, r3, r5, r6); | 290 SetVar(new_target_var, r3, r5, r6); |
| 285 } | 291 } |
| 286 | 292 |
| 287 // Possibly allocate RestParameters | 293 // Possibly allocate RestParameters |
| 288 int rest_index; | 294 int rest_index; |
| 289 Variable* rest_param = scope()->rest_parameter(&rest_index); | 295 Variable* rest_param = scope()->rest_parameter(&rest_index); |
| 290 if (rest_param) { | 296 if (rest_param) { |
| 291 Comment cmnt(masm_, "[ Allocate rest parameter array"); | 297 Comment cmnt(masm_, "[ Allocate rest parameter array"); |
| 292 | 298 |
| 293 int num_parameters = info->scope()->num_parameters(); | 299 int num_parameters = info->scope()->num_parameters(); |
| 294 int offset = num_parameters * kPointerSize; | 300 int offset = num_parameters * kPointerSize; |
| 295 | 301 |
| 296 __ addi(r6, fp, Operand(StandardFrameConstants::kCallerSPOffset + offset)); | 302 __ addi(r6, fp, Operand(StandardFrameConstants::kCallerSPOffset + offset)); |
| 297 __ LoadSmiLiteral(r5, Smi::FromInt(num_parameters)); | 303 __ LoadSmiLiteral(r5, Smi::FromInt(num_parameters)); |
| 298 __ LoadSmiLiteral(r4, Smi::FromInt(rest_index)); | 304 __ LoadSmiLiteral(r4, Smi::FromInt(rest_index)); |
| 299 __ LoadSmiLiteral(r3, Smi::FromInt(language_mode())); | 305 __ LoadSmiLiteral(r3, Smi::FromInt(language_mode())); |
| 300 __ Push(r6, r5, r4, r3); | 306 __ Push(r6, r5, r4, r3); |
| 307 function_in_register_r4 = false; |
| 301 | 308 |
| 302 RestParamAccessStub stub(isolate()); | 309 RestParamAccessStub stub(isolate()); |
| 303 __ CallStub(&stub); | 310 __ CallStub(&stub); |
| 304 | 311 |
| 305 SetVar(rest_param, r3, r4, r5); | 312 SetVar(rest_param, r3, r4, r5); |
| 306 } | 313 } |
| 307 | 314 |
| 308 Variable* arguments = scope()->arguments(); | 315 Variable* arguments = scope()->arguments(); |
| 309 if (arguments != NULL) { | 316 if (arguments != NULL) { |
| 310 // Function uses arguments object. | 317 // Function uses arguments object. |
| 311 Comment cmnt(masm_, "[ Allocate arguments object"); | 318 Comment cmnt(masm_, "[ Allocate arguments object"); |
| 312 if (!function_in_register) { | 319 if (!function_in_register_r4) { |
| 313 // Load this again, if it's used by the local context below. | 320 // Load this again, if it's used by the local context below. |
| 314 __ LoadP(r6, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); | 321 __ LoadP(r6, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); |
| 315 } else { | 322 } else { |
| 316 __ mr(r6, r4); | 323 __ mr(r6, r4); |
| 317 } | 324 } |
| 318 // Receiver is just before the parameters on the caller's stack. | 325 // Receiver is just before the parameters on the caller's stack. |
| 319 int num_parameters = info->scope()->num_parameters(); | 326 int num_parameters = info->scope()->num_parameters(); |
| 320 int offset = num_parameters * kPointerSize; | 327 int offset = num_parameters * kPointerSize; |
| 321 __ addi(r5, fp, Operand(StandardFrameConstants::kCallerSPOffset + offset)); | 328 __ addi(r5, fp, Operand(StandardFrameConstants::kCallerSPOffset + offset)); |
| 322 __ LoadSmiLiteral(r4, Smi::FromInt(num_parameters)); | 329 __ LoadSmiLiteral(r4, Smi::FromInt(num_parameters)); |
| (...skipping 4992 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 5315 return ON_STACK_REPLACEMENT; | 5322 return ON_STACK_REPLACEMENT; |
| 5316 } | 5323 } |
| 5317 | 5324 |
| 5318 DCHECK(interrupt_address == | 5325 DCHECK(interrupt_address == |
| 5319 isolate->builtins()->OsrAfterStackCheck()->entry()); | 5326 isolate->builtins()->OsrAfterStackCheck()->entry()); |
| 5320 return OSR_AFTER_STACK_CHECK; | 5327 return OSR_AFTER_STACK_CHECK; |
| 5321 } | 5328 } |
| 5322 } // namespace internal | 5329 } // namespace internal |
| 5323 } // namespace v8 | 5330 } // namespace v8 |
| 5324 #endif // V8_TARGET_ARCH_PPC | 5331 #endif // V8_TARGET_ARCH_PPC |
| OLD | NEW |