| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org> | 
					
						
							|  |  |  |  * All rights reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions are met: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 1. Redistributions of source code must retain the above copyright notice, this | 
					
						
							|  |  |  |  *    list of conditions and the following disclaimer. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 2. Redistributions in binary form must reproduce the above copyright notice, | 
					
						
							|  |  |  |  *    this list of conditions and the following disclaimer in the documentation | 
					
						
							|  |  |  |  *    and/or other materials provided with the distribution. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
					
						
							|  |  |  |  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
					
						
							|  |  |  |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
					
						
							|  |  |  |  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
					
						
							|  |  |  |  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
					
						
							|  |  |  |  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
					
						
							|  |  |  |  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
					
						
							|  |  |  |  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
					
						
							|  |  |  |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
					
						
							|  |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Function.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | #include <LibJS/Runtime/LexicalEnvironment.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LexicalEnvironment::LexicalEnvironment() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | LexicalEnvironment::LexicalEnvironment(EnvironmentRecordType environment_record_type) | 
					
						
							|  |  |  |     : m_environment_record_type(environment_record_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | LexicalEnvironment::LexicalEnvironment(HashMap<FlyString, Variable> variables, LexicalEnvironment* parent) | 
					
						
							|  |  |  |     : m_parent(parent) | 
					
						
							|  |  |  |     , m_variables(move(variables)) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | LexicalEnvironment::LexicalEnvironment(HashMap<FlyString, Variable> variables, LexicalEnvironment* parent, EnvironmentRecordType environment_record_type) | 
					
						
							|  |  |  |     : m_parent(parent) | 
					
						
							|  |  |  |     , m_variables(move(variables)) | 
					
						
							|  |  |  |     , m_environment_record_type(environment_record_type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | LexicalEnvironment::~LexicalEnvironment() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void LexicalEnvironment::visit_children(Visitor& visitor) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Cell::visit_children(visitor); | 
					
						
							| 
									
										
										
										
											2020-04-16 16:07:50 +02:00
										 |  |  |     visitor.visit(m_parent); | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  |     visitor.visit(m_this_value); | 
					
						
							|  |  |  |     visitor.visit(m_home_object); | 
					
						
							|  |  |  |     visitor.visit(m_new_target); | 
					
						
							|  |  |  |     visitor.visit(m_current_function); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |     for (auto& it : m_variables) | 
					
						
							|  |  |  |         visitor.visit(it.value.value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Optional<Variable> LexicalEnvironment::get(const FlyString& name) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     ASSERT(type() != EnvironmentRecordType::Global); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |     return m_variables.get(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  | void LexicalEnvironment::set(GlobalObject& global_object, const FlyString& name, Variable variable) | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     if (type() == EnvironmentRecordType::Global) | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  |         global_object.put(name, variable.value); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     else | 
					
						
							|  |  |  |         m_variables.set(name, variable); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | bool LexicalEnvironment::has_super_binding() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_environment_record_type == EnvironmentRecordType::Function && this_binding_status() != ThisBindingStatus::Lexical && m_home_object.is_object(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value LexicalEnvironment::get_super_base() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(has_super_binding()); | 
					
						
							|  |  |  |     if (m_home_object.is_object()) | 
					
						
							|  |  |  |         return m_home_object.as_object().prototype(); | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool LexicalEnvironment::has_this_binding() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // More like "is_capable_of_having_a_this_binding".
 | 
					
						
							|  |  |  |     switch (m_environment_record_type) { | 
					
						
							|  |  |  |     case EnvironmentRecordType::Declarative: | 
					
						
							|  |  |  |     case EnvironmentRecordType::Object: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     case EnvironmentRecordType::Function: | 
					
						
							|  |  |  |         return this_binding_status() != ThisBindingStatus::Lexical; | 
					
						
							|  |  |  |     case EnvironmentRecordType::Module: | 
					
						
							|  |  |  |     case EnvironmentRecordType::Global: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  | Value LexicalEnvironment::get_this_binding(GlobalObject& global_object) const | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | { | 
					
						
							|  |  |  |     ASSERT(has_this_binding()); | 
					
						
							| 
									
										
										
										
											2020-08-25 12:52:32 +02:00
										 |  |  |     if (this_binding_status() == ThisBindingStatus::Uninitialized) { | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  |         vm().throw_exception<ReferenceError>(global_object, ErrorType::ThisHasNotBeenInitialized); | 
					
						
							| 
									
										
										
										
											2020-08-25 12:52:32 +02:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  |     return m_this_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  | void LexicalEnvironment::bind_this_value(GlobalObject& global_object, Value this_value) | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  | { | 
					
						
							|  |  |  |     ASSERT(has_this_binding()); | 
					
						
							|  |  |  |     if (m_this_binding_status == ThisBindingStatus::Initialized) { | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  |         vm().throw_exception<ReferenceError>(global_object, ErrorType::ThisIsAlreadyInitialized); | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     m_this_value = this_value; | 
					
						
							|  |  |  |     m_this_binding_status = ThisBindingStatus::Initialized; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  | } |