| 
									
										
										
										
											2020-09-20 19:24:44 +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-09-27 15:18:55 +02:00
										 |  |  | #include <AK/ScopeGuard.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Reference.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/ScriptFunction.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-22 16:18:51 +02:00
										 |  |  | #include <LibJS/Runtime/Symbol.h>
 | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  | #include <LibJS/Runtime/VM.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | //#define VM_DEBUG
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NonnullRefPtr<VM> VM::create() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return adopt(*new VM); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VM::VM() | 
					
						
							|  |  |  |     : m_heap(*this) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-22 16:36:33 +02:00
										 |  |  |     m_empty_string = m_heap.allocate_without_global_object<PrimitiveString>(String::empty()); | 
					
						
							| 
									
										
										
										
											2020-10-22 17:43:48 +02:00
										 |  |  |     for (size_t i = 0; i < 128; ++i) { | 
					
						
							|  |  |  |         m_single_ascii_character_strings[i] = m_heap.allocate_without_global_object<PrimitiveString>(String::format("%c", i)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     m_scope_object_shape = m_heap.allocate_without_global_object<Shape>(Shape::ShapeWithoutGlobalObjectTag::Tag); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-22 16:18:51 +02:00
										 |  |  | #define __JS_ENUMERATE(SymbolName, snake_name) \
 | 
					
						
							|  |  |  |     m_well_known_symbol_##snake_name = js_symbol(*this, "Symbol." #SymbolName, false); | 
					
						
							|  |  |  |     JS_ENUMERATE_WELL_KNOWN_SYMBOLS | 
					
						
							|  |  |  | #undef __JS_ENUMERATE
 | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VM::~VM() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Interpreter& VM::interpreter() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-21 14:42:26 +02:00
										 |  |  |     ASSERT(!m_interpreters.is_empty()); | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  |     return *m_interpreters.last(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Interpreter* VM::interpreter_if_exists() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_interpreters.is_empty()) | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     return m_interpreters.last(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void VM::push_interpreter(Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_interpreters.append(&interpreter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void VM::pop_interpreter(Interpreter& interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(!m_interpreters.is_empty()); | 
					
						
							|  |  |  |     auto* popped_interpreter = m_interpreters.take_last(); | 
					
						
							|  |  |  |     ASSERT(popped_interpreter == &interpreter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-21 13:36:32 +02:00
										 |  |  | VM::InterpreterExecutionScope::InterpreterExecutionScope(Interpreter& interpreter) | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  |     : m_interpreter(interpreter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_interpreter.vm().push_interpreter(m_interpreter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-21 13:36:32 +02:00
										 |  |  | VM::InterpreterExecutionScope::~InterpreterExecutionScope() | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     m_interpreter.vm().pop_interpreter(m_interpreter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-21 13:47:33 +02:00
										 |  |  | void VM::gather_roots(HashTable<Cell*>& roots) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-22 16:36:33 +02:00
										 |  |  |     roots.set(m_empty_string); | 
					
						
							| 
									
										
										
										
											2020-10-22 17:43:48 +02:00
										 |  |  |     for (auto* string : m_single_ascii_character_strings) | 
					
						
							|  |  |  |         roots.set(string); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     roots.set(m_scope_object_shape); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-21 15:28:09 +02:00
										 |  |  |     if (m_exception) | 
					
						
							|  |  |  |         roots.set(m_exception); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_last_value.is_cell()) | 
					
						
							|  |  |  |         roots.set(m_last_value.as_cell()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& call_frame : m_call_stack) { | 
					
						
							| 
									
										
										
										
											2020-11-07 11:07:17 +01:00
										 |  |  |         if (call_frame->this_value.is_cell()) | 
					
						
							|  |  |  |             roots.set(call_frame->this_value.as_cell()); | 
					
						
							|  |  |  |         for (auto& argument : call_frame->arguments) { | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |             if (argument.is_cell()) | 
					
						
							|  |  |  |                 roots.set(argument.as_cell()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |         roots.set(call_frame->scope); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-09-22 16:18:51 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define __JS_ENUMERATE(SymbolName, snake_name) \
 | 
					
						
							|  |  |  |     roots.set(well_known_symbol_##snake_name()); | 
					
						
							|  |  |  |     JS_ENUMERATE_WELL_KNOWN_SYMBOLS | 
					
						
							|  |  |  | #undef __JS_ENUMERATE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& symbol : m_global_symbol_map) | 
					
						
							|  |  |  |         roots.set(symbol.value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Symbol* VM::get_global_symbol(const String& description) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto result = m_global_symbol_map.get(description); | 
					
						
							|  |  |  |     if (result.has_value()) | 
					
						
							|  |  |  |         return result.value(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto new_global_symbol = js_symbol(*this, description, true); | 
					
						
							|  |  |  |     m_global_symbol_map.set(description, new_global_symbol); | 
					
						
							|  |  |  |     return new_global_symbol; | 
					
						
							| 
									
										
										
										
											2020-09-21 13:47:33 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | void VM::set_variable(const FlyString& name, Value value, GlobalObject& global_object, bool first_assignment) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_call_stack.size()) { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |         for (auto* scope = current_scope(); scope; scope = scope->parent()) { | 
					
						
							|  |  |  |             auto possible_match = scope->get_from_scope(name); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |             if (possible_match.has_value()) { | 
					
						
							|  |  |  |                 if (!first_assignment && possible_match.value().declaration_kind == DeclarationKind::Const) { | 
					
						
							|  |  |  |                     throw_exception<TypeError>(global_object, ErrorType::InvalidAssignToConst); | 
					
						
							|  |  |  |                     return; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |                 scope->put_to_scope(name, { value, possible_match.value().declaration_kind }); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     global_object.put(move(name), move(value)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value VM::get_variable(const FlyString& name, GlobalObject& global_object) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_call_stack.size()) { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |         for (auto* scope = current_scope(); scope; scope = scope->parent()) { | 
					
						
							|  |  |  |             auto possible_match = scope->get_from_scope(name); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |             if (possible_match.has_value()) | 
					
						
							|  |  |  |                 return possible_match.value().value; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     auto value = global_object.get(name); | 
					
						
							|  |  |  |     if (m_underscore_is_last_value && name == "_" && value.is_empty()) | 
					
						
							|  |  |  |         return m_last_value; | 
					
						
							|  |  |  |     return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Reference VM::get_reference(const FlyString& name) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (m_call_stack.size()) { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |         for (auto* scope = current_scope(); scope; scope = scope->parent()) { | 
					
						
							|  |  |  |             if (scope->is_global_object()) | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |             auto possible_match = scope->get_from_scope(name); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |             if (possible_match.has_value()) | 
					
						
							|  |  |  |                 return { Reference::LocalVariable, name }; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return { Reference::GlobalVariable, name }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value VM::construct(Function& function, Function& new_target, Optional<MarkedValueList> arguments, GlobalObject& global_object) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-07 11:07:17 +01:00
										 |  |  |     CallFrame call_frame; | 
					
						
							|  |  |  |     call_frame.is_strict_mode = function.is_strict_mode(); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-08 12:54:52 +00:00
										 |  |  |     push_call_frame(call_frame, function.global_object()); | 
					
						
							|  |  |  |     if (exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     ArmedScopeGuard call_frame_popper = [&] { | 
					
						
							|  |  |  |         pop_call_frame(); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     call_frame.function_name = function.name(); | 
					
						
							|  |  |  |     call_frame.arguments = function.bound_arguments(); | 
					
						
							|  |  |  |     if (arguments.has_value()) | 
					
						
							|  |  |  |         call_frame.arguments.append(arguments.value().values()); | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     auto* environment = function.create_environment(); | 
					
						
							|  |  |  |     call_frame.scope = environment; | 
					
						
							|  |  |  |     environment->set_new_target(&new_target); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Object* new_object = nullptr; | 
					
						
							|  |  |  |     if (function.constructor_kind() == Function::ConstructorKind::Base) { | 
					
						
							|  |  |  |         new_object = Object::create_empty(global_object); | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |         environment->bind_this_value(global_object, new_object); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |         if (exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-10-13 23:49:19 +02:00
										 |  |  |         auto prototype = new_target.get(names.prototype); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |         if (exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         if (prototype.is_object()) { | 
					
						
							|  |  |  |             new_object->set_prototype(&prototype.as_object()); | 
					
						
							|  |  |  |             if (exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If we are a Derived constructor, |this| has not been constructed before super is called.
 | 
					
						
							|  |  |  |     Value this_value = function.constructor_kind() == Function::ConstructorKind::Base ? new_object : Value {}; | 
					
						
							|  |  |  |     call_frame.this_value = this_value; | 
					
						
							| 
									
										
										
										
											2020-09-27 18:45:21 +02:00
										 |  |  |     auto result = function.construct(new_target); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     this_value = call_frame.scope->get_this_binding(global_object); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     pop_call_frame(); | 
					
						
							|  |  |  |     call_frame_popper.disarm(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If we are constructing an instance of a derived class,
 | 
					
						
							|  |  |  |     // set the prototype on objects created by constructors that return an object (i.e. NativeFunction subclasses).
 | 
					
						
							|  |  |  |     if (function.constructor_kind() == Function::ConstructorKind::Base && new_target.constructor_kind() == Function::ConstructorKind::Derived && result.is_object()) { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |         ASSERT(current_scope()->is_lexical_environment()); | 
					
						
							|  |  |  |         static_cast<LexicalEnvironment*>(current_scope())->replace_this_binding(result); | 
					
						
							| 
									
										
										
										
											2020-10-13 23:49:19 +02:00
										 |  |  |         auto prototype = new_target.get(names.prototype); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |         if (exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         if (prototype.is_object()) { | 
					
						
							|  |  |  |             result.as_object().set_prototype(&prototype.as_object()); | 
					
						
							|  |  |  |             if (exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (result.is_object()) | 
					
						
							|  |  |  |         return result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return this_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void VM::throw_exception(Exception* exception) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-29 16:48:14 +01:00
										 |  |  |     if (should_log_exceptions() && exception->value().is_object() && exception->value().as_object().is_error()) { | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |         auto& error = static_cast<Error&>(exception->value().as_object()); | 
					
						
							| 
									
										
										
										
											2020-10-04 15:44:40 +01:00
										 |  |  |         dbgln("Throwing JavaScript Error: {}, {}", error.name(), error.message()); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for (ssize_t i = m_call_stack.size() - 1; i >= 0; --i) { | 
					
						
							| 
									
										
										
										
											2020-11-21 23:38:50 +00:00
										 |  |  |             auto function_name = m_call_stack[i]->function_name; | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |             if (function_name.is_empty()) | 
					
						
							|  |  |  |                 function_name = "<anonymous>"; | 
					
						
							| 
									
										
										
										
											2020-10-04 15:44:40 +01:00
										 |  |  |             dbgln("  {}", function_name); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-29 16:48:14 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     m_exception = exception; | 
					
						
							|  |  |  |     unwind(ScopeType::Try); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | String VM::join_arguments() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StringBuilder joined_arguments; | 
					
						
							|  |  |  |     for (size_t i = 0; i < argument_count(); ++i) { | 
					
						
							|  |  |  |         joined_arguments.append(argument(i).to_string_without_side_effects().characters()); | 
					
						
							|  |  |  |         if (i != argument_count() - 1) | 
					
						
							|  |  |  |             joined_arguments.append(' '); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return joined_arguments.build(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-29 16:41:28 +02:00
										 |  |  | Value VM::resolve_this_binding(GlobalObject& global_object) const | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     return find_this_scope()->get_this_binding(global_object); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  | const ScopeObject* VM::find_this_scope() const | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     // We will always return because the Global environment will always be reached, which has a |this| binding.
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     for (auto* scope = current_scope(); scope; scope = scope->parent()) { | 
					
						
							|  |  |  |         if (scope->has_this_binding()) | 
					
						
							|  |  |  |             return scope; | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value VM::get_new_target() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     ASSERT(find_this_scope()->is_lexical_environment()); | 
					
						
							|  |  |  |     return static_cast<const LexicalEnvironment*>(find_this_scope())->new_target(); | 
					
						
							| 
									
										
										
										
											2020-09-27 15:18:55 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  | Value VM::call_internal(Function& function, Value this_value, Optional<MarkedValueList> arguments) | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     ASSERT(!exception()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 11:07:17 +01:00
										 |  |  |     CallFrame call_frame; | 
					
						
							|  |  |  |     call_frame.is_strict_mode = function.is_strict_mode(); | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  |     call_frame.function_name = function.name(); | 
					
						
							|  |  |  |     call_frame.this_value = function.bound_this().value_or(this_value); | 
					
						
							|  |  |  |     call_frame.arguments = function.bound_arguments(); | 
					
						
							|  |  |  |     if (arguments.has_value()) | 
					
						
							| 
									
										
										
										
											2020-10-04 22:42:24 +02:00
										 |  |  |         call_frame.arguments.append(move(arguments.release_value().values())); | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     auto* environment = function.create_environment(); | 
					
						
							|  |  |  |     call_frame.scope = environment; | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-28 16:02:27 +01:00
										 |  |  |     ASSERT(environment->this_binding_status() == LexicalEnvironment::ThisBindingStatus::Uninitialized); | 
					
						
							|  |  |  |     environment->bind_this_value(function.global_object(), call_frame.this_value); | 
					
						
							| 
									
										
										
										
											2020-11-08 12:54:52 +00:00
										 |  |  |     if (exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-08 12:54:52 +00:00
										 |  |  |     push_call_frame(call_frame, function.global_object()); | 
					
						
							|  |  |  |     if (exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  |     auto result = function.call(); | 
					
						
							|  |  |  |     pop_call_frame(); | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-04 13:54:44 +02:00
										 |  |  | bool VM::in_strict_mode() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (call_stack().is_empty()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     return call_frame().is_strict_mode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-20 19:24:44 +02:00
										 |  |  | } |