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 "src/v8.h" | 5 #include "src/v8.h" |
6 | 6 |
7 #if V8_TARGET_ARCH_X64 | 7 #if V8_TARGET_ARCH_X64 |
8 | 8 |
9 #include "src/code-stubs.h" | 9 #include "src/code-stubs.h" |
10 #include "src/codegen.h" | 10 #include "src/codegen.h" |
(...skipping 2364 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2375 Variable* var, MemOperand location) { | 2375 Variable* var, MemOperand location) { |
2376 __ movp(location, rax); | 2376 __ movp(location, rax); |
2377 if (var->IsContextSlot()) { | 2377 if (var->IsContextSlot()) { |
2378 __ movp(rdx, rax); | 2378 __ movp(rdx, rax); |
2379 __ RecordWriteContextSlot( | 2379 __ RecordWriteContextSlot( |
2380 rcx, Context::SlotOffset(var->index()), rdx, rbx, kDontSaveFPRegs); | 2380 rcx, Context::SlotOffset(var->index()), rdx, rbx, kDontSaveFPRegs); |
2381 } | 2381 } |
2382 } | 2382 } |
2383 | 2383 |
2384 | 2384 |
2385 void FullCodeGenerator::EmitCallStoreContextSlot( | |
2386 Handle<String> name, StrictMode strict_mode) { | |
2387 __ Push(rax); // Value. | |
2388 __ Push(rsi); // Context. | |
2389 __ Push(name); | |
2390 __ Push(Smi::FromInt(strict_mode)); | |
2391 __ CallRuntime(Runtime::kStoreContextSlot, 4); | |
2392 } | |
2393 | |
2394 | |
2395 void FullCodeGenerator::EmitVariableAssignment(Variable* var, | 2385 void FullCodeGenerator::EmitVariableAssignment(Variable* var, |
2396 Token::Value op) { | 2386 Token::Value op) { |
2397 if (var->IsUnallocated()) { | 2387 if (var->IsUnallocated()) { |
2398 // Global var, const, or let. | 2388 // Global var, const, or let. |
2399 __ Move(rcx, var->name()); | 2389 __ Move(rcx, var->name()); |
2400 __ movp(rdx, GlobalObjectOperand()); | 2390 __ movp(rdx, GlobalObjectOperand()); |
2401 CallStoreIC(); | 2391 CallStoreIC(); |
2402 | 2392 |
2403 } else if (op == Token::INIT_CONST_LEGACY) { | 2393 } else if (op == Token::INIT_CONST_LEGACY) { |
2404 // Const initializers need a write barrier. | 2394 // Const initializers need a write barrier. |
2405 ASSERT(!var->IsParameter()); // No const parameters. | 2395 ASSERT(!var->IsParameter()); // No const parameters. |
2406 if (var->IsLookupSlot()) { | 2396 if (var->IsLookupSlot()) { |
2407 __ Push(rax); | 2397 __ Push(rax); |
2408 __ Push(rsi); | 2398 __ Push(rsi); |
2409 __ Push(var->name()); | 2399 __ Push(var->name()); |
2410 __ CallRuntime(Runtime::kInitializeConstContextSlot, 3); | 2400 __ CallRuntime(Runtime::kInitializeConstContextSlot, 3); |
2411 } else { | 2401 } else { |
2412 ASSERT(var->IsStackLocal() || var->IsContextSlot()); | 2402 ASSERT(var->IsStackLocal() || var->IsContextSlot()); |
2413 Label skip; | 2403 Label skip; |
2414 MemOperand location = VarOperand(var, rcx); | 2404 MemOperand location = VarOperand(var, rcx); |
2415 __ movp(rdx, location); | 2405 __ movp(rdx, location); |
2416 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); | 2406 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); |
2417 __ j(not_equal, &skip); | 2407 __ j(not_equal, &skip); |
2418 EmitStoreToStackLocalOrContextSlot(var, location); | 2408 EmitStoreToStackLocalOrContextSlot(var, location); |
2419 __ bind(&skip); | 2409 __ bind(&skip); |
2420 } | 2410 } |
2421 | 2411 |
2422 } else if (var->mode() == LET && op != Token::INIT_LET) { | 2412 } else if (var->mode() == LET && op != Token::INIT_LET) { |
2423 // Non-initializing assignment to let variable needs a write barrier. | 2413 // Non-initializing assignment to let variable needs a write barrier. |
2424 if (var->IsLookupSlot()) { | 2414 ASSERT(!var->IsLookupSlot()); |
2425 EmitCallStoreContextSlot(var->name(), strict_mode()); | 2415 ASSERT(var->IsStackAllocated() || var->IsContextSlot()); |
2426 } else { | 2416 Label assign; |
2427 ASSERT(var->IsStackAllocated() || var->IsContextSlot()); | 2417 MemOperand location = VarOperand(var, rcx); |
2428 Label assign; | 2418 __ movp(rdx, location); |
2429 MemOperand location = VarOperand(var, rcx); | 2419 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); |
2430 __ movp(rdx, location); | 2420 __ j(not_equal, &assign, Label::kNear); |
2431 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); | 2421 __ Push(var->name()); |
2432 __ j(not_equal, &assign, Label::kNear); | 2422 __ CallRuntime(Runtime::kThrowReferenceError, 1); |
2433 __ Push(var->name()); | 2423 __ bind(&assign); |
2434 __ CallRuntime(Runtime::kThrowReferenceError, 1); | 2424 EmitStoreToStackLocalOrContextSlot(var, location); |
2435 __ bind(&assign); | |
2436 EmitStoreToStackLocalOrContextSlot(var, location); | |
2437 } | |
2438 | 2425 |
2439 } else if (!var->is_const_mode() || op == Token::INIT_CONST) { | 2426 } else if (!var->is_const_mode() || op == Token::INIT_CONST) { |
2440 // Assignment to var or initializing assignment to let/const | |
2441 // in harmony mode. | |
2442 if (var->IsLookupSlot()) { | 2427 if (var->IsLookupSlot()) { |
2443 EmitCallStoreContextSlot(var->name(), strict_mode()); | 2428 ASSERT(opt == Token::ASSIGN || op == Token::INIT_VAR || |
| 2429 op == Token::ASSIGN_ADD); |
| 2430 // Assignment to var. |
| 2431 __ Push(rax); // Value. |
| 2432 __ Push(rsi); // Context. |
| 2433 __ Push(var->name()); |
| 2434 __ Push(Smi::FromInt(strict_mode())); |
| 2435 __ CallRuntime(Runtime::kStoreLookupSlot, 4); |
2444 } else { | 2436 } else { |
| 2437 // Assignment to var or initializing assignment to let/const in harmony |
| 2438 // mode. |
2445 ASSERT(var->IsStackAllocated() || var->IsContextSlot()); | 2439 ASSERT(var->IsStackAllocated() || var->IsContextSlot()); |
2446 MemOperand location = VarOperand(var, rcx); | 2440 MemOperand location = VarOperand(var, rcx); |
2447 if (generate_debug_code_ && op == Token::INIT_LET) { | 2441 if (generate_debug_code_ && op == Token::INIT_LET) { |
2448 // Check for an uninitialized let binding. | 2442 // Check for an uninitialized let binding. |
2449 __ movp(rdx, location); | 2443 __ movp(rdx, location); |
2450 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); | 2444 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); |
2451 __ Check(equal, kLetBindingReInitialization); | 2445 __ Check(equal, kLetBindingReInitialization); |
2452 } | 2446 } |
2453 EmitStoreToStackLocalOrContextSlot(var, location); | 2447 EmitStoreToStackLocalOrContextSlot(var, location); |
2454 } | 2448 } |
(...skipping 2357 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4812 ASSERT_EQ(isolate->builtins()->OsrAfterStackCheck()->entry(), | 4806 ASSERT_EQ(isolate->builtins()->OsrAfterStackCheck()->entry(), |
4813 Assembler::target_address_at(call_target_address, | 4807 Assembler::target_address_at(call_target_address, |
4814 unoptimized_code)); | 4808 unoptimized_code)); |
4815 return OSR_AFTER_STACK_CHECK; | 4809 return OSR_AFTER_STACK_CHECK; |
4816 } | 4810 } |
4817 | 4811 |
4818 | 4812 |
4819 } } // namespace v8::internal | 4813 } } // namespace v8::internal |
4820 | 4814 |
4821 #endif // V8_TARGET_ARCH_X64 | 4815 #endif // V8_TARGET_ARCH_X64 |
OLD | NEW |