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 #if V8_TARGET_ARCH_IA32 | 5 #if V8_TARGET_ARCH_IA32 |
6 | 6 |
7 #include "src/code-factory.h" | 7 #include "src/code-factory.h" |
8 #include "src/codegen.h" | 8 #include "src/codegen.h" |
9 #include "src/deoptimizer.h" | 9 #include "src/deoptimizer.h" |
10 #include "src/full-codegen/full-codegen.h" | 10 #include "src/full-codegen/full-codegen.h" |
(...skipping 100 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
111 CallRuntimePassFunction(masm, Runtime::kTryInstallOptimizedCode); | 111 CallRuntimePassFunction(masm, Runtime::kTryInstallOptimizedCode); |
112 GenerateTailCallToReturnedCode(masm); | 112 GenerateTailCallToReturnedCode(masm); |
113 | 113 |
114 __ bind(&ok); | 114 __ bind(&ok); |
115 GenerateTailCallToSharedCode(masm); | 115 GenerateTailCallToSharedCode(masm); |
116 } | 116 } |
117 | 117 |
118 | 118 |
119 static void Generate_JSConstructStubHelper(MacroAssembler* masm, | 119 static void Generate_JSConstructStubHelper(MacroAssembler* masm, |
120 bool is_api_function, | 120 bool is_api_function, |
121 bool create_implicit_receiver) { | 121 bool create_implicit_receiver, |
| 122 bool check_derived_construct) { |
122 // ----------- S t a t e ------------- | 123 // ----------- S t a t e ------------- |
123 // -- eax: number of arguments | 124 // -- eax: number of arguments |
124 // -- edi: constructor function | 125 // -- edi: constructor function |
125 // -- ebx: allocation site or undefined | 126 // -- ebx: allocation site or undefined |
126 // -- edx: new target | 127 // -- edx: new target |
127 // ----------------------------------- | 128 // ----------------------------------- |
128 | 129 |
129 // Enter a construct frame. | 130 // Enter a construct frame. |
130 { | 131 { |
131 FrameScope scope(masm, StackFrame::CONSTRUCT); | 132 FrameScope scope(masm, StackFrame::CONSTRUCT); |
(...skipping 192 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
324 // Store offset of return address for deoptimizer. | 325 // Store offset of return address for deoptimizer. |
325 if (create_implicit_receiver && !is_api_function) { | 326 if (create_implicit_receiver && !is_api_function) { |
326 masm->isolate()->heap()->SetConstructStubDeoptPCOffset(masm->pc_offset()); | 327 masm->isolate()->heap()->SetConstructStubDeoptPCOffset(masm->pc_offset()); |
327 } | 328 } |
328 | 329 |
329 // Restore context from the frame. | 330 // Restore context from the frame. |
330 __ mov(esi, Operand(ebp, StandardFrameConstants::kContextOffset)); | 331 __ mov(esi, Operand(ebp, StandardFrameConstants::kContextOffset)); |
331 | 332 |
332 if (create_implicit_receiver) { | 333 if (create_implicit_receiver) { |
333 // If the result is an object (in the ECMA sense), we should get rid | 334 // If the result is an object (in the ECMA sense), we should get rid |
334 // of the receiver and use the result; see ECMA-262 section 13.2.2-7 | 335 // of the receiver and use the result. |
335 // on page 74. | |
336 Label use_receiver, exit; | 336 Label use_receiver, exit; |
337 | 337 |
338 // If the result is a smi, it is *not* an object in the ECMA sense. | 338 // If the result is a smi, it is *not* an object in the ECMA sense. |
339 __ JumpIfSmi(eax, &use_receiver); | 339 __ JumpIfSmi(eax, &use_receiver); |
340 | 340 |
341 // If the type of the result (stored in its map) is less than | 341 // If the type of the result (stored in its map) is less than |
342 // FIRST_JS_RECEIVER_TYPE, it is not an object in the ECMA sense. | 342 // FIRST_JS_RECEIVER_TYPE, it is not an object in the ECMA sense. |
343 __ CmpObjectType(eax, FIRST_JS_RECEIVER_TYPE, ecx); | 343 __ CmpObjectType(eax, FIRST_JS_RECEIVER_TYPE, ecx); |
344 __ j(above_equal, &exit); | 344 __ j(above_equal, &exit); |
345 | 345 |
346 // Throw away the result of the constructor invocation and use the | 346 // Throw away the result of the constructor invocation and use the |
347 // on-stack receiver as the result. | 347 // on-stack receiver as the result. |
348 __ bind(&use_receiver); | 348 __ bind(&use_receiver); |
349 __ mov(eax, Operand(esp, 0)); | 349 __ mov(eax, Operand(esp, 0)); |
350 | 350 |
351 // Restore the arguments count and leave the construct frame. The | 351 // Restore the arguments count and leave the construct frame. The |
352 // arguments count is stored below the receiver. | 352 // arguments count is stored below the receiver. |
353 __ bind(&exit); | 353 __ bind(&exit); |
354 __ mov(ebx, Operand(esp, 1 * kPointerSize)); | 354 __ mov(ebx, Operand(esp, 1 * kPointerSize)); |
355 } else { | 355 } else { |
356 __ mov(ebx, Operand(esp, 0)); | 356 __ mov(ebx, Operand(esp, 0)); |
357 } | 357 } |
358 | 358 |
359 // Leave construct frame. | 359 // Leave construct frame. |
360 } | 360 } |
361 | 361 |
| 362 // ES6 9.2.2. Step 13+ |
| 363 // Check that the result is not a Smi, indicating that the constructor result |
| 364 // from a derived class is neither undefined nor an Object. |
| 365 if (check_derived_construct) { |
| 366 Label dont_throw; |
| 367 __ JumpIfNotSmi(eax, &dont_throw); |
| 368 { |
| 369 FrameScope scope(masm, StackFrame::INTERNAL); |
| 370 __ CallRuntime(Runtime::kThrowDerivedConstructorReturnedNonObject); |
| 371 } |
| 372 __ bind(&dont_throw); |
| 373 } |
| 374 |
362 // Remove caller arguments from the stack and return. | 375 // Remove caller arguments from the stack and return. |
363 STATIC_ASSERT(kSmiTagSize == 1 && kSmiTag == 0); | 376 STATIC_ASSERT(kSmiTagSize == 1 && kSmiTag == 0); |
364 __ pop(ecx); | 377 __ pop(ecx); |
365 __ lea(esp, Operand(esp, ebx, times_2, 1 * kPointerSize)); // 1 ~ receiver | 378 __ lea(esp, Operand(esp, ebx, times_2, 1 * kPointerSize)); // 1 ~ receiver |
366 __ push(ecx); | 379 __ push(ecx); |
367 if (create_implicit_receiver) { | 380 if (create_implicit_receiver) { |
368 __ IncrementCounter(masm->isolate()->counters()->constructed_objects(), 1); | 381 __ IncrementCounter(masm->isolate()->counters()->constructed_objects(), 1); |
369 } | 382 } |
370 __ ret(0); | 383 __ ret(0); |
371 } | 384 } |
372 | 385 |
373 | 386 |
374 void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { | 387 void Builtins::Generate_JSConstructStubGeneric(MacroAssembler* masm) { |
375 Generate_JSConstructStubHelper(masm, false, true); | 388 Generate_JSConstructStubHelper(masm, false, true, false); |
376 } | 389 } |
377 | 390 |
378 | 391 |
379 void Builtins::Generate_JSConstructStubApi(MacroAssembler* masm) { | 392 void Builtins::Generate_JSConstructStubApi(MacroAssembler* masm) { |
380 Generate_JSConstructStubHelper(masm, true, true); | 393 Generate_JSConstructStubHelper(masm, true, true, false); |
381 } | 394 } |
382 | 395 |
383 | 396 |
384 void Builtins::Generate_JSBuiltinsConstructStub(MacroAssembler* masm) { | 397 void Builtins::Generate_JSBuiltinsConstructStub(MacroAssembler* masm) { |
385 Generate_JSConstructStubHelper(masm, false, false); | 398 Generate_JSConstructStubHelper(masm, false, false, false); |
386 } | 399 } |
387 | 400 |
388 | 401 |
| 402 void Builtins::Generate_JSBuiltinsConstructStubForDerived( |
| 403 MacroAssembler* masm) { |
| 404 Generate_JSConstructStubHelper(masm, false, false, true); |
| 405 } |
| 406 |
| 407 |
389 void Builtins::Generate_ConstructedNonConstructable(MacroAssembler* masm) { | 408 void Builtins::Generate_ConstructedNonConstructable(MacroAssembler* masm) { |
390 FrameScope scope(masm, StackFrame::INTERNAL); | 409 FrameScope scope(masm, StackFrame::INTERNAL); |
391 __ push(edi); | 410 __ push(edi); |
392 __ CallRuntime(Runtime::kThrowConstructedNonConstructable); | 411 __ CallRuntime(Runtime::kThrowConstructedNonConstructable); |
393 } | 412 } |
394 | 413 |
395 | 414 |
396 enum IsTagged { kEaxIsSmiTagged, kEaxIsUntaggedInt }; | 415 enum IsTagged { kEaxIsSmiTagged, kEaxIsUntaggedInt }; |
397 | 416 |
398 | 417 |
(...skipping 2100 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2499 | 2518 |
2500 __ bind(&ok); | 2519 __ bind(&ok); |
2501 __ ret(0); | 2520 __ ret(0); |
2502 } | 2521 } |
2503 | 2522 |
2504 #undef __ | 2523 #undef __ |
2505 } // namespace internal | 2524 } // namespace internal |
2506 } // namespace v8 | 2525 } // namespace v8 |
2507 | 2526 |
2508 #endif // V8_TARGET_ARCH_IA32 | 2527 #endif // V8_TARGET_ARCH_IA32 |
OLD | NEW |