| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Copyright (c) 2020, Andreas Kling <kling@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-04-23 15:43:10 +01:00
										 |  |  |  * Copyright (c) 2020, Linus Groh <mail@linusgroh.de> | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |  * 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-03-19 17:39:13 +01:00
										 |  |  | #include <AK/Function.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-12 19:22:13 +08:00
										 |  |  | #include <AK/HashMap.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-05 00:24:32 +02:00
										 |  |  | #include <AK/ScopeGuard.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-12 19:22:13 +08:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | #include <LibJS/AST.h>
 | 
					
						
							|  |  |  | #include <LibJS/Interpreter.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:57 +01:00
										 |  |  | #include <LibJS/Runtime/Array.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-08 11:05:38 +02:00
										 |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-19 17:24:56 +02:00
										 |  |  | #include <LibJS/Runtime/MarkedValueList.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-01 18:31:24 +01:00
										 |  |  | #include <LibJS/Runtime/NativeFunction.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Runtime/PrimitiveString.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | #include <LibJS/Runtime/Reference.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-16 14:20:30 +01:00
										 |  |  | #include <LibJS/Runtime/ScriptFunction.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/Value.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value ScopeNode::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return interpreter.run(*this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value FunctionDeclaration::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-17 19:59:32 +02:00
										 |  |  |     auto* function = ScriptFunction::create(interpreter.global_object(), name(), body(), parameters(), interpreter.current_environment()); | 
					
						
							| 
									
										
										
										
											2020-03-19 11:12:08 +01:00
										 |  |  |     interpreter.set_variable(name(), function); | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 11:12:08 +01:00
										 |  |  | Value FunctionExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-17 19:59:32 +02:00
										 |  |  |     return ScriptFunction::create(interpreter.global_object(), name(), body(), parameters(), interpreter.current_environment()); | 
					
						
							| 
									
										
										
										
											2020-03-19 11:12:08 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | Value ExpressionStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_expression->execute(interpreter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 18:51:27 +02:00
										 |  |  | CallExpression::ThisAndCallee CallExpression::compute_this_and_callee(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (is_new_expression()) { | 
					
						
							|  |  |  |         // Computing |this| is irrelevant for "new" expression.
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         return { js_undefined(), m_callee->execute(interpreter) }; | 
					
						
							| 
									
										
										
										
											2020-04-01 18:51:27 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_callee->is_member_expression()) { | 
					
						
							|  |  |  |         auto& member_expression = static_cast<const MemberExpression&>(*m_callee); | 
					
						
							|  |  |  |         auto object_value = member_expression.object().execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         auto* this_value = object_value.to_object(interpreter.heap()); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         auto callee = this_value->get(member_expression.computed_property_name(interpreter)).value_or(js_undefined()); | 
					
						
							| 
									
										
										
										
											2020-04-01 18:51:27 +02:00
										 |  |  |         return { this_value, callee }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return { &interpreter.global_object(), m_callee->execute(interpreter) }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | Value CallExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-01 18:51:27 +02:00
										 |  |  |     auto [this_value, callee] = compute_this_and_callee(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     ASSERT(!callee.is_empty()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 01:12:51 +01:00
										 |  |  |     if (!callee.is_object() | 
					
						
							|  |  |  |         || !callee.as_object().is_function() | 
					
						
							| 
									
										
										
										
											2020-04-19 14:50:27 +02:00
										 |  |  |         || (is_new_expression() && (callee.as_object().is_native_function() && !static_cast<NativeFunction&>(callee.as_object()).has_constructor()))) { | 
					
						
							| 
									
										
										
										
											2020-04-19 01:12:51 +01:00
										 |  |  |         String error_message; | 
					
						
							|  |  |  |         auto call_type = is_new_expression() ? "constructor" : "function"; | 
					
						
							|  |  |  |         if (m_callee->is_identifier() || m_callee->is_member_expression()) { | 
					
						
							|  |  |  |             String expression_string; | 
					
						
							|  |  |  |             if (m_callee->is_identifier()) | 
					
						
							|  |  |  |                 expression_string = static_cast<const Identifier&>(*m_callee).string(); | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 expression_string = static_cast<const MemberExpression&>(*m_callee).to_string_approximation(); | 
					
						
							|  |  |  |             error_message = String::format("%s is not a %s (evaluated from '%s')", callee.to_string().characters(), call_type, expression_string.characters()); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             error_message = String::format("%s is not a %s", callee.to_string().characters(), call_type); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return interpreter.throw_exception<TypeError>(error_message); | 
					
						
							| 
									
										
										
										
											2020-04-01 18:31:24 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |     auto& function = static_cast<Function&>(callee.as_object()); | 
					
						
							| 
									
										
										
										
											2020-03-12 19:53:31 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 17:24:56 +02:00
										 |  |  |     MarkedValueList arguments(interpreter.heap()); | 
					
						
							| 
									
										
										
										
											2020-04-19 15:03:02 -05:00
										 |  |  |     arguments.values().append(function.bound_arguments()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |     for (size_t i = 0; i < m_arguments.size(); ++i) { | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |         auto value = m_arguments[i].execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-01 22:28:48 -05:00
										 |  |  |         arguments.append(value); | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-12 19:22:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-01 22:28:48 -05:00
										 |  |  |     auto& call_frame = interpreter.push_call_frame(); | 
					
						
							| 
									
										
										
										
											2020-04-11 12:56:20 +01:00
										 |  |  |     call_frame.function_name = function.name(); | 
					
						
							| 
									
										
										
										
											2020-04-19 17:24:56 +02:00
										 |  |  |     call_frame.arguments = arguments.values(); | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |     call_frame.environment = function.create_environment(); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:28:48 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-28 16:33:52 +01:00
										 |  |  |     Object* new_object = nullptr; | 
					
						
							| 
									
										
										
										
											2020-04-01 18:31:24 +01:00
										 |  |  |     Value result; | 
					
						
							| 
									
										
										
										
											2020-03-28 16:33:52 +01:00
										 |  |  |     if (is_new_expression()) { | 
					
						
							| 
									
										
										
										
											2020-04-18 10:27:57 +02:00
										 |  |  |         new_object = Object::create_empty(interpreter, interpreter.global_object()); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         auto prototype = function.get("prototype"); | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |         if (prototype.is_object()) | 
					
						
							|  |  |  |             new_object->set_prototype(&prototype.as_object()); | 
					
						
							| 
									
										
										
										
											2020-03-28 16:33:52 +01:00
										 |  |  |         call_frame.this_value = new_object; | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         result = function.construct(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-28 16:33:52 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-04-19 14:51:17 -05:00
										 |  |  |         call_frame.this_value = function.bound_this().value_or(this_value); | 
					
						
							| 
									
										
										
										
											2020-04-01 22:18:47 +02:00
										 |  |  |         result = function.call(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-15 15:01:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 21:58:22 +02:00
										 |  |  |     interpreter.pop_call_frame(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-28 16:33:52 +01:00
										 |  |  |     if (is_new_expression()) { | 
					
						
							|  |  |  |         if (result.is_object()) | 
					
						
							|  |  |  |             return result; | 
					
						
							|  |  |  |         return new_object; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-15 15:01:10 +01:00
										 |  |  |     return result; | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value ReturnStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     auto value = argument() ? argument()->execute(interpreter) : js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-03-23 19:19:03 +01:00
										 |  |  |     interpreter.unwind(ScopeType::Function); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     return value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  | Value IfStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto predicate_result = m_predicate->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-10 08:46:20 +01:00
										 |  |  |     if (predicate_result.to_boolean()) | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  |         return interpreter.run(*m_consequent); | 
					
						
							| 
									
										
										
										
											2020-03-21 18:40:17 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_alternate) | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  |         return interpreter.run(*m_alternate); | 
					
						
							| 
									
										
										
										
											2020-03-21 18:40:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 03:22:21 +08:00
										 |  |  | Value WhileStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Value last_value = js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-04-04 21:21:19 +02:00
										 |  |  |     while (m_test->execute(interpreter).to_boolean()) { | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-03-09 03:22:21 +08:00
										 |  |  |         last_value = interpreter.run(*m_body); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-03-09 03:22:21 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return last_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-04 21:29:23 +02:00
										 |  |  | Value DoWhileStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Value last_value = js_undefined(); | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         last_value = interpreter.run(*m_body); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |     } while (m_test->execute(interpreter).to_boolean()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return last_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  | Value ForStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |     RefPtr<BlockStatement> wrapper; | 
					
						
							| 
									
										
										
										
											2020-03-14 13:56:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-08 11:59:18 +02:00
										 |  |  |     if (m_init && m_init->is_variable_declaration() && static_cast<const VariableDeclaration*>(m_init.ptr())->declaration_kind() != DeclarationKind::Var) { | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |         wrapper = create_ast_node<BlockStatement>(); | 
					
						
							| 
									
										
										
										
											2020-03-14 13:56:49 +02:00
										 |  |  |         interpreter.enter_scope(*wrapper, {}, ScopeType::Block); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 00:24:32 +02:00
										 |  |  |     auto wrapper_cleanup = ScopeGuard([&] { | 
					
						
							|  |  |  |         if (wrapper) | 
					
						
							|  |  |  |             interpreter.exit_scope(*wrapper); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |     Value last_value = js_undefined(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (m_init) { | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |         m_init->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_test) { | 
					
						
							|  |  |  |         while (m_test->execute(interpreter).to_boolean()) { | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |             last_value = interpreter.run(*m_body); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  |             if (interpreter.should_unwind()) { | 
					
						
							|  |  |  |                 if (interpreter.should_unwind_until(ScopeType::Continuable)) { | 
					
						
							|  |  |  |                     interpreter.stop_unwind(); | 
					
						
							|  |  |  |                 } else if (interpreter.should_unwind_until(ScopeType::Breakable)) { | 
					
						
							|  |  |  |                     interpreter.stop_unwind(); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |                     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |             if (m_update) { | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |                 m_update->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |                 if (interpreter.exception()) | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         while (true) { | 
					
						
							|  |  |  |             last_value = interpreter.run(*m_body); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  |             if (interpreter.should_unwind()) { | 
					
						
							|  |  |  |                 if (interpreter.should_unwind_until(ScopeType::Continuable)) { | 
					
						
							|  |  |  |                     interpreter.stop_unwind(); | 
					
						
							|  |  |  |                 } else if (interpreter.should_unwind_until(ScopeType::Breakable)) { | 
					
						
							|  |  |  |                     interpreter.stop_unwind(); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |                     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |             if (m_update) { | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |                 m_update->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |                 if (interpreter.exception()) | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return last_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | Value BinaryExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto lhs_result = m_lhs->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     auto rhs_result = m_rhs->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (m_op) { | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Addition: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return add(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Subtraction: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return sub(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Multiplication: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return mul(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Division: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return div(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-04 21:17:34 +02:00
										 |  |  |     case BinaryOp::Modulo: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return mod(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-05 13:40:00 +01:00
										 |  |  |     case BinaryOp::Exponentiation: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return exp(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-08 07:53:02 +02:00
										 |  |  |     case BinaryOp::TypedEquals: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return typed_eq(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |     case BinaryOp::TypedInequals: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return Value(!typed_eq(interpreter, lhs_result, rhs_result).as_bool()); | 
					
						
							| 
									
										
										
										
											2020-03-16 00:23:38 +02:00
										 |  |  |     case BinaryOp::AbstractEquals: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return eq(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-16 00:23:38 +02:00
										 |  |  |     case BinaryOp::AbstractInequals: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return Value(!eq(interpreter, lhs_result, rhs_result).as_bool()); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::GreaterThan: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return greater_than(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:07:08 +11:00
										 |  |  |     case BinaryOp::GreaterThanEquals: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return greater_than_equals(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::LessThan: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return less_than(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:07:08 +11:00
										 |  |  |     case BinaryOp::LessThanEquals: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return less_than_equals(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::BitwiseAnd: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return bitwise_and(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::BitwiseOr: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return bitwise_or(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::BitwiseXor: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return bitwise_xor(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::LeftShift: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return left_shift(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::RightShift: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return right_shift(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-23 15:43:10 +01:00
										 |  |  |     case BinaryOp::UnsignedRightShift: | 
					
						
							|  |  |  |         return unsigned_right_shift(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-23 16:06:01 +01:00
										 |  |  |     case BinaryOp::In: | 
					
						
							|  |  |  |         return in(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-28 16:56:54 +01:00
										 |  |  |     case BinaryOp::InstanceOf: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return instance_of(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-08 07:53:02 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-03-08 07:55:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | Value LogicalExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-03 14:33:28 +01:00
										 |  |  |     auto lhs_result = m_lhs->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-04-03 19:11:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 07:55:44 +02:00
										 |  |  |     switch (m_op) { | 
					
						
							|  |  |  |     case LogicalOp::And: | 
					
						
							| 
									
										
										
										
											2020-04-03 19:11:31 +02:00
										 |  |  |         if (lhs_result.to_boolean()) { | 
					
						
							|  |  |  |             auto rhs_result = m_rhs->execute(interpreter); | 
					
						
							|  |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-18 00:49:11 +01:00
										 |  |  |             return rhs_result; | 
					
						
							| 
									
										
										
										
											2020-04-03 19:11:31 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-18 00:49:11 +01:00
										 |  |  |         return lhs_result; | 
					
						
							|  |  |  |     case LogicalOp::Or: { | 
					
						
							| 
									
										
										
										
											2020-04-03 14:33:28 +01:00
										 |  |  |         if (lhs_result.to_boolean()) | 
					
						
							| 
									
										
										
										
											2020-04-18 00:49:11 +01:00
										 |  |  |             return lhs_result; | 
					
						
							| 
									
										
										
										
											2020-04-03 19:11:31 +02:00
										 |  |  |         auto rhs_result = m_rhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-18 00:49:11 +01:00
										 |  |  |         return rhs_result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     case LogicalOp::NullishCoalescing: | 
					
						
							|  |  |  |         if (lhs_result.is_null() || lhs_result.is_undefined()) { | 
					
						
							|  |  |  |             auto rhs_result = m_rhs->execute(interpreter); | 
					
						
							|  |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							|  |  |  |             return rhs_result; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return lhs_result; | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | Reference Expression::to_reference(Interpreter&) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  | Reference Identifier::to_reference(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return interpreter.get_reference(string()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  | Reference MemberExpression::to_reference(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto object_value = m_object->execute(interpreter); | 
					
						
							|  |  |  |     if (object_value.is_empty()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     auto* object = object_value.to_object(interpreter.heap()); | 
					
						
							|  |  |  |     if (!object) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     auto property_name = computed_property_name(interpreter); | 
					
						
							|  |  |  |     if (!property_name.is_valid()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return { object, property_name }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  | Value UnaryExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-26 13:53:40 +02:00
										 |  |  |     if (m_op == UnaryOp::Delete) { | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |         auto reference = m_lhs->to_reference(interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-26 13:53:40 +02:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |         if (reference.is_unresolvable()) | 
					
						
							|  |  |  |             return Value(true); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:37:27 +02:00
										 |  |  |         // FIXME: Support deleting locals
 | 
					
						
							|  |  |  |         ASSERT(!reference.is_local_variable()); | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |         auto* base_object = reference.base().to_object(interpreter.heap()); | 
					
						
							|  |  |  |         if (!base_object) | 
					
						
							| 
									
										
										
										
											2020-04-26 13:53:40 +02:00
										 |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-27 12:10:16 +02:00
										 |  |  |         return base_object->delete_property(reference.name()); | 
					
						
							| 
									
										
										
										
											2020-04-26 13:53:40 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |     auto lhs_result = m_lhs->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |     switch (m_op) { | 
					
						
							| 
									
										
										
										
											2020-03-14 20:43:35 +02:00
										 |  |  |     case UnaryOp::BitwiseNot: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return bitwise_not(interpreter, lhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-09 19:04:44 +02:00
										 |  |  |     case UnaryOp::Not: | 
					
						
							| 
									
										
										
										
											2020-03-10 11:08:37 +01:00
										 |  |  |         return Value(!lhs_result.to_boolean()); | 
					
						
							| 
									
										
										
										
											2020-04-02 17:58:39 +01:00
										 |  |  |     case UnaryOp::Plus: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return unary_plus(interpreter, lhs_result); | 
					
						
							| 
									
										
										
										
											2020-04-02 17:58:39 +01:00
										 |  |  |     case UnaryOp::Minus: | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         return unary_minus(interpreter, lhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |     case UnaryOp::Typeof: | 
					
						
							|  |  |  |         switch (lhs_result.type()) { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         case Value::Type::Empty: | 
					
						
							|  |  |  |             ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         case Value::Type::Undefined: | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |             return js_string(interpreter, "undefined"); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         case Value::Type::Null: | 
					
						
							|  |  |  |             // yes, this is on purpose. yes, this is how javascript works.
 | 
					
						
							|  |  |  |             // yes, it's silly.
 | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |             return js_string(interpreter, "object"); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         case Value::Type::Number: | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |             return js_string(interpreter, "number"); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         case Value::Type::String: | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |             return js_string(interpreter, "string"); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         case Value::Type::Object: | 
					
						
							| 
									
										
										
										
											2020-04-04 14:34:31 +01:00
										 |  |  |             if (lhs_result.as_object().is_function()) | 
					
						
							|  |  |  |                 return js_string(interpreter, "function"); | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |             return js_string(interpreter, "object"); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         case Value::Type::Boolean: | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |             return js_string(interpreter, "boolean"); | 
					
						
							| 
									
										
										
										
											2020-04-15 17:55:03 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             ASSERT_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-15 17:55:03 +01:00
										 |  |  |     case UnaryOp::Void: | 
					
						
							|  |  |  |         return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-04-26 13:53:40 +02:00
										 |  |  |     case UnaryOp::Delete: | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | static void print_indent(int indent) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (int i = 0; i < indent * 2; ++i) | 
					
						
							|  |  |  |         putchar(' '); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ASTNode::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("%s\n", class_name()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ScopeNode::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:42:54 +02:00
										 |  |  |     if (!m_variables.is_empty()) { | 
					
						
							|  |  |  |         print_indent(indent + 1); | 
					
						
							|  |  |  |         printf("(Variables)\n"); | 
					
						
							|  |  |  |         for (auto& variable : m_variables) | 
					
						
							|  |  |  |             variable.dump(indent + 2); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!m_children.is_empty()) { | 
					
						
							|  |  |  |         print_indent(indent + 1); | 
					
						
							|  |  |  |         printf("(Children)\n"); | 
					
						
							|  |  |  |         for (auto& child : children()) | 
					
						
							|  |  |  |             child.dump(indent + 2); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BinaryExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-07 23:16:34 +01:00
										 |  |  |     const char* op_string = nullptr; | 
					
						
							|  |  |  |     switch (m_op) { | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Addition: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:16:34 +01:00
										 |  |  |         op_string = "+"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Subtraction: | 
					
						
							| 
									
										
										
										
											2020-03-07 23:16:34 +01:00
										 |  |  |         op_string = "-"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Multiplication: | 
					
						
							| 
									
										
										
										
											2020-03-12 23:04:52 +11:00
										 |  |  |         op_string = "*"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-05 12:56:53 +01:00
										 |  |  |     case BinaryOp::Division: | 
					
						
							| 
									
										
										
										
											2020-03-12 23:04:52 +11:00
										 |  |  |         op_string = "/"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-04 21:17:34 +02:00
										 |  |  |     case BinaryOp::Modulo: | 
					
						
							|  |  |  |         op_string = "%"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-05 13:40:00 +01:00
										 |  |  |     case BinaryOp::Exponentiation: | 
					
						
							|  |  |  |         op_string = "**"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-08 07:53:02 +02:00
										 |  |  |     case BinaryOp::TypedEquals: | 
					
						
							|  |  |  |         op_string = "==="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |     case BinaryOp::TypedInequals: | 
					
						
							|  |  |  |         op_string = "!=="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-16 00:23:38 +02:00
										 |  |  |     case BinaryOp::AbstractEquals: | 
					
						
							|  |  |  |         op_string = "=="; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case BinaryOp::AbstractInequals: | 
					
						
							|  |  |  |         op_string = "!="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::GreaterThan: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = ">"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 23:07:08 +11:00
										 |  |  |     case BinaryOp::GreaterThanEquals: | 
					
						
							|  |  |  |         op_string = ">="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::LessThan: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = "<"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 23:07:08 +11:00
										 |  |  |     case BinaryOp::LessThanEquals: | 
					
						
							|  |  |  |         op_string = "<="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::BitwiseAnd: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = "&"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::BitwiseOr: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = "|"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::BitwiseXor: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = "^"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::LeftShift: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = "<<"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-10 11:35:05 +01:00
										 |  |  |     case BinaryOp::RightShift: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = ">>"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 15:43:10 +01:00
										 |  |  |     case BinaryOp::UnsignedRightShift: | 
					
						
							|  |  |  |         op_string = ">>>"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 16:06:01 +01:00
										 |  |  |     case BinaryOp::In: | 
					
						
							|  |  |  |         op_string = "in"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-28 16:56:54 +01:00
										 |  |  |     case BinaryOp::InstanceOf: | 
					
						
							|  |  |  |         op_string = "instanceof"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-08 07:53:02 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("%s\n", class_name()); | 
					
						
							|  |  |  |     m_lhs->dump(indent + 1); | 
					
						
							|  |  |  |     print_indent(indent + 1); | 
					
						
							|  |  |  |     printf("%s\n", op_string); | 
					
						
							|  |  |  |     m_rhs->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 07:55:44 +02:00
										 |  |  | void LogicalExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char* op_string = nullptr; | 
					
						
							|  |  |  |     switch (m_op) { | 
					
						
							|  |  |  |     case LogicalOp::And: | 
					
						
							|  |  |  |         op_string = "&&"; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case LogicalOp::Or: | 
					
						
							|  |  |  |         op_string = "||"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-18 00:49:11 +01:00
										 |  |  |     case LogicalOp::NullishCoalescing: | 
					
						
							|  |  |  |         op_string = "??"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-07 23:16:34 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("%s\n", class_name()); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     m_lhs->dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-07 23:16:34 +01:00
										 |  |  |     print_indent(indent + 1); | 
					
						
							|  |  |  |     printf("%s\n", op_string); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     m_rhs->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  | void UnaryExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char* op_string = nullptr; | 
					
						
							|  |  |  |     switch (m_op) { | 
					
						
							| 
									
										
										
										
											2020-03-14 20:43:35 +02:00
										 |  |  |     case UnaryOp::BitwiseNot: | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |         op_string = "~"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-09 19:04:44 +02:00
										 |  |  |     case UnaryOp::Not: | 
					
						
							|  |  |  |         op_string = "!"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-02 17:58:39 +01:00
										 |  |  |     case UnaryOp::Plus: | 
					
						
							|  |  |  |         op_string = "+"; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case UnaryOp::Minus: | 
					
						
							|  |  |  |         op_string = "-"; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-18 06:33:32 +11:00
										 |  |  |     case UnaryOp::Typeof: | 
					
						
							|  |  |  |         op_string = "typeof "; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-15 17:55:03 +01:00
										 |  |  |     case UnaryOp::Void: | 
					
						
							|  |  |  |         op_string = "void "; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-26 13:53:40 +02:00
										 |  |  |     case UnaryOp::Delete: | 
					
						
							|  |  |  |         op_string = "delete "; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-08 23:27:18 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("%s\n", class_name()); | 
					
						
							|  |  |  |     print_indent(indent + 1); | 
					
						
							|  |  |  |     printf("%s\n", op_string); | 
					
						
							|  |  |  |     m_lhs->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | void CallExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("CallExpression %s\n", is_new_expression() ? "[new]" : ""); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:02:41 +01:00
										 |  |  |     m_callee->dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-12 19:34:59 +01:00
										 |  |  |     for (auto& argument : m_arguments) | 
					
						
							|  |  |  |         argument.dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 12:19:11 +01:00
										 |  |  | void StringLiteral::dump(int indent) const | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							| 
									
										
										
										
											2020-03-12 12:19:11 +01:00
										 |  |  |     printf("StringLiteral \"%s\"\n", m_value.characters()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NumericLiteral::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							| 
									
										
										
										
											2020-03-12 19:36:24 +01:00
										 |  |  |     printf("NumericLiteral %g\n", m_value); | 
					
						
							| 
									
										
										
										
											2020-03-12 12:19:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BooleanLiteral::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("BooleanLiteral %s\n", m_value ? "true" : "false"); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-15 23:32:34 +02:00
										 |  |  | void NullLiteral::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("null\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 11:12:08 +01:00
										 |  |  | void FunctionNode::dump(int indent, const char* class_name) const | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-12 19:22:13 +08:00
										 |  |  |     StringBuilder parameters_builder; | 
					
						
							| 
									
										
										
										
											2020-03-20 14:40:08 +01:00
										 |  |  |     parameters_builder.join(',', parameters()); | 
					
						
							| 
									
										
										
										
											2020-03-12 19:22:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  |     print_indent(indent); | 
					
						
							| 
									
										
										
										
											2020-03-19 11:12:08 +01:00
										 |  |  |     printf("%s '%s(%s)'\n", class_name, name().characters(), parameters_builder.build().characters()); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:42:54 +02:00
										 |  |  |     if (!m_variables.is_empty()) { | 
					
						
							|  |  |  |         print_indent(indent + 1); | 
					
						
							|  |  |  |         printf("(Variables)\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (auto& variable : m_variables) | 
					
						
							|  |  |  |         variable.dump(indent + 2); | 
					
						
							|  |  |  |     print_indent(indent + 1); | 
					
						
							|  |  |  |     printf("(Body)\n"); | 
					
						
							|  |  |  |     body().dump(indent + 2); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 11:12:08 +01:00
										 |  |  | void FunctionDeclaration::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FunctionNode::dump(indent, class_name()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void FunctionExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     FunctionNode::dump(indent, class_name()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | void ReturnStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     if (argument()) | 
					
						
							|  |  |  |         argument()->dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  | void IfStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("If\n"); | 
					
						
							|  |  |  |     predicate().dump(indent + 1); | 
					
						
							|  |  |  |     consequent().dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-21 18:40:17 +01:00
										 |  |  |     if (alternate()) { | 
					
						
							|  |  |  |         print_indent(indent); | 
					
						
							|  |  |  |         printf("Else\n"); | 
					
						
							|  |  |  |         alternate()->dump(indent + 1); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-08 07:58:58 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 03:22:21 +08:00
										 |  |  | void WhileStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("While\n"); | 
					
						
							| 
									
										
										
										
											2020-04-04 21:21:19 +02:00
										 |  |  |     test().dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-09 03:22:21 +08:00
										 |  |  |     body().dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-04 21:29:23 +02:00
										 |  |  | void DoWhileStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("DoWhile\n"); | 
					
						
							|  |  |  |     test().dump(indent + 1); | 
					
						
							|  |  |  |     body().dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 23:12:12 +11:00
										 |  |  | void ForStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("For\n"); | 
					
						
							|  |  |  |     if (init()) | 
					
						
							|  |  |  |         init()->dump(indent + 1); | 
					
						
							|  |  |  |     if (test()) | 
					
						
							|  |  |  |         test()->dump(indent + 1); | 
					
						
							|  |  |  |     if (update()) | 
					
						
							|  |  |  |         update()->dump(indent + 1); | 
					
						
							|  |  |  |     body().dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | Value Identifier::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |     auto value = interpreter.get_variable(string()); | 
					
						
							|  |  |  |     if (value.is_empty()) | 
					
						
							| 
									
										
										
										
											2020-04-10 12:48:31 +02:00
										 |  |  |         return interpreter.throw_exception<ReferenceError>(String::format("'%s' not known", string().characters())); | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |     return value; | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Identifier::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("Identifier \"%s\"\n", m_string.characters()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 23:05:37 -07:00
										 |  |  | void SpreadExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     m_target->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value SpreadExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return m_target->execute(interpreter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 00:42:14 +02:00
										 |  |  | Value ThisExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return interpreter.this_value(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ThisExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | Value AssignmentExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto rhs_result = m_rhs->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     Value lhs_result; | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |     switch (m_op) { | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::Assignment: | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::AdditionAssignment: | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         rhs_result = add(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::SubtractionAssignment: | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         rhs_result = sub(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::MultiplicationAssignment: | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         rhs_result = mul(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::DivisionAssignment: | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |         rhs_result = div(interpreter, lhs_result, rhs_result); | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 13:36:14 +01:00
										 |  |  |     case AssignmentOp::LeftShiftAssignment: | 
					
						
							|  |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         rhs_result = left_shift(interpreter, lhs_result, rhs_result); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 13:45:19 +01:00
										 |  |  |     case AssignmentOp::RightShiftAssignment: | 
					
						
							|  |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         rhs_result = right_shift(interpreter, lhs_result, rhs_result); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 15:43:10 +01:00
										 |  |  |     case AssignmentOp::UnsignedRightShiftAssignment: | 
					
						
							|  |  |  |         lhs_result = m_lhs->execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         rhs_result = unsigned_right_shift(interpreter, lhs_result, rhs_result); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-04-05 19:21:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (m_lhs->is_identifier()) { | 
					
						
							|  |  |  |         auto name = static_cast<const Identifier&>(*m_lhs).string(); | 
					
						
							|  |  |  |         interpreter.set_variable(name, rhs_result); | 
					
						
							|  |  |  |     } else if (m_lhs->is_member_expression()) { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |         auto object_value = static_cast<const MemberExpression&>(*m_lhs).object().execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         if (auto* object = object_value.to_object(interpreter.heap())) { | 
					
						
							| 
									
										
										
										
											2020-04-05 19:21:36 +02:00
										 |  |  |             auto property_name = static_cast<const MemberExpression&>(*m_lhs).computed_property_name(interpreter); | 
					
						
							|  |  |  |             object->put(property_name, rhs_result); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2020-04-10 12:48:31 +02:00
										 |  |  |         return interpreter.throw_exception<ReferenceError>("Invalid left-hand side in assignment"); | 
					
						
							| 
									
										
										
										
											2020-04-05 19:21:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |     return rhs_result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  | Value UpdateExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASSERT(m_argument->is_identifier()); | 
					
						
							|  |  |  |     auto name = static_cast<const Identifier&>(*m_argument).string(); | 
					
						
							| 
									
										
										
										
											2020-04-21 23:27:11 +01:00
										 |  |  |     auto old_value = m_argument->execute(interpreter); | 
					
						
							|  |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     old_value = old_value.to_number(); | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 20:44:57 +02:00
										 |  |  |     int op_result = 0; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  |     switch (m_op) { | 
					
						
							|  |  |  |     case UpdateOp::Increment: | 
					
						
							| 
									
										
										
										
											2020-03-14 20:44:57 +02:00
										 |  |  |         op_result = 1; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     case UpdateOp::Decrement: | 
					
						
							| 
									
										
										
										
											2020-03-14 20:44:57 +02:00
										 |  |  |         op_result = -1; | 
					
						
							| 
									
										
										
										
											2020-03-12 14:24:34 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-21 23:27:11 +01:00
										 |  |  |     default: | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-21 23:27:11 +01:00
										 |  |  |     auto new_value = Value(old_value.as_double() + op_result); | 
					
						
							|  |  |  |     interpreter.set_variable(name, new_value); | 
					
						
							|  |  |  |     return m_prefixed ? new_value : old_value; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | void AssignmentExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char* op_string = nullptr; | 
					
						
							|  |  |  |     switch (m_op) { | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::Assignment: | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |         op_string = "="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::AdditionAssignment: | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         op_string = "+="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::SubtractionAssignment: | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         op_string = "-="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::MultiplicationAssignment: | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         op_string = "*="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-12 13:54:56 +01:00
										 |  |  |     case AssignmentOp::DivisionAssignment: | 
					
						
							| 
									
										
										
										
											2020-03-12 23:09:15 +11:00
										 |  |  |         op_string = "/="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 13:36:14 +01:00
										 |  |  |     case AssignmentOp::LeftShiftAssignment: | 
					
						
							|  |  |  |         op_string = "<<="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 13:45:19 +01:00
										 |  |  |     case AssignmentOp::RightShiftAssignment: | 
					
						
							|  |  |  |         op_string = ">>="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-23 15:43:10 +01:00
										 |  |  |     case AssignmentOp::UnsignedRightShiftAssignment: | 
					
						
							|  |  |  |         op_string = ">>>="; | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     print_indent(indent + 1); | 
					
						
							|  |  |  |     printf("%s\n", op_string); | 
					
						
							|  |  |  |     m_lhs->dump(indent + 1); | 
					
						
							|  |  |  |     m_rhs->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  | void UpdateExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char* op_string = nullptr; | 
					
						
							|  |  |  |     switch (m_op) { | 
					
						
							|  |  |  |     case UpdateOp::Increment: | 
					
						
							|  |  |  |         op_string = "++"; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     case UpdateOp::Decrement: | 
					
						
							|  |  |  |         op_string = "--"; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     print_indent(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-14 20:44:57 +02:00
										 |  |  |     if (m_prefixed) | 
					
						
							|  |  |  |         printf("%s\n", op_string); | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  |     m_argument->dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-14 20:44:57 +02:00
										 |  |  |     if (!m_prefixed) { | 
					
						
							|  |  |  |         print_indent(indent + 1); | 
					
						
							|  |  |  |         printf("%s\n", op_string); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-12 13:45:45 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | Value VariableDeclaration::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-04 21:46:25 +02:00
										 |  |  |     for (auto& declarator : m_declarations) { | 
					
						
							|  |  |  |         if (auto* init = declarator.init()) { | 
					
						
							|  |  |  |             auto initalizer_result = init->execute(interpreter); | 
					
						
							|  |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							|  |  |  |             interpreter.set_variable(declarator.id().string(), initalizer_result, true); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  | Value VariableDeclarator::execute(Interpreter&) const | 
					
						
							| 
									
										
										
										
											2020-04-04 21:46:25 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     // NOTE: This node is handled by VariableDeclaration.
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | void VariableDeclaration::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-08 11:59:18 +02:00
										 |  |  |     const char* declaration_kind_string = nullptr; | 
					
						
							|  |  |  |     switch (m_declaration_kind) { | 
					
						
							|  |  |  |     case DeclarationKind::Let: | 
					
						
							|  |  |  |         declaration_kind_string = "Let"; | 
					
						
							| 
									
										
										
										
											2020-03-11 21:09:20 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-08 11:59:18 +02:00
										 |  |  |     case DeclarationKind::Var: | 
					
						
							|  |  |  |         declaration_kind_string = "Var"; | 
					
						
							| 
									
										
										
										
											2020-03-12 14:24:34 +02:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2020-04-08 11:59:18 +02:00
										 |  |  |     case DeclarationKind::Const: | 
					
						
							|  |  |  |         declaration_kind_string = "Const"; | 
					
						
							| 
									
										
										
										
											2020-03-11 21:09:20 +02:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  |     ASTNode::dump(indent); | 
					
						
							| 
									
										
										
										
											2020-03-11 21:09:20 +02:00
										 |  |  |     print_indent(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-04-08 11:59:18 +02:00
										 |  |  |     printf("%s\n", declaration_kind_string); | 
					
						
							| 
									
										
										
										
											2020-04-04 21:46:25 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (auto& declarator : m_declarations) | 
					
						
							|  |  |  |         declarator.dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void VariableDeclarator::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     m_id->dump(indent + 1); | 
					
						
							|  |  |  |     if (m_init) | 
					
						
							|  |  |  |         m_init->dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-09 21:13:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 19:37:53 +01:00
										 |  |  | void ObjectProperty::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     m_key->dump(indent + 1); | 
					
						
							|  |  |  |     m_value->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:28:31 +01:00
										 |  |  | void ObjectExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							| 
									
										
										
										
											2020-04-23 19:37:53 +01:00
										 |  |  |     for (auto& property : m_properties) { | 
					
						
							|  |  |  |         property.dump(indent + 1); | 
					
						
							| 
									
										
										
										
											2020-03-21 02:29:00 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-09 21:28:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | void ExpressionStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     m_expression->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 19:37:53 +01:00
										 |  |  | Value ObjectProperty::execute(Interpreter&) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     // NOTE: ObjectProperty execution is handled by ObjectExpression.
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 21:28:31 +01:00
										 |  |  | Value ObjectExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-18 10:27:57 +02:00
										 |  |  |     auto* object = Object::create_empty(interpreter, interpreter.global_object()); | 
					
						
							| 
									
										
										
										
											2020-04-23 19:37:53 +01:00
										 |  |  |     for (auto& property : m_properties) { | 
					
						
							| 
									
										
										
										
											2020-04-23 23:00:58 +01:00
										 |  |  |         auto key_result = property.key().execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-23 19:37:53 +01:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |         auto key = key_result.to_string(); | 
					
						
							| 
									
										
										
										
											2020-04-23 23:00:58 +01:00
										 |  |  |         auto value = property.value().execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							| 
									
										
										
										
											2020-04-23 19:37:53 +01:00
										 |  |  |         object->put(key, value); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-21 02:29:00 +02:00
										 |  |  |     return object; | 
					
						
							| 
									
										
										
										
											2020-03-09 21:28:31 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 18:58:19 +01:00
										 |  |  | void MemberExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-20 20:51:03 +01:00
										 |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("%s (computed=%s)\n", class_name(), is_computed() ? "true" : "false"); | 
					
						
							| 
									
										
										
										
											2020-03-11 18:58:19 +01:00
										 |  |  |     m_object->dump(indent + 1); | 
					
						
							|  |  |  |     m_property->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 17:08:23 +02:00
										 |  |  | PropertyName MemberExpression::computed_property_name(Interpreter& interpreter) const | 
					
						
							| 
									
										
										
										
											2020-03-20 20:51:03 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!is_computed()) { | 
					
						
							|  |  |  |         ASSERT(m_property->is_identifier()); | 
					
						
							| 
									
										
										
										
											2020-04-06 17:08:23 +02:00
										 |  |  |         return PropertyName(static_cast<const Identifier&>(*m_property).string()); | 
					
						
							| 
									
										
										
										
											2020-03-20 20:51:03 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-06 17:08:23 +02:00
										 |  |  |     auto index = m_property->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     ASSERT(!index.is_empty()); | 
					
						
							| 
									
										
										
										
											2020-04-06 17:08:23 +02:00
										 |  |  |     // FIXME: What about non-integer numbers tho.
 | 
					
						
							| 
									
										
										
										
											2020-04-06 21:21:11 +02:00
										 |  |  |     if (index.is_number() && index.to_i32() >= 0) | 
					
						
							| 
									
										
										
										
											2020-04-06 17:08:23 +02:00
										 |  |  |         return PropertyName(index.to_i32()); | 
					
						
							|  |  |  |     return PropertyName(index.to_string()); | 
					
						
							| 
									
										
										
										
											2020-03-20 20:51:03 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-19 01:12:51 +01:00
										 |  |  | String MemberExpression::to_string_approximation() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     String object_string = "<object>"; | 
					
						
							|  |  |  |     if (m_object->is_identifier()) | 
					
						
							|  |  |  |         object_string = static_cast<const Identifier&>(*m_object).string(); | 
					
						
							|  |  |  |     if (is_computed()) | 
					
						
							|  |  |  |         return String::format("%s[<computed>]", object_string.characters()); | 
					
						
							|  |  |  |     ASSERT(m_property->is_identifier()); | 
					
						
							|  |  |  |     return String::format("%s.%s", object_string.characters(), static_cast<const Identifier&>(*m_property).string().characters()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 18:58:19 +01:00
										 |  |  | Value MemberExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     auto object_value = m_object->execute(interpreter); | 
					
						
							|  |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     auto* object_result = object_value.to_object(interpreter.heap()); | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-04-25 18:43:34 +02:00
										 |  |  |     return object_result->get(computed_property_name(interpreter)).value_or(js_undefined()); | 
					
						
							| 
									
										
										
										
											2020-03-11 18:58:19 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 12:19:11 +01:00
										 |  |  | Value StringLiteral::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-04 12:57:37 +02:00
										 |  |  |     return js_string(interpreter, m_value); | 
					
						
							| 
									
										
										
										
											2020-03-12 12:19:11 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value NumericLiteral::execute(Interpreter&) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return Value(m_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value BooleanLiteral::execute(Interpreter&) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return Value(m_value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-15 23:32:34 +02:00
										 |  |  | Value NullLiteral::execute(Interpreter&) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return js_null(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:57 +01:00
										 |  |  | void ArrayExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     for (auto& element : m_elements) { | 
					
						
							| 
									
										
										
										
											2020-04-15 20:09:06 +01:00
										 |  |  |         if (element) { | 
					
						
							|  |  |  |             element->dump(indent + 1); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             print_indent(indent + 1); | 
					
						
							|  |  |  |             printf("<empty>\n"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value ArrayExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-04-17 18:24:01 +02:00
										 |  |  |     auto* array = Array::create(interpreter.global_object()); | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:57 +01:00
										 |  |  |     for (auto& element : m_elements) { | 
					
						
							| 
									
										
										
										
											2020-04-15 20:09:06 +01:00
										 |  |  |         auto value = Value(); | 
					
						
							|  |  |  |         if (element) { | 
					
						
							|  |  |  |             value = element->execute(interpreter); | 
					
						
							| 
									
										
										
										
											2020-04-26 23:05:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-15 20:09:06 +01:00
										 |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-26 23:05:37 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (element->is_spread_expression()) { | 
					
						
							|  |  |  |                 if (!value.is_array()) { | 
					
						
							|  |  |  |                     interpreter.throw_exception<TypeError>(String::format("%s is not iterable", value.to_string().characters())); | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 auto& array_to_spread = static_cast<Array&>(value.as_object()); | 
					
						
							|  |  |  |                 for (auto& it : array_to_spread.elements()) { | 
					
						
							|  |  |  |                     if (it.is_empty()) { | 
					
						
							|  |  |  |                         array->elements().append(js_undefined()); | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         array->elements().append(it); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 continue; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-04-15 20:09:06 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-04-13 16:21:54 +01:00
										 |  |  |         array->elements().append(value); | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return array; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 14:03:55 +01:00
										 |  |  | void TryStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("(Block)\n"); | 
					
						
							|  |  |  |     block().dump(indent + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (handler()) { | 
					
						
							|  |  |  |         print_indent(indent); | 
					
						
							|  |  |  |         printf("(Handler)\n"); | 
					
						
							|  |  |  |         handler()->dump(indent + 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (finalizer()) { | 
					
						
							|  |  |  |         print_indent(indent); | 
					
						
							|  |  |  |         printf("(Finalizer)\n"); | 
					
						
							|  |  |  |         finalizer()->dump(indent + 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CatchClause::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("CatchClause"); | 
					
						
							|  |  |  |     if (!m_parameter.is_null()) | 
					
						
							|  |  |  |         printf(" (%s)", m_parameter.characters()); | 
					
						
							|  |  |  |     printf("\n"); | 
					
						
							|  |  |  |     body().dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 22:03:50 +01:00
										 |  |  | void ThrowStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     argument().dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  | Value TryStatement::execute(Interpreter& interpreter) const | 
					
						
							| 
									
										
										
										
											2020-03-24 14:03:55 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |     interpreter.run(block(), {}, ScopeType::Try); | 
					
						
							|  |  |  |     if (auto* exception = interpreter.exception()) { | 
					
						
							|  |  |  |         if (m_handler) { | 
					
						
							|  |  |  |             interpreter.clear_exception(); | 
					
						
							| 
									
										
										
										
											2020-04-06 19:22:12 +02:00
										 |  |  |             ArgumentVector arguments { { m_handler->parameter(), exception->value() } }; | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |             interpreter.run(m_handler->body(), move(arguments)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (m_finalizer) | 
					
						
							|  |  |  |         m_finalizer->execute(interpreter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-24 14:03:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value CatchClause::execute(Interpreter&) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-24 14:37:39 +01:00
										 |  |  |     // NOTE: CatchClause execution is handled by TryStatement.
 | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							| 
									
										
										
										
											2020-03-24 14:03:55 +01:00
										 |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  | Value ThrowStatement::execute(Interpreter& interpreter) const | 
					
						
							| 
									
										
										
										
											2020-03-24 22:03:50 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-27 15:35:35 +01:00
										 |  |  |     auto value = m_argument->execute(interpreter); | 
					
						
							|  |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return interpreter.throw_exception(value); | 
					
						
							| 
									
										
										
										
											2020-03-24 22:03:50 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-29 13:09:54 +02:00
										 |  |  | Value SwitchStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-29 14:34:25 +02:00
										 |  |  |     auto discriminant_result = m_discriminant->execute(interpreter); | 
					
						
							|  |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool falling_through = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& switch_case : m_cases) { | 
					
						
							|  |  |  |         if (!falling_through && switch_case.test()) { | 
					
						
							|  |  |  |             auto test_result = switch_case.test()->execute(interpreter); | 
					
						
							|  |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-15 09:28:41 +02:00
										 |  |  |             if (!eq(interpreter, discriminant_result, test_result).to_boolean()) | 
					
						
							| 
									
										
										
										
											2020-03-29 14:34:25 +02:00
										 |  |  |                 continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         falling_through = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (auto& statement : switch_case.consequent()) { | 
					
						
							|  |  |  |             statement.execute(interpreter); | 
					
						
							|  |  |  |             if (interpreter.exception()) | 
					
						
							|  |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-05 00:09:48 +02:00
										 |  |  |             if (interpreter.should_unwind()) { | 
					
						
							|  |  |  |                 if (interpreter.should_unwind_until(ScopeType::Breakable)) { | 
					
						
							|  |  |  |                     interpreter.stop_unwind(); | 
					
						
							|  |  |  |                     return {}; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2020-03-29 14:34:25 +02:00
										 |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2020-04-05 00:09:48 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-03-29 14:34:25 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-29 13:09:54 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value SwitchCase::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     (void)interpreter; | 
					
						
							|  |  |  |     return {}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value BreakStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-03-29 14:34:25 +02:00
										 |  |  |     interpreter.unwind(ScopeType::Breakable); | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-03-29 13:09:54 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  | Value ContinueStatement::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     interpreter.unwind(ScopeType::Continuable); | 
					
						
							| 
									
										
										
										
											2020-04-06 20:24:45 +02:00
										 |  |  |     return js_undefined(); | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-29 13:09:54 +02:00
										 |  |  | void SwitchStatement::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     m_discriminant->dump(indent + 1); | 
					
						
							|  |  |  |     for (auto& switch_case : m_cases) { | 
					
						
							|  |  |  |         switch_case.dump(indent + 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SwitchCase::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     if (m_test) { | 
					
						
							|  |  |  |         printf("(Test)\n"); | 
					
						
							|  |  |  |         m_test->dump(indent + 1); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         printf("(Default)\n"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("(Consequent)\n"); | 
					
						
							|  |  |  |     int i = 0; | 
					
						
							|  |  |  |     for (auto& statement : m_consequent) { | 
					
						
							|  |  |  |         print_indent(indent); | 
					
						
							|  |  |  |         printf("[%d]\n", i++); | 
					
						
							|  |  |  |         statement.dump(indent + 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-04-01 18:31:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-03 12:14:28 +02:00
										 |  |  | Value ConditionalExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto test_result = m_test->execute(interpreter); | 
					
						
							|  |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     Value result; | 
					
						
							|  |  |  |     if (test_result.to_boolean()) { | 
					
						
							|  |  |  |         result = m_consequent->execute(interpreter); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         result = m_alternate->execute(interpreter); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (interpreter.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ConditionalExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("(Test)\n"); | 
					
						
							|  |  |  |     m_test->dump(indent + 1); | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("(Consequent)\n"); | 
					
						
							|  |  |  |     m_test->dump(indent + 1); | 
					
						
							|  |  |  |     print_indent(indent); | 
					
						
							|  |  |  |     printf("(Alternate)\n"); | 
					
						
							|  |  |  |     m_test->dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 15:11:05 +02:00
										 |  |  | void SequenceExpression::dump(int indent) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     ASTNode::dump(indent); | 
					
						
							|  |  |  |     for (auto& expression : m_expressions) | 
					
						
							|  |  |  |         expression.dump(indent + 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Value SequenceExpression::execute(Interpreter& interpreter) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Value last_value; | 
					
						
							|  |  |  |     for (auto& expression : m_expressions) { | 
					
						
							|  |  |  |         last_value = expression.execute(interpreter); | 
					
						
							|  |  |  |         if (interpreter.exception()) | 
					
						
							|  |  |  |             return {}; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return last_value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-13 16:42:54 +02:00
										 |  |  | void ScopeNode::add_variables(NonnullRefPtrVector<VariableDeclaration> variables) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     m_variables.append(move(variables)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-07 19:42:11 +01:00
										 |  |  | } |