| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
| 4 // met: | 4 // met: |
| 5 // | 5 // |
| 6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
| 7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
| 8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
| 9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
| 10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
| (...skipping 115 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 126 | 126 |
| 127 ProfileEntryHookStub::MaybeCallEntryHook(masm_); | 127 ProfileEntryHookStub::MaybeCallEntryHook(masm_); |
| 128 | 128 |
| 129 #ifdef DEBUG | 129 #ifdef DEBUG |
| 130 if (strlen(FLAG_stop_at) > 0 && | 130 if (strlen(FLAG_stop_at) > 0 && |
| 131 info->function()->name()->IsUtf8EqualTo(CStrVector(FLAG_stop_at))) { | 131 info->function()->name()->IsUtf8EqualTo(CStrVector(FLAG_stop_at))) { |
| 132 __ int3(); | 132 __ int3(); |
| 133 } | 133 } |
| 134 #endif | 134 #endif |
| 135 | 135 |
| 136 // Strict mode functions and builtins need to replace the receiver | 136 // Classic mode functions and builtins need to replace the receiver with the |
| 137 // with undefined when called as functions (without an explicit | 137 // global proxy when called as functions (without an explicit receiver |
| 138 // receiver object). ecx is zero for method calls and non-zero for | 138 // object). |
| 139 // function calls. | 139 if (info->is_classic_mode() && !info->is_native()) { |
| 140 if (!info->is_classic_mode() || info->is_native()) { | |
| 141 Label ok; | 140 Label ok; |
| 142 __ test(ecx, ecx); | 141 __ test(ecx, ecx); |
| 143 __ j(zero, &ok, Label::kNear); | 142 __ j(zero, &ok, Label::kNear); |
| 143 |
| 144 // +1 for return address. | 144 // +1 for return address. |
| 145 int receiver_offset = (info->scope()->num_parameters() + 1) * kPointerSize; | 145 int receiver_offset = (info->scope()->num_parameters() + 1) * kPointerSize; |
| 146 __ mov(ecx, Operand(esp, receiver_offset)); | 146 __ mov(ecx, Operand(esp, receiver_offset)); |
| 147 __ JumpIfSmi(ecx, &ok); | 147 |
| 148 __ CmpObjectType(ecx, JS_GLOBAL_PROXY_TYPE, ecx); | 148 __ cmp(ecx, isolate()->factory()->undefined_value()); |
| 149 __ j(not_equal, &ok, Label::kNear); | 149 __ j(not_equal, &ok, Label::kNear); |
| 150 __ mov(Operand(esp, receiver_offset), | 150 |
| 151 Immediate(isolate()->factory()->undefined_value())); | 151 __ mov(ecx, GlobalObjectOperand()); |
| 152 __ mov(ecx, FieldOperand(ecx, GlobalObject::kGlobalReceiverOffset)); |
| 153 |
| 154 __ mov(Operand(esp, receiver_offset), ecx); |
| 155 |
| 152 __ bind(&ok); | 156 __ bind(&ok); |
| 153 } | 157 } |
| 154 | 158 |
| 155 // Open a frame scope to indicate that there is a frame on the stack. The | 159 // Open a frame scope to indicate that there is a frame on the stack. The |
| 156 // MANUAL indicates that the scope shouldn't actually generate code to set up | 160 // MANUAL indicates that the scope shouldn't actually generate code to set up |
| 157 // the frame (that is done below). | 161 // the frame (that is done below). |
| 158 FrameScope frame_scope(masm_, StackFrame::MANUAL); | 162 FrameScope frame_scope(masm_, StackFrame::MANUAL); |
| 159 | 163 |
| 160 info->set_prologue_offset(masm_->pc_offset()); | 164 info->set_prologue_offset(masm_->pc_offset()); |
| 161 __ Prologue(BUILD_FUNCTION_FRAME); | 165 __ Prologue(BUILD_FUNCTION_FRAME); |
| (...skipping 2519 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2681 __ push(Operand(esp, (arg_count + 1) * kPointerSize)); | 2685 __ push(Operand(esp, (arg_count + 1) * kPointerSize)); |
| 2682 EmitResolvePossiblyDirectEval(arg_count); | 2686 EmitResolvePossiblyDirectEval(arg_count); |
| 2683 | 2687 |
| 2684 // The runtime call returns a pair of values in eax (function) and | 2688 // The runtime call returns a pair of values in eax (function) and |
| 2685 // edx (receiver). Touch up the stack with the right values. | 2689 // edx (receiver). Touch up the stack with the right values. |
| 2686 __ mov(Operand(esp, (arg_count + 0) * kPointerSize), edx); | 2690 __ mov(Operand(esp, (arg_count + 0) * kPointerSize), edx); |
| 2687 __ mov(Operand(esp, (arg_count + 1) * kPointerSize), eax); | 2691 __ mov(Operand(esp, (arg_count + 1) * kPointerSize), eax); |
| 2688 } | 2692 } |
| 2689 // Record source position for debugger. | 2693 // Record source position for debugger. |
| 2690 SetSourcePosition(expr->position()); | 2694 SetSourcePosition(expr->position()); |
| 2691 CallFunctionStub stub(arg_count, RECEIVER_MIGHT_BE_IMPLICIT); | 2695 CallFunctionStub stub(arg_count, NO_CALL_FUNCTION_FLAGS); |
| 2692 __ mov(edi, Operand(esp, (arg_count + 1) * kPointerSize)); | 2696 __ mov(edi, Operand(esp, (arg_count + 1) * kPointerSize)); |
| 2693 __ CallStub(&stub); | 2697 __ CallStub(&stub); |
| 2694 RecordJSReturnSite(expr); | 2698 RecordJSReturnSite(expr); |
| 2695 // Restore context register. | 2699 // Restore context register. |
| 2696 __ mov(esi, Operand(ebp, StandardFrameConstants::kContextOffset)); | 2700 __ mov(esi, Operand(ebp, StandardFrameConstants::kContextOffset)); |
| 2697 context()->DropAndPlug(1, eax); | 2701 context()->DropAndPlug(1, eax); |
| 2698 | 2702 |
| 2699 } else if (proxy != NULL && proxy->var()->IsUnallocated()) { | 2703 } else if (proxy != NULL && proxy->var()->IsUnallocated()) { |
| 2700 // Push global object as receiver for the call IC. | 2704 // Push global object as receiver for the call IC. |
| 2701 __ push(GlobalObjectOperand()); | 2705 __ push(GlobalObjectOperand()); |
| (...skipping 18 matching lines...) Expand all Loading... |
| 2720 // If fast case code has been generated, emit code to push the function | 2724 // If fast case code has been generated, emit code to push the function |
| 2721 // and receiver and have the slow path jump around this code. | 2725 // and receiver and have the slow path jump around this code. |
| 2722 if (done.is_linked()) { | 2726 if (done.is_linked()) { |
| 2723 Label call; | 2727 Label call; |
| 2724 __ jmp(&call, Label::kNear); | 2728 __ jmp(&call, Label::kNear); |
| 2725 __ bind(&done); | 2729 __ bind(&done); |
| 2726 // Push function. | 2730 // Push function. |
| 2727 __ push(eax); | 2731 __ push(eax); |
| 2728 // The receiver is implicitly the global receiver. Indicate this by | 2732 // The receiver is implicitly the global receiver. Indicate this by |
| 2729 // passing the hole to the call function stub. | 2733 // passing the hole to the call function stub. |
| 2730 __ push(Immediate(isolate()->factory()->the_hole_value())); | 2734 __ push(Immediate(isolate()->factory()->undefined_value())); |
| 2731 __ bind(&call); | 2735 __ bind(&call); |
| 2732 } | 2736 } |
| 2733 | 2737 |
| 2734 // The receiver is either the global receiver or an object found by | 2738 // The receiver is either the global receiver or an object found by |
| 2735 // LoadContextSlot. That object could be the hole if the receiver is | 2739 // LoadContextSlot. |
| 2736 // implicitly the global object. | 2740 EmitCallWithStub(expr, NO_CALL_FUNCTION_FLAGS); |
| 2737 EmitCallWithStub(expr, RECEIVER_MIGHT_BE_IMPLICIT); | |
| 2738 | 2741 |
| 2739 } else if (property != NULL) { | 2742 } else if (property != NULL) { |
| 2740 { PreservePositionScope scope(masm()->positions_recorder()); | 2743 { PreservePositionScope scope(masm()->positions_recorder()); |
| 2741 VisitForStackValue(property->obj()); | 2744 VisitForStackValue(property->obj()); |
| 2742 } | 2745 } |
| 2743 if (property->key()->IsPropertyName()) { | 2746 if (property->key()->IsPropertyName()) { |
| 2744 EmitCallWithIC(expr, | 2747 EmitCallWithIC(expr, |
| 2745 property->key()->AsLiteral()->value(), | 2748 property->key()->AsLiteral()->value(), |
| 2746 NOT_CONTEXTUAL); | 2749 NOT_CONTEXTUAL); |
| 2747 } else { | 2750 } else { |
| 2748 EmitKeyedCallWithIC(expr, property->key()); | 2751 EmitKeyedCallWithIC(expr, property->key()); |
| 2749 } | 2752 } |
| 2750 | 2753 |
| 2751 } else { | 2754 } else { |
| 2752 // Call to an arbitrary expression not handled specially above. | 2755 // Call to an arbitrary expression not handled specially above. |
| 2753 { PreservePositionScope scope(masm()->positions_recorder()); | 2756 { PreservePositionScope scope(masm()->positions_recorder()); |
| 2754 VisitForStackValue(callee); | 2757 VisitForStackValue(callee); |
| 2755 } | 2758 } |
| 2756 // Push the hole as receiver. | 2759 __ push(Immediate(isolate()->factory()->undefined_value())); |
| 2757 // It will be correctly replaced in the call stub. | |
| 2758 __ push(Immediate(isolate()->factory()->the_hole_value())); | |
| 2759 // Emit function call. | 2760 // Emit function call. |
| 2760 EmitCallWithStub(expr, RECEIVER_IS_IMPLICIT); | 2761 EmitCallWithStub(expr, NO_CALL_FUNCTION_FLAGS); |
| 2761 } | 2762 } |
| 2762 | 2763 |
| 2763 #ifdef DEBUG | 2764 #ifdef DEBUG |
| 2764 // RecordJSReturnSite should have been called. | 2765 // RecordJSReturnSite should have been called. |
| 2765 ASSERT(expr->return_is_recorded_); | 2766 ASSERT(expr->return_is_recorded_); |
| 2766 #endif | 2767 #endif |
| 2767 } | 2768 } |
| 2768 | 2769 |
| 2769 | 2770 |
| 2770 void FullCodeGenerator::VisitCallNew(CallNew* expr) { | 2771 void FullCodeGenerator::VisitCallNew(CallNew* expr) { |
| (...skipping 915 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 3686 Label runtime, done; | 3687 Label runtime, done; |
| 3687 // Check for non-function argument (including proxy). | 3688 // Check for non-function argument (including proxy). |
| 3688 __ JumpIfSmi(eax, &runtime); | 3689 __ JumpIfSmi(eax, &runtime); |
| 3689 __ CmpObjectType(eax, JS_FUNCTION_TYPE, ebx); | 3690 __ CmpObjectType(eax, JS_FUNCTION_TYPE, ebx); |
| 3690 __ j(not_equal, &runtime); | 3691 __ j(not_equal, &runtime); |
| 3691 | 3692 |
| 3692 // InvokeFunction requires the function in edi. Move it in there. | 3693 // InvokeFunction requires the function in edi. Move it in there. |
| 3693 __ mov(edi, result_register()); | 3694 __ mov(edi, result_register()); |
| 3694 ParameterCount count(arg_count); | 3695 ParameterCount count(arg_count); |
| 3695 __ InvokeFunction(edi, count, CALL_FUNCTION, | 3696 __ InvokeFunction(edi, count, CALL_FUNCTION, |
| 3696 NullCallWrapper(), CALL_AS_METHOD); | 3697 NullCallWrapper(), CALL_AS_FUNCTION); |
| 3697 __ mov(esi, Operand(ebp, StandardFrameConstants::kContextOffset)); | 3698 __ mov(esi, Operand(ebp, StandardFrameConstants::kContextOffset)); |
| 3698 __ jmp(&done); | 3699 __ jmp(&done); |
| 3699 | 3700 |
| 3700 __ bind(&runtime); | 3701 __ bind(&runtime); |
| 3701 __ push(eax); | 3702 __ push(eax); |
| 3702 __ CallRuntime(Runtime::kCall, args->length()); | 3703 __ CallRuntime(Runtime::kCall, args->length()); |
| 3703 __ bind(&done); | 3704 __ bind(&done); |
| 3704 | 3705 |
| 3705 context()->Plug(eax); | 3706 context()->Plug(eax); |
| 3706 } | 3707 } |
| (...skipping 1182 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4889 | 4890 |
| 4890 ASSERT_EQ(isolate->builtins()->OsrAfterStackCheck()->entry(), | 4891 ASSERT_EQ(isolate->builtins()->OsrAfterStackCheck()->entry(), |
| 4891 Assembler::target_address_at(call_target_address)); | 4892 Assembler::target_address_at(call_target_address)); |
| 4892 return OSR_AFTER_STACK_CHECK; | 4893 return OSR_AFTER_STACK_CHECK; |
| 4893 } | 4894 } |
| 4894 | 4895 |
| 4895 | 4896 |
| 4896 } } // namespace v8::internal | 4897 } } // namespace v8::internal |
| 4897 | 4898 |
| 4898 #endif // V8_TARGET_ARCH_IA32 | 4899 #endif // V8_TARGET_ARCH_IA32 |
| OLD | NEW |