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 "v8.h" | 5 #include "v8.h" |
6 | 6 |
7 #if V8_TARGET_ARCH_MIPS | 7 #if V8_TARGET_ARCH_MIPS |
8 | 8 |
9 #include "bootstrapper.h" | 9 #include "bootstrapper.h" |
10 #include "code-stubs.h" | 10 #include "code-stubs.h" |
(...skipping 3076 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3087 { FrameScope frame_scope(masm, StackFrame::INTERNAL); | 3087 { FrameScope frame_scope(masm, StackFrame::INTERNAL); |
3088 __ Push(a1, a3); | 3088 __ Push(a1, a3); |
3089 __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION); | 3089 __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION); |
3090 __ pop(a1); | 3090 __ pop(a1); |
3091 } | 3091 } |
3092 __ Branch(USE_DELAY_SLOT, cont); | 3092 __ Branch(USE_DELAY_SLOT, cont); |
3093 __ sw(v0, MemOperand(sp, argc * kPointerSize)); | 3093 __ sw(v0, MemOperand(sp, argc * kPointerSize)); |
3094 } | 3094 } |
3095 | 3095 |
3096 | 3096 |
3097 static void CallFunctionNoFeedback(MacroAssembler* masm, | 3097 void CallFunctionStub::Generate(MacroAssembler* masm) { |
3098 int argc, bool needs_checks, | |
3099 bool call_as_method) { | |
3100 // a1 : the function to call | 3098 // a1 : the function to call |
3101 Label slow, non_function, wrap, cont; | 3099 Label slow, non_function, wrap, cont; |
3102 | 3100 |
3103 if (needs_checks) { | 3101 if (NeedsChecks()) { |
3104 // Check that the function is really a JavaScript function. | 3102 // Check that the function is really a JavaScript function. |
3105 // a1: pushed function (to be verified) | 3103 // a1: pushed function (to be verified) |
3106 __ JumpIfSmi(a1, &non_function); | 3104 __ JumpIfSmi(a1, &non_function); |
3107 | 3105 |
3108 // Goto slow case if we do not have a function. | 3106 // Goto slow case if we do not have a function. |
3109 __ GetObjectType(a1, t0, t0); | 3107 __ GetObjectType(a1, t0, t0); |
3110 __ Branch(&slow, ne, t0, Operand(JS_FUNCTION_TYPE)); | 3108 __ Branch(&slow, ne, t0, Operand(JS_FUNCTION_TYPE)); |
3111 } | 3109 } |
3112 | 3110 |
3113 // Fast-case: Invoke the function now. | 3111 // Fast-case: Invoke the function now. |
3114 // a1: pushed function | 3112 // a1: pushed function |
| 3113 int argc = argc_; |
3115 ParameterCount actual(argc); | 3114 ParameterCount actual(argc); |
3116 | 3115 |
3117 if (call_as_method) { | 3116 if (CallAsMethod()) { |
3118 if (needs_checks) { | 3117 if (NeedsChecks()) { |
3119 EmitContinueIfStrictOrNative(masm, &cont); | 3118 EmitContinueIfStrictOrNative(masm, &cont); |
3120 } | 3119 } |
3121 | 3120 |
3122 // Compute the receiver in sloppy mode. | 3121 // Compute the receiver in sloppy mode. |
3123 __ lw(a3, MemOperand(sp, argc * kPointerSize)); | 3122 __ lw(a3, MemOperand(sp, argc * kPointerSize)); |
3124 | 3123 |
3125 if (needs_checks) { | 3124 if (NeedsChecks()) { |
3126 __ JumpIfSmi(a3, &wrap); | 3125 __ JumpIfSmi(a3, &wrap); |
3127 __ GetObjectType(a3, t0, t0); | 3126 __ GetObjectType(a3, t0, t0); |
3128 __ Branch(&wrap, lt, t0, Operand(FIRST_SPEC_OBJECT_TYPE)); | 3127 __ Branch(&wrap, lt, t0, Operand(FIRST_SPEC_OBJECT_TYPE)); |
3129 } else { | 3128 } else { |
3130 __ jmp(&wrap); | 3129 __ jmp(&wrap); |
3131 } | 3130 } |
3132 | 3131 |
3133 __ bind(&cont); | 3132 __ bind(&cont); |
3134 } | 3133 } |
3135 | 3134 |
3136 __ InvokeFunction(a1, actual, JUMP_FUNCTION, NullCallWrapper()); | 3135 __ InvokeFunction(a1, actual, JUMP_FUNCTION, NullCallWrapper()); |
3137 | 3136 |
3138 if (needs_checks) { | 3137 if (NeedsChecks()) { |
3139 // Slow-case: Non-function called. | 3138 // Slow-case: Non-function called. |
3140 __ bind(&slow); | 3139 __ bind(&slow); |
3141 EmitSlowCase(masm, argc, &non_function); | 3140 EmitSlowCase(masm, argc, &non_function); |
3142 } | 3141 } |
3143 | 3142 |
3144 if (call_as_method) { | 3143 if (CallAsMethod()) { |
3145 __ bind(&wrap); | 3144 __ bind(&wrap); |
3146 // Wrap the receiver and patch it back onto the stack. | 3145 // Wrap the receiver and patch it back onto the stack. |
3147 EmitWrapCase(masm, argc, &cont); | 3146 EmitWrapCase(masm, argc, &cont); |
3148 } | 3147 } |
3149 } | 3148 } |
3150 | 3149 |
3151 | 3150 |
3152 void CallFunctionStub::Generate(MacroAssembler* masm) { | |
3153 CallFunctionNoFeedback(masm, argc_, NeedsChecks(), CallAsMethod()); | |
3154 } | |
3155 | |
3156 | |
3157 void CallConstructStub::Generate(MacroAssembler* masm) { | 3151 void CallConstructStub::Generate(MacroAssembler* masm) { |
3158 // a0 : number of arguments | 3152 // a0 : number of arguments |
3159 // a1 : the function to call | 3153 // a1 : the function to call |
3160 // a2 : feedback vector | 3154 // a2 : feedback vector |
3161 // a3 : (only if a2 is not undefined) slot in feedback vector (Smi) | 3155 // a3 : (only if a2 is not undefined) slot in feedback vector (Smi) |
3162 Label slow, non_function_call; | 3156 Label slow, non_function_call; |
3163 | 3157 |
3164 // Check that the function is not a smi. | 3158 // Check that the function is not a smi. |
3165 __ JumpIfSmi(a1, &non_function_call); | 3159 __ JumpIfSmi(a1, &non_function_call); |
3166 // Check that the function is a JSFunction. | 3160 // Check that the function is a JSFunction. |
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3206 __ bind(&slow); | 3200 __ bind(&slow); |
3207 __ Branch(&non_function_call, ne, t0, Operand(JS_FUNCTION_PROXY_TYPE)); | 3201 __ Branch(&non_function_call, ne, t0, Operand(JS_FUNCTION_PROXY_TYPE)); |
3208 __ GetBuiltinFunction(a1, Builtins::CALL_FUNCTION_PROXY_AS_CONSTRUCTOR); | 3202 __ GetBuiltinFunction(a1, Builtins::CALL_FUNCTION_PROXY_AS_CONSTRUCTOR); |
3209 __ jmp(&do_call); | 3203 __ jmp(&do_call); |
3210 | 3204 |
3211 __ bind(&non_function_call); | 3205 __ bind(&non_function_call); |
3212 __ GetBuiltinFunction(a1, Builtins::CALL_NON_FUNCTION_AS_CONSTRUCTOR); | 3206 __ GetBuiltinFunction(a1, Builtins::CALL_NON_FUNCTION_AS_CONSTRUCTOR); |
3213 __ bind(&do_call); | 3207 __ bind(&do_call); |
3214 // Set expected number of arguments to zero (not changing r0). | 3208 // Set expected number of arguments to zero (not changing r0). |
3215 __ li(a2, Operand(0, RelocInfo::NONE32)); | 3209 __ li(a2, Operand(0, RelocInfo::NONE32)); |
3216 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), | 3210 __ Jump(isolate()->builtins()->ArgumentsAdaptorTrampoline(), |
3217 RelocInfo::CODE_TARGET); | 3211 RelocInfo::CODE_TARGET); |
3218 } | 3212 } |
3219 | 3213 |
3220 | 3214 |
3221 static void EmitLoadTypeFeedbackVector(MacroAssembler* masm, Register vector) { | 3215 static void EmitLoadTypeFeedbackVector(MacroAssembler* masm, Register vector) { |
3222 __ lw(vector, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); | 3216 __ lw(vector, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset)); |
3223 __ lw(vector, FieldMemOperand(vector, | 3217 __ lw(vector, FieldMemOperand(vector, |
3224 JSFunction::kSharedFunctionInfoOffset)); | 3218 JSFunction::kSharedFunctionInfoOffset)); |
3225 __ lw(vector, FieldMemOperand(vector, | 3219 __ lw(vector, FieldMemOperand(vector, |
3226 SharedFunctionInfo::kFeedbackVectorOffset)); | 3220 SharedFunctionInfo::kFeedbackVectorOffset)); |
3227 } | 3221 } |
3228 | 3222 |
3229 | 3223 |
3230 void CallICStub::Generate_MonomorphicArray(MacroAssembler* masm, Label* miss) { | |
3231 // a1 - function | |
3232 // a2 - feedback vector | |
3233 // a3 - slot id | |
3234 __ LoadGlobalFunction(Context::ARRAY_FUNCTION_INDEX, at); | |
3235 __ Branch(miss, ne, a1, Operand(at)); | |
3236 | |
3237 __ li(a0, Operand(arg_count())); | |
3238 __ sll(at, a3, kPointerSizeLog2 - kSmiTagSize); | |
3239 __ Addu(at, a2, Operand(at)); | |
3240 __ lw(a2, FieldMemOperand(at, FixedArray::kHeaderSize)); | |
3241 // Verify that a2 contains an AllocationSite | |
3242 __ AssertUndefinedOrAllocationSite(a2, at); | |
3243 ArrayConstructorStub stub(masm->isolate(), arg_count()); | |
3244 __ TailCallStub(&stub); | |
3245 } | |
3246 | |
3247 | |
3248 void CallICStub::Generate_CustomFeedbackCall(MacroAssembler* masm) { | |
3249 // a1 - function | |
3250 // a2 - feedback vector | |
3251 // a3 - slot id | |
3252 Label miss; | |
3253 | |
3254 if (state_.stub_type() == CallIC::MONOMORPHIC_ARRAY) { | |
3255 Generate_MonomorphicArray(masm, &miss); | |
3256 } else { | |
3257 // So far there is only one customer for our custom feedback scheme. | |
3258 UNREACHABLE(); | |
3259 } | |
3260 | |
3261 __ bind(&miss); | |
3262 GenerateMiss(masm); | |
3263 | |
3264 // The slow case, we need this no matter what to complete a call after a miss. | |
3265 CallFunctionNoFeedback(masm, | |
3266 arg_count(), | |
3267 true, | |
3268 CallAsMethod()); | |
3269 | |
3270 // Unreachable. | |
3271 __ stop("Unexpected code address"); | |
3272 } | |
3273 | |
3274 | |
3275 void CallICStub::Generate(MacroAssembler* masm) { | 3224 void CallICStub::Generate(MacroAssembler* masm) { |
3276 // r1 - function | 3225 // r1 - function |
3277 // r3 - slot id (Smi) | 3226 // r3 - slot id (Smi) |
3278 Label extra_checks_or_miss, slow_start; | 3227 Label extra_checks_or_miss, slow_start; |
3279 Label slow, non_function, wrap, cont; | 3228 Label slow, non_function, wrap, cont; |
3280 Label have_js_function; | 3229 Label have_js_function; |
3281 int argc = state_.arg_count(); | 3230 int argc = state_.arg_count(); |
3282 ParameterCount actual(argc); | 3231 ParameterCount actual(argc); |
3283 | 3232 |
3284 EmitLoadTypeFeedbackVector(masm, a2); | 3233 EmitLoadTypeFeedbackVector(masm, a2); |
3285 | 3234 |
3286 if (state_.stub_type() != CallIC::DEFAULT) { | |
3287 Generate_CustomFeedbackCall(masm); | |
3288 return; | |
3289 } | |
3290 | |
3291 // The checks. First, does r1 match the recorded monomorphic target? | 3235 // The checks. First, does r1 match the recorded monomorphic target? |
3292 __ sll(t0, a3, kPointerSizeLog2 - kSmiTagSize); | 3236 __ sll(t0, a3, kPointerSizeLog2 - kSmiTagSize); |
3293 __ Addu(t0, a2, Operand(t0)); | 3237 __ Addu(t0, a2, Operand(t0)); |
3294 __ lw(t0, FieldMemOperand(t0, FixedArray::kHeaderSize)); | 3238 __ lw(t0, FieldMemOperand(t0, FixedArray::kHeaderSize)); |
3295 __ Branch(&extra_checks_or_miss, ne, a1, Operand(t0)); | 3239 __ Branch(&extra_checks_or_miss, ne, a1, Operand(t0)); |
3296 | 3240 |
3297 __ bind(&have_js_function); | 3241 __ bind(&have_js_function); |
3298 if (state_.CallAsMethod()) { | 3242 if (state_.CallAsMethod()) { |
3299 EmitContinueIfStrictOrNative(masm, &cont); | 3243 EmitContinueIfStrictOrNative(masm, &cont); |
3300 // Compute the receiver in sloppy mode. | 3244 // Compute the receiver in sloppy mode. |
(...skipping 2200 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
5501 MemOperand(fp, 6 * kPointerSize), | 5445 MemOperand(fp, 6 * kPointerSize), |
5502 NULL); | 5446 NULL); |
5503 } | 5447 } |
5504 | 5448 |
5505 | 5449 |
5506 #undef __ | 5450 #undef __ |
5507 | 5451 |
5508 } } // namespace v8::internal | 5452 } } // namespace v8::internal |
5509 | 5453 |
5510 #endif // V8_TARGET_ARCH_MIPS | 5454 #endif // V8_TARGET_ARCH_MIPS |
OLD | NEW |