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_X64 | 5 #if V8_TARGET_ARCH_X64 |
6 | 6 |
7 #include "src/ast/scopes.h" | 7 #include "src/ast/scopes.h" |
8 #include "src/code-factory.h" | 8 #include "src/code-factory.h" |
9 #include "src/code-stubs.h" | 9 #include "src/code-stubs.h" |
10 #include "src/codegen.h" | 10 #include "src/codegen.h" |
(...skipping 701 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
712 __ CompareRoot(rbx, Heap::kWithContextMapRootIndex); | 712 __ CompareRoot(rbx, Heap::kWithContextMapRootIndex); |
713 __ Check(not_equal, kDeclarationInWithContext); | 713 __ Check(not_equal, kDeclarationInWithContext); |
714 __ CompareRoot(rbx, Heap::kCatchContextMapRootIndex); | 714 __ CompareRoot(rbx, Heap::kCatchContextMapRootIndex); |
715 __ Check(not_equal, kDeclarationInCatchContext); | 715 __ Check(not_equal, kDeclarationInCatchContext); |
716 } | 716 } |
717 } | 717 } |
718 | 718 |
719 | 719 |
720 void FullCodeGenerator::VisitVariableDeclaration( | 720 void FullCodeGenerator::VisitVariableDeclaration( |
721 VariableDeclaration* declaration) { | 721 VariableDeclaration* declaration) { |
722 // If it was not possible to allocate the variable at compile time, we | |
723 // need to "declare" it at runtime to make sure it actually exists in the | |
724 // local context. | |
725 VariableProxy* proxy = declaration->proxy(); | 722 VariableProxy* proxy = declaration->proxy(); |
726 VariableMode mode = declaration->mode(); | |
727 Variable* variable = proxy->var(); | 723 Variable* variable = proxy->var(); |
728 bool hole_init = mode == LET || mode == CONST; | |
729 switch (variable->location()) { | 724 switch (variable->location()) { |
730 case VariableLocation::GLOBAL: | 725 case VariableLocation::GLOBAL: |
731 case VariableLocation::UNALLOCATED: { | 726 case VariableLocation::UNALLOCATED: { |
732 DCHECK(!variable->binding_needs_init()); | 727 DCHECK(!variable->binding_needs_init()); |
733 FeedbackVectorSlot slot = proxy->VariableFeedbackSlot(); | 728 FeedbackVectorSlot slot = proxy->VariableFeedbackSlot(); |
734 DCHECK(!slot.IsInvalid()); | 729 DCHECK(!slot.IsInvalid()); |
735 globals_->Add(handle(Smi::FromInt(slot.ToInt()), isolate()), zone()); | 730 globals_->Add(handle(Smi::FromInt(slot.ToInt()), isolate()), zone()); |
736 globals_->Add(isolate()->factory()->undefined_value(), zone()); | 731 globals_->Add(isolate()->factory()->undefined_value(), zone()); |
737 break; | 732 break; |
738 } | 733 } |
739 case VariableLocation::PARAMETER: | 734 case VariableLocation::PARAMETER: |
740 case VariableLocation::LOCAL: | 735 case VariableLocation::LOCAL: |
741 if (hole_init) { | 736 if (variable->binding_needs_init()) { |
742 Comment cmnt(masm_, "[ VariableDeclaration"); | 737 Comment cmnt(masm_, "[ VariableDeclaration"); |
743 __ LoadRoot(kScratchRegister, Heap::kTheHoleValueRootIndex); | 738 __ LoadRoot(kScratchRegister, Heap::kTheHoleValueRootIndex); |
744 __ movp(StackOperand(variable), kScratchRegister); | 739 __ movp(StackOperand(variable), kScratchRegister); |
745 } | 740 } |
746 break; | 741 break; |
747 | 742 |
748 case VariableLocation::CONTEXT: | 743 case VariableLocation::CONTEXT: |
749 if (hole_init) { | 744 if (variable->binding_needs_init()) { |
750 Comment cmnt(masm_, "[ VariableDeclaration"); | 745 Comment cmnt(masm_, "[ VariableDeclaration"); |
751 EmitDebugCheckDeclarationContext(variable); | 746 EmitDebugCheckDeclarationContext(variable); |
752 __ LoadRoot(kScratchRegister, Heap::kTheHoleValueRootIndex); | 747 __ LoadRoot(kScratchRegister, Heap::kTheHoleValueRootIndex); |
753 __ movp(ContextOperand(rsi, variable->index()), kScratchRegister); | 748 __ movp(ContextOperand(rsi, variable->index()), kScratchRegister); |
754 // No write barrier since the hole value is in old space. | 749 // No write barrier since the hole value is in old space. |
755 PrepareForBailoutForId(proxy->id(), BailoutState::NO_REGISTERS); | 750 PrepareForBailoutForId(proxy->id(), BailoutState::NO_REGISTERS); |
756 } | 751 } |
757 break; | 752 break; |
758 | 753 |
759 case VariableLocation::LOOKUP: { | 754 case VariableLocation::LOOKUP: { |
760 Comment cmnt(masm_, "[ VariableDeclaration"); | 755 Comment cmnt(masm_, "[ VariableDeclaration"); |
761 DCHECK_EQ(VAR, mode); | 756 DCHECK_EQ(VAR, variable->mode()); |
762 DCHECK(!hole_init); | 757 DCHECK(!variable->binding_needs_init()); |
763 __ Push(variable->name()); | 758 __ Push(variable->name()); |
764 __ CallRuntime(Runtime::kDeclareEvalVar); | 759 __ CallRuntime(Runtime::kDeclareEvalVar); |
765 PrepareForBailoutForId(proxy->id(), BailoutState::NO_REGISTERS); | 760 PrepareForBailoutForId(proxy->id(), BailoutState::NO_REGISTERS); |
766 break; | 761 break; |
767 } | 762 } |
768 } | 763 } |
769 } | 764 } |
770 | 765 |
771 | 766 |
772 void FullCodeGenerator::VisitFunctionDeclaration( | 767 void FullCodeGenerator::VisitFunctionDeclaration( |
(...skipping 443 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1216 // introducing variables. In those cases, we do not want to | 1211 // introducing variables. In those cases, we do not want to |
1217 // perform a runtime call for all variables in the scope | 1212 // perform a runtime call for all variables in the scope |
1218 // containing the eval. | 1213 // containing the eval. |
1219 Variable* var = proxy->var(); | 1214 Variable* var = proxy->var(); |
1220 if (var->mode() == DYNAMIC_GLOBAL) { | 1215 if (var->mode() == DYNAMIC_GLOBAL) { |
1221 EmitLoadGlobalCheckExtensions(proxy, typeof_mode, slow); | 1216 EmitLoadGlobalCheckExtensions(proxy, typeof_mode, slow); |
1222 __ jmp(done); | 1217 __ jmp(done); |
1223 } else if (var->mode() == DYNAMIC_LOCAL) { | 1218 } else if (var->mode() == DYNAMIC_LOCAL) { |
1224 Variable* local = var->local_if_not_shadowed(); | 1219 Variable* local = var->local_if_not_shadowed(); |
1225 __ movp(rax, ContextSlotOperandCheckExtensions(local, slow)); | 1220 __ movp(rax, ContextSlotOperandCheckExtensions(local, slow)); |
1226 if (local->mode() == LET || local->mode() == CONST) { | 1221 if (local->binding_needs_init()) { |
1227 __ CompareRoot(rax, Heap::kTheHoleValueRootIndex); | 1222 __ CompareRoot(rax, Heap::kTheHoleValueRootIndex); |
1228 __ j(not_equal, done); | 1223 __ j(not_equal, done); |
1229 __ Push(var->name()); | 1224 __ Push(var->name()); |
1230 __ CallRuntime(Runtime::kThrowReferenceError); | 1225 __ CallRuntime(Runtime::kThrowReferenceError); |
| 1226 } else { |
| 1227 __ jmp(done); |
1231 } | 1228 } |
1232 __ jmp(done); | |
1233 } | 1229 } |
1234 } | 1230 } |
1235 | 1231 |
1236 | 1232 |
1237 void FullCodeGenerator::EmitGlobalVariableLoad(VariableProxy* proxy, | 1233 void FullCodeGenerator::EmitGlobalVariableLoad(VariableProxy* proxy, |
1238 TypeofMode typeof_mode) { | 1234 TypeofMode typeof_mode) { |
1239 #ifdef DEBUG | 1235 #ifdef DEBUG |
1240 Variable* var = proxy->var(); | 1236 Variable* var = proxy->var(); |
1241 DCHECK(var->IsUnallocatedOrGlobalSlot() || | 1237 DCHECK(var->IsUnallocatedOrGlobalSlot() || |
1242 (var->IsLookupSlot() && var->mode() == DYNAMIC_GLOBAL)); | 1238 (var->IsLookupSlot() && var->mode() == DYNAMIC_GLOBAL)); |
(...skipping 22 matching lines...) Expand all Loading... |
1265 break; | 1261 break; |
1266 } | 1262 } |
1267 | 1263 |
1268 case VariableLocation::PARAMETER: | 1264 case VariableLocation::PARAMETER: |
1269 case VariableLocation::LOCAL: | 1265 case VariableLocation::LOCAL: |
1270 case VariableLocation::CONTEXT: { | 1266 case VariableLocation::CONTEXT: { |
1271 DCHECK_EQ(NOT_INSIDE_TYPEOF, typeof_mode); | 1267 DCHECK_EQ(NOT_INSIDE_TYPEOF, typeof_mode); |
1272 Comment cmnt(masm_, var->IsContextSlot() ? "[ Context slot" | 1268 Comment cmnt(masm_, var->IsContextSlot() ? "[ Context slot" |
1273 : "[ Stack slot"); | 1269 : "[ Stack slot"); |
1274 if (NeedsHoleCheckForLoad(proxy)) { | 1270 if (NeedsHoleCheckForLoad(proxy)) { |
1275 // Let and const need a read barrier. | 1271 // Throw a reference error when using an uninitialized let/const |
| 1272 // binding in harmony mode. |
| 1273 DCHECK(IsLexicalVariableMode(var->mode())); |
1276 Label done; | 1274 Label done; |
1277 GetVar(rax, var); | 1275 GetVar(rax, var); |
1278 __ CompareRoot(rax, Heap::kTheHoleValueRootIndex); | 1276 __ CompareRoot(rax, Heap::kTheHoleValueRootIndex); |
1279 __ j(not_equal, &done, Label::kNear); | 1277 __ j(not_equal, &done, Label::kNear); |
1280 if (var->mode() == LET || var->mode() == CONST) { | 1278 __ Push(var->name()); |
1281 // Throw a reference error when using an uninitialized let/const | 1279 __ CallRuntime(Runtime::kThrowReferenceError); |
1282 // binding in harmony mode. | |
1283 __ Push(var->name()); | |
1284 __ CallRuntime(Runtime::kThrowReferenceError); | |
1285 } | |
1286 __ bind(&done); | 1280 __ bind(&done); |
1287 context()->Plug(rax); | 1281 context()->Plug(rax); |
1288 break; | 1282 break; |
1289 } | 1283 } |
1290 context()->Plug(var); | 1284 context()->Plug(var); |
1291 break; | 1285 break; |
1292 } | 1286 } |
1293 | 1287 |
1294 case VariableLocation::LOOKUP: { | 1288 case VariableLocation::LOOKUP: { |
1295 Comment cmnt(masm_, "[ Lookup slot"); | 1289 Comment cmnt(masm_, "[ Lookup slot"); |
(...skipping 761 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2057 | 2051 |
2058 void FullCodeGenerator::EmitVariableAssignment(Variable* var, Token::Value op, | 2052 void FullCodeGenerator::EmitVariableAssignment(Variable* var, Token::Value op, |
2059 FeedbackVectorSlot slot) { | 2053 FeedbackVectorSlot slot) { |
2060 if (var->IsUnallocated()) { | 2054 if (var->IsUnallocated()) { |
2061 // Global var, const, or let. | 2055 // Global var, const, or let. |
2062 __ Move(StoreDescriptor::NameRegister(), var->name()); | 2056 __ Move(StoreDescriptor::NameRegister(), var->name()); |
2063 __ LoadGlobalObject(StoreDescriptor::ReceiverRegister()); | 2057 __ LoadGlobalObject(StoreDescriptor::ReceiverRegister()); |
2064 EmitLoadStoreICSlot(slot); | 2058 EmitLoadStoreICSlot(slot); |
2065 CallStoreIC(); | 2059 CallStoreIC(); |
2066 | 2060 |
2067 } else if (var->mode() == LET && op != Token::INIT) { | 2061 } else if (IsLexicalVariableMode(var->mode()) && op != Token::INIT) { |
2068 // Non-initializing assignment to let variable needs a write barrier. | |
2069 DCHECK(!var->IsLookupSlot()); | 2062 DCHECK(!var->IsLookupSlot()); |
2070 DCHECK(var->IsStackAllocated() || var->IsContextSlot()); | 2063 DCHECK(var->IsStackAllocated() || var->IsContextSlot()); |
2071 Label assign; | |
2072 MemOperand location = VarOperand(var, rcx); | 2064 MemOperand location = VarOperand(var, rcx); |
2073 __ movp(rdx, location); | 2065 // Perform an initialization check for lexically declared variables. |
2074 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); | 2066 if (var->binding_needs_init()) { |
2075 __ j(not_equal, &assign, Label::kNear); | 2067 Label assign; |
2076 __ Push(var->name()); | 2068 __ movp(rdx, location); |
2077 __ CallRuntime(Runtime::kThrowReferenceError); | 2069 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); |
2078 __ bind(&assign); | 2070 __ j(not_equal, &assign, Label::kNear); |
2079 EmitStoreToStackLocalOrContextSlot(var, location); | 2071 __ Push(var->name()); |
2080 | 2072 __ CallRuntime(Runtime::kThrowReferenceError); |
2081 } else if (var->mode() == CONST && op != Token::INIT) { | 2073 __ bind(&assign); |
2082 // Assignment to const variable needs a write barrier. | 2074 } |
2083 DCHECK(!var->IsLookupSlot()); | 2075 if (var->mode() == CONST) { |
2084 DCHECK(var->IsStackAllocated() || var->IsContextSlot()); | 2076 __ CallRuntime(Runtime::kThrowConstAssignError); |
2085 Label const_error; | 2077 } else { |
2086 MemOperand location = VarOperand(var, rcx); | 2078 EmitStoreToStackLocalOrContextSlot(var, location); |
2087 __ movp(rdx, location); | 2079 } |
2088 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); | |
2089 __ j(not_equal, &const_error, Label::kNear); | |
2090 __ Push(var->name()); | |
2091 __ CallRuntime(Runtime::kThrowReferenceError); | |
2092 __ bind(&const_error); | |
2093 __ CallRuntime(Runtime::kThrowConstAssignError); | |
2094 | 2080 |
2095 } else if (var->is_this() && var->mode() == CONST && op == Token::INIT) { | 2081 } else if (var->is_this() && var->mode() == CONST && op == Token::INIT) { |
2096 // Initializing assignment to const {this} needs a write barrier. | 2082 // Initializing assignment to const {this} needs a write barrier. |
2097 DCHECK(var->IsStackAllocated() || var->IsContextSlot()); | 2083 DCHECK(var->IsStackAllocated() || var->IsContextSlot()); |
2098 Label uninitialized_this; | 2084 Label uninitialized_this; |
2099 MemOperand location = VarOperand(var, rcx); | 2085 MemOperand location = VarOperand(var, rcx); |
2100 __ movp(rdx, location); | 2086 __ movp(rdx, location); |
2101 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); | 2087 __ CompareRoot(rdx, Heap::kTheHoleValueRootIndex); |
2102 __ j(equal, &uninitialized_this); | 2088 __ j(equal, &uninitialized_this); |
2103 __ Push(var->name()); | 2089 __ Push(var->name()); |
(...skipping 1554 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3658 DCHECK_EQ( | 3644 DCHECK_EQ( |
3659 isolate->builtins()->OnStackReplacement()->entry(), | 3645 isolate->builtins()->OnStackReplacement()->entry(), |
3660 Assembler::target_address_at(call_target_address, unoptimized_code)); | 3646 Assembler::target_address_at(call_target_address, unoptimized_code)); |
3661 return ON_STACK_REPLACEMENT; | 3647 return ON_STACK_REPLACEMENT; |
3662 } | 3648 } |
3663 | 3649 |
3664 } // namespace internal | 3650 } // namespace internal |
3665 } // namespace v8 | 3651 } // namespace v8 |
3666 | 3652 |
3667 #endif // V8_TARGET_ARCH_X64 | 3653 #endif // V8_TARGET_ARCH_X64 |
OLD | NEW |