| OLD | NEW |
| 1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 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_X64 | 7 #if V8_TARGET_ARCH_X64 |
| 8 | 8 |
| 9 #include "bootstrapper.h" | 9 #include "bootstrapper.h" |
| 10 #include "code-stubs.h" | 10 #include "code-stubs.h" |
| (...skipping 2210 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2221 __ Push(rdi); | 2221 __ Push(rdi); |
| 2222 __ Push(rax); | 2222 __ Push(rax); |
| 2223 __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION); | 2223 __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION); |
| 2224 __ Pop(rdi); | 2224 __ Pop(rdi); |
| 2225 } | 2225 } |
| 2226 __ movp(args->GetReceiverOperand(), rax); | 2226 __ movp(args->GetReceiverOperand(), rax); |
| 2227 __ jmp(cont); | 2227 __ jmp(cont); |
| 2228 } | 2228 } |
| 2229 | 2229 |
| 2230 | 2230 |
| 2231 void CallFunctionStub::Generate(MacroAssembler* masm) { | 2231 static void CallFunctionNoFeedback(MacroAssembler* masm, |
| 2232 int argc, bool needs_checks, |
| 2233 bool call_as_method) { |
| 2232 // rdi : the function to call | 2234 // rdi : the function to call |
| 2233 | 2235 |
| 2234 // wrap_and_call can only be true if we are compiling a monomorphic method. | 2236 // wrap_and_call can only be true if we are compiling a monomorphic method. |
| 2235 Isolate* isolate = masm->isolate(); | 2237 Isolate* isolate = masm->isolate(); |
| 2236 Label slow, non_function, wrap, cont; | 2238 Label slow, non_function, wrap, cont; |
| 2237 int argc = argc_; | |
| 2238 StackArgumentsAccessor args(rsp, argc); | 2239 StackArgumentsAccessor args(rsp, argc); |
| 2239 | 2240 |
| 2240 if (NeedsChecks()) { | 2241 if (needs_checks) { |
| 2241 // Check that the function really is a JavaScript function. | 2242 // Check that the function really is a JavaScript function. |
| 2242 __ JumpIfSmi(rdi, &non_function); | 2243 __ JumpIfSmi(rdi, &non_function); |
| 2243 | 2244 |
| 2244 // Goto slow case if we do not have a function. | 2245 // Goto slow case if we do not have a function. |
| 2245 __ CmpObjectType(rdi, JS_FUNCTION_TYPE, rcx); | 2246 __ CmpObjectType(rdi, JS_FUNCTION_TYPE, rcx); |
| 2246 __ j(not_equal, &slow); | 2247 __ j(not_equal, &slow); |
| 2247 } | 2248 } |
| 2248 | 2249 |
| 2249 // Fast-case: Just invoke the function. | 2250 // Fast-case: Just invoke the function. |
| 2250 ParameterCount actual(argc); | 2251 ParameterCount actual(argc); |
| 2251 | 2252 |
| 2252 if (CallAsMethod()) { | 2253 if (call_as_method) { |
| 2253 if (NeedsChecks()) { | 2254 if (needs_checks) { |
| 2254 EmitContinueIfStrictOrNative(masm, &cont); | 2255 EmitContinueIfStrictOrNative(masm, &cont); |
| 2255 } | 2256 } |
| 2256 | 2257 |
| 2257 // Load the receiver from the stack. | 2258 // Load the receiver from the stack. |
| 2258 __ movp(rax, args.GetReceiverOperand()); | 2259 __ movp(rax, args.GetReceiverOperand()); |
| 2259 | 2260 |
| 2260 if (NeedsChecks()) { | 2261 if (needs_checks) { |
| 2261 __ JumpIfSmi(rax, &wrap); | 2262 __ JumpIfSmi(rax, &wrap); |
| 2262 | 2263 |
| 2263 __ CmpObjectType(rax, FIRST_SPEC_OBJECT_TYPE, rcx); | 2264 __ CmpObjectType(rax, FIRST_SPEC_OBJECT_TYPE, rcx); |
| 2264 __ j(below, &wrap); | 2265 __ j(below, &wrap); |
| 2265 } else { | 2266 } else { |
| 2266 __ jmp(&wrap); | 2267 __ jmp(&wrap); |
| 2267 } | 2268 } |
| 2268 | 2269 |
| 2269 __ bind(&cont); | 2270 __ bind(&cont); |
| 2270 } | 2271 } |
| 2271 | 2272 |
| 2272 __ InvokeFunction(rdi, actual, JUMP_FUNCTION, NullCallWrapper()); | 2273 __ InvokeFunction(rdi, actual, JUMP_FUNCTION, NullCallWrapper()); |
| 2273 | 2274 |
| 2274 if (NeedsChecks()) { | 2275 if (needs_checks) { |
| 2275 // Slow-case: Non-function called. | 2276 // Slow-case: Non-function called. |
| 2276 __ bind(&slow); | 2277 __ bind(&slow); |
| 2277 EmitSlowCase(isolate, masm, &args, argc, &non_function); | 2278 EmitSlowCase(isolate, masm, &args, argc, &non_function); |
| 2278 } | 2279 } |
| 2279 | 2280 |
| 2280 if (CallAsMethod()) { | 2281 if (call_as_method) { |
| 2281 __ bind(&wrap); | 2282 __ bind(&wrap); |
| 2282 EmitWrapCase(masm, &args, &cont); | 2283 EmitWrapCase(masm, &args, &cont); |
| 2283 } | 2284 } |
| 2284 } | 2285 } |
| 2285 | 2286 |
| 2286 | 2287 |
| 2288 void CallFunctionStub::Generate(MacroAssembler* masm) { |
| 2289 CallFunctionNoFeedback(masm, argc_, NeedsChecks(), CallAsMethod()); |
| 2290 } |
| 2291 |
| 2292 |
| 2287 void CallConstructStub::Generate(MacroAssembler* masm) { | 2293 void CallConstructStub::Generate(MacroAssembler* masm) { |
| 2288 // rax : number of arguments | 2294 // rax : number of arguments |
| 2289 // rbx : feedback vector | 2295 // rbx : feedback vector |
| 2290 // rdx : (only if rbx is not the megamorphic symbol) slot in feedback | 2296 // rdx : (only if rbx is not the megamorphic symbol) slot in feedback |
| 2291 // vector (Smi) | 2297 // vector (Smi) |
| 2292 // rdi : constructor function | 2298 // rdi : constructor function |
| 2293 Label slow, non_function_call; | 2299 Label slow, non_function_call; |
| 2294 | 2300 |
| 2295 // Check that function is not a smi. | 2301 // Check that function is not a smi. |
| 2296 __ JumpIfSmi(rdi, &non_function_call); | 2302 __ JumpIfSmi(rdi, &non_function_call); |
| (...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2351 | 2357 |
| 2352 | 2358 |
| 2353 static void EmitLoadTypeFeedbackVector(MacroAssembler* masm, Register vector) { | 2359 static void EmitLoadTypeFeedbackVector(MacroAssembler* masm, Register vector) { |
| 2354 __ movp(vector, Operand(rbp, JavaScriptFrameConstants::kFunctionOffset)); | 2360 __ movp(vector, Operand(rbp, JavaScriptFrameConstants::kFunctionOffset)); |
| 2355 __ movp(vector, FieldOperand(vector, JSFunction::kSharedFunctionInfoOffset)); | 2361 __ movp(vector, FieldOperand(vector, JSFunction::kSharedFunctionInfoOffset)); |
| 2356 __ movp(vector, FieldOperand(vector, | 2362 __ movp(vector, FieldOperand(vector, |
| 2357 SharedFunctionInfo::kFeedbackVectorOffset)); | 2363 SharedFunctionInfo::kFeedbackVectorOffset)); |
| 2358 } | 2364 } |
| 2359 | 2365 |
| 2360 | 2366 |
| 2367 void CallICStub::Generate_MonomorphicArray(MacroAssembler* masm, Label* miss) { |
| 2368 // rdi - function |
| 2369 // rbx - feedback vector |
| 2370 // rdx - slot id (as integer) |
| 2371 __ LoadGlobalFunction(Context::ARRAY_FUNCTION_INDEX, rcx); |
| 2372 __ cmpq(rdi, rcx); |
| 2373 __ j(not_equal, miss); |
| 2374 |
| 2375 __ movq(rax, Immediate(arg_count())); |
| 2376 __ movp(rbx, FieldOperand(rbx, rdx, times_pointer_size, |
| 2377 FixedArray::kHeaderSize)); |
| 2378 |
| 2379 // Verify that ecx contains an AllocationSite |
| 2380 __ AssertUndefinedOrAllocationSite(rbx); |
| 2381 ArrayConstructorStub stub(masm->isolate(), arg_count()); |
| 2382 __ TailCallStub(&stub); |
| 2383 } |
| 2384 |
| 2385 |
| 2386 void CallICStub::Generate_CustomFeedbackCall(MacroAssembler* masm) { |
| 2387 // rdi - function |
| 2388 // rbx - feedback vector |
| 2389 // rdx - slot id |
| 2390 Label miss; |
| 2391 |
| 2392 __ SmiToInteger32(rdx, rdx); |
| 2393 |
| 2394 if (state_.stub_type() == CallIC::MONOMORPHIC_ARRAY) { |
| 2395 Generate_MonomorphicArray(masm, &miss); |
| 2396 } else { |
| 2397 // So far there is only one customer for our custom feedback scheme. |
| 2398 UNREACHABLE(); |
| 2399 } |
| 2400 |
| 2401 __ bind(&miss); |
| 2402 GenerateMiss(masm); |
| 2403 |
| 2404 // The slow case, we need this no matter what to complete a call after a miss. |
| 2405 CallFunctionNoFeedback(masm, |
| 2406 arg_count(), |
| 2407 true, |
| 2408 CallAsMethod()); |
| 2409 |
| 2410 // Unreachable. |
| 2411 __ int3(); |
| 2412 } |
| 2413 |
| 2414 |
| 2361 void CallICStub::Generate(MacroAssembler* masm) { | 2415 void CallICStub::Generate(MacroAssembler* masm) { |
| 2362 // rdi - function | 2416 // rdi - function |
| 2363 // rbx - vector | 2417 // rbx - vector |
| 2364 // rdx - slot id | 2418 // rdx - slot id |
| 2365 Isolate* isolate = masm->isolate(); | 2419 Isolate* isolate = masm->isolate(); |
| 2366 Label extra_checks_or_miss, slow_start; | 2420 Label extra_checks_or_miss, slow_start; |
| 2367 Label slow, non_function, wrap, cont; | 2421 Label slow, non_function, wrap, cont; |
| 2368 Label have_js_function; | 2422 Label have_js_function; |
| 2369 int argc = state_.arg_count(); | 2423 int argc = state_.arg_count(); |
| 2370 StackArgumentsAccessor args(rsp, argc); | 2424 StackArgumentsAccessor args(rsp, argc); |
| 2371 ParameterCount actual(argc); | 2425 ParameterCount actual(argc); |
| 2372 | 2426 |
| 2373 EmitLoadTypeFeedbackVector(masm, rbx); | 2427 EmitLoadTypeFeedbackVector(masm, rbx); |
| 2374 | 2428 |
| 2429 if (state_.stub_type() != CallIC::DEFAULT) { |
| 2430 Generate_CustomFeedbackCall(masm); |
| 2431 return; |
| 2432 } |
| 2433 |
| 2375 // The checks. First, does rdi match the recorded monomorphic target? | 2434 // The checks. First, does rdi match the recorded monomorphic target? |
| 2376 __ SmiToInteger32(rdx, rdx); | 2435 __ SmiToInteger32(rdx, rdx); |
| 2377 __ cmpq(rdi, FieldOperand(rbx, rdx, times_pointer_size, | 2436 __ cmpq(rdi, FieldOperand(rbx, rdx, times_pointer_size, |
| 2378 FixedArray::kHeaderSize)); | 2437 FixedArray::kHeaderSize)); |
| 2379 __ j(not_equal, &extra_checks_or_miss); | 2438 __ j(not_equal, &extra_checks_or_miss); |
| 2380 | 2439 |
| 2381 __ bind(&have_js_function); | 2440 __ bind(&have_js_function); |
| 2382 if (state_.CallAsMethod()) { | 2441 if (state_.CallAsMethod()) { |
| 2383 EmitContinueIfStrictOrNative(masm, &cont); | 2442 EmitContinueIfStrictOrNative(masm, &cont); |
| 2384 | 2443 |
| (...skipping 2651 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 5036 return_value_operand, | 5095 return_value_operand, |
| 5037 NULL); | 5096 NULL); |
| 5038 } | 5097 } |
| 5039 | 5098 |
| 5040 | 5099 |
| 5041 #undef __ | 5100 #undef __ |
| 5042 | 5101 |
| 5043 } } // namespace v8::internal | 5102 } } // namespace v8::internal |
| 5044 | 5103 |
| 5045 #endif // V8_TARGET_ARCH_X64 | 5104 #endif // V8_TARGET_ARCH_X64 |
| OLD | NEW |