2020-03-07 19:42:11 +01:00
|
|
|
/*
|
2021-06-03 10:46:30 +02:00
|
|
|
* Copyright (c) 2020-2021, Andreas Kling <kling@serenityos.org>
|
2021-04-22 22:51:19 +02:00
|
|
|
* Copyright (c) 2020-2021, Linus Groh <linusg@serenityos.org>
|
2020-03-07 19:42:11 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-03-07 19:42:11 +01:00
|
|
|
*/
|
|
|
|
|
2021-01-17 09:21:15 +01:00
|
|
|
#include <AK/Demangle.h>
|
2020-03-12 19:22:13 +08:00
|
|
|
#include <AK/HashMap.h>
|
2020-09-18 18:00:57 +04:30
|
|
|
#include <AK/HashTable.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-11-28 16:14:26 +01:00
|
|
|
#include <AK/TemporaryChange.h>
|
2020-06-06 01:14:10 +01:00
|
|
|
#include <LibCrypto/BigInt/SignedBigInteger.h>
|
2020-03-07 19:42:11 +01:00
|
|
|
#include <LibJS/AST.h>
|
|
|
|
#include <LibJS/Interpreter.h>
|
2021-06-22 01:14:27 +02:00
|
|
|
#include <LibJS/Runtime/AbstractOperations.h>
|
2020-05-21 17:28:28 -07:00
|
|
|
#include <LibJS/Runtime/Accessor.h>
|
2020-03-20 20:29:57 +01:00
|
|
|
#include <LibJS/Runtime/Array.h>
|
2020-06-06 01:14:10 +01:00
|
|
|
#include <LibJS/Runtime/BigInt.h>
|
2021-09-24 22:40:38 +02:00
|
|
|
#include <LibJS/Runtime/ECMAScriptFunctionObject.h>
|
2020-03-24 14:37:39 +01:00
|
|
|
#include <LibJS/Runtime/Error.h>
|
2021-07-01 12:24:46 +02:00
|
|
|
#include <LibJS/Runtime/FunctionEnvironment.h>
|
2020-04-08 11:05:38 +02:00
|
|
|
#include <LibJS/Runtime/GlobalObject.h>
|
2020-07-13 08:27:20 -07:00
|
|
|
#include <LibJS/Runtime/IteratorOperations.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>
|
2021-07-01 12:24:46 +02:00
|
|
|
#include <LibJS/Runtime/ObjectEnvironment.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-06-03 16:05:49 -07:00
|
|
|
#include <LibJS/Runtime/RegExpObject.h>
|
2020-04-27 21:52:47 -07:00
|
|
|
#include <LibJS/Runtime/Shape.h>
|
2021-02-10 12:21:14 +01:00
|
|
|
#include <typeinfo>
|
2020-03-07 19:42:11 +01:00
|
|
|
|
|
|
|
namespace JS {
|
|
|
|
|
2021-03-16 10:51:55 +01:00
|
|
|
class InterpreterNodeScope {
|
|
|
|
AK_MAKE_NONCOPYABLE(InterpreterNodeScope);
|
|
|
|
AK_MAKE_NONMOVABLE(InterpreterNodeScope);
|
|
|
|
|
|
|
|
public:
|
2021-06-11 02:13:06 +04:30
|
|
|
InterpreterNodeScope(Interpreter& interpreter, ASTNode const& node)
|
2021-03-16 10:51:55 +01:00
|
|
|
: m_interpreter(interpreter)
|
2021-03-21 17:38:42 +01:00
|
|
|
, m_chain_node { nullptr, node }
|
2021-03-16 10:51:55 +01:00
|
|
|
{
|
2021-06-24 19:17:45 +02:00
|
|
|
m_interpreter.vm().running_execution_context().current_node = &node;
|
2021-03-21 17:38:42 +01:00
|
|
|
m_interpreter.push_ast_node(m_chain_node);
|
2021-03-16 10:51:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
~InterpreterNodeScope()
|
|
|
|
{
|
2021-03-21 17:38:42 +01:00
|
|
|
m_interpreter.pop_ast_node();
|
2021-03-16 10:51:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Interpreter& m_interpreter;
|
2021-03-21 17:38:42 +01:00
|
|
|
ExecutingASTNodeChain m_chain_node;
|
2021-03-16 10:51:55 +01:00
|
|
|
};
|
|
|
|
|
2021-01-17 09:21:15 +01:00
|
|
|
String ASTNode::class_name() const
|
|
|
|
{
|
|
|
|
// NOTE: We strip the "JS::" prefix.
|
|
|
|
return demangle(typeid(*this).name()).substring(4);
|
|
|
|
}
|
|
|
|
|
2021-06-11 02:13:06 +04:30
|
|
|
static void update_function_name(Value value, FlyString const& name)
|
2020-09-18 18:00:57 +04:30
|
|
|
{
|
2021-03-21 17:14:20 +01:00
|
|
|
if (!value.is_function())
|
|
|
|
return;
|
|
|
|
auto& function = value.as_function();
|
2021-09-24 22:40:38 +02:00
|
|
|
if (is<ECMAScriptFunctionObject>(function) && function.name().is_empty())
|
|
|
|
static_cast<ECMAScriptFunctionObject&>(function).set_name(name);
|
2020-09-18 18:00:57 +04:30
|
|
|
}
|
|
|
|
|
2020-09-27 18:36:49 +02:00
|
|
|
static String get_function_name(GlobalObject& global_object, Value value)
|
2020-07-07 21:38:46 -07:00
|
|
|
{
|
|
|
|
if (value.is_symbol())
|
2020-10-04 15:18:52 +01:00
|
|
|
return String::formatted("[{}]", value.as_symbol().description());
|
2020-07-07 21:38:46 -07:00
|
|
|
if (value.is_string())
|
|
|
|
return value.as_string().string();
|
2020-09-27 18:36:49 +02:00
|
|
|
return value.to_string(global_object);
|
2020-07-07 21:38:46 -07:00
|
|
|
}
|
|
|
|
|
2020-06-08 21:25:16 +02:00
|
|
|
Value ScopeNode::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-07 19:42:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-09-27 16:56:58 +02:00
|
|
|
return interpreter.execute_statement(global_object, *this);
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
2020-10-03 17:02:43 -07:00
|
|
|
Value Program::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-08 18:21:55 +01:00
|
|
|
return interpreter.execute_statement(global_object, *this, ScopeType::Block);
|
2020-10-03 17:02:43 -07:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value FunctionDeclaration::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-07 19:42:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2021-03-16 09:12:34 +01:00
|
|
|
return {};
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
2021-07-07 22:53:32 +02:00
|
|
|
// 15.2.5 Runtime Semantics: InstantiateOrdinaryFunctionExpression, https://tc39.es/ecma262/#sec-runtime-semantics-instantiateordinaryfunctionexpression
|
2020-06-08 20:57:54 +02:00
|
|
|
Value FunctionExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-19 11:12:08 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2021-07-07 22:53:32 +02:00
|
|
|
auto* func_env = interpreter.lexical_environment();
|
|
|
|
bool has_identifier = !name().is_empty() && !is_auto_renamed();
|
|
|
|
|
|
|
|
if (has_identifier) {
|
|
|
|
func_env = interpreter.heap().allocate<DeclarativeEnvironment>(global_object, func_env);
|
|
|
|
func_env->create_immutable_binding(global_object, name(), false);
|
|
|
|
}
|
|
|
|
|
2021-09-24 22:40:38 +02:00
|
|
|
auto closure = ECMAScriptFunctionObject::create(global_object, name(), body(), parameters(), function_length(), func_env, kind(), is_strict_mode(), is_arrow_function());
|
2021-07-07 22:53:32 +02:00
|
|
|
|
|
|
|
if (has_identifier)
|
|
|
|
func_env->initialize_binding(global_object, name(), closure);
|
|
|
|
|
|
|
|
return closure;
|
2020-03-19 11:12:08 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ExpressionStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-11 19:27:43 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-06-08 20:57:54 +02:00
|
|
|
return m_expression->execute(interpreter, global_object);
|
2020-03-11 19:27:43 +01:00
|
|
|
}
|
|
|
|
|
2021-09-14 06:56:31 +04:30
|
|
|
CallExpression::ThisAndCallee CallExpression::compute_this_and_callee(Interpreter& interpreter, GlobalObject& global_object, Reference const& callee_reference) const
|
2020-04-01 18:51:27 +02:00
|
|
|
{
|
2020-09-27 18:36:49 +02:00
|
|
|
auto& vm = interpreter.vm();
|
|
|
|
|
2021-09-14 06:56:31 +04:30
|
|
|
if (callee_reference.is_property_reference()) {
|
|
|
|
auto this_value = callee_reference.get_this_value();
|
|
|
|
auto callee = callee_reference.get_value(global_object);
|
|
|
|
if (vm.exception())
|
|
|
|
return {};
|
2020-06-08 13:31:21 -05:00
|
|
|
|
2020-04-01 18:51:27 +02:00
|
|
|
return { this_value, callee };
|
|
|
|
}
|
2021-06-04 11:52:20 +01:00
|
|
|
|
2021-07-12 01:37:51 +02:00
|
|
|
// [[Call]] will handle that in non-strict mode the this value becomes the global object
|
2021-09-14 06:56:31 +04:30
|
|
|
return {
|
|
|
|
js_undefined(),
|
|
|
|
callee_reference.is_unresolvable()
|
|
|
|
? m_callee->execute(interpreter, global_object)
|
|
|
|
: callee_reference.get_value(global_object)
|
|
|
|
};
|
2020-04-01 18:51:27 +02:00
|
|
|
}
|
|
|
|
|
2021-07-02 14:24:50 +02:00
|
|
|
// 13.3.8.1 Runtime Semantics: ArgumentListEvaluation, https://tc39.es/ecma262/#sec-runtime-semantics-argumentlistevaluation
|
|
|
|
static void argument_list_evaluation(Interpreter& interpreter, GlobalObject& global_object, Vector<CallExpression::Argument> const& arguments, MarkedValueList& list)
|
|
|
|
{
|
|
|
|
auto& vm = global_object.vm();
|
|
|
|
list.ensure_capacity(arguments.size());
|
|
|
|
|
|
|
|
for (auto& argument : arguments) {
|
|
|
|
auto value = argument.value->execute(interpreter, global_object);
|
|
|
|
if (vm.exception())
|
|
|
|
return;
|
|
|
|
if (argument.is_spread) {
|
|
|
|
get_iterator_values(global_object, value, [&](Value iterator_value) {
|
|
|
|
if (vm.exception())
|
|
|
|
return IterationDecision::Break;
|
|
|
|
list.append(iterator_value);
|
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
|
|
|
if (vm.exception())
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
list.append(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-02 18:25:32 +02:00
|
|
|
Value NewExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-07 19:42:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-10-13 19:00:37 +02:00
|
|
|
auto& vm = interpreter.vm();
|
2021-07-02 18:25:32 +02:00
|
|
|
|
2021-07-02 18:43:25 +02:00
|
|
|
auto callee_value = m_callee->execute(interpreter, global_object);
|
2020-10-13 19:00:37 +02:00
|
|
|
if (vm.exception())
|
2020-03-24 14:37:39 +01:00
|
|
|
return {};
|
|
|
|
|
2021-09-25 09:52:49 +02:00
|
|
|
if (!callee_value.is_function() || !callee_value.as_function().has_constructor()) {
|
2021-07-02 18:43:25 +02:00
|
|
|
throw_type_error_for_callee(interpreter, global_object, callee_value, "constructor"sv);
|
2021-07-02 18:25:32 +02:00
|
|
|
return {};
|
|
|
|
}
|
2020-04-06 20:24:45 +02:00
|
|
|
|
2021-07-02 18:25:32 +02:00
|
|
|
MarkedValueList arg_list(vm.heap());
|
|
|
|
argument_list_evaluation(interpreter, global_object, m_arguments, arg_list);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-07-02 18:43:25 +02:00
|
|
|
auto& function = callee_value.as_function();
|
2021-07-02 18:25:32 +02:00
|
|
|
return vm.construct(function, function, move(arg_list));
|
|
|
|
}
|
|
|
|
|
|
|
|
void CallExpression::throw_type_error_for_callee(Interpreter& interpreter, GlobalObject& global_object, Value callee_value, StringView call_type) const
|
|
|
|
{
|
|
|
|
auto& vm = interpreter.vm();
|
|
|
|
if (is<Identifier>(*m_callee) || is<MemberExpression>(*m_callee)) {
|
|
|
|
String expression_string;
|
|
|
|
if (is<Identifier>(*m_callee)) {
|
|
|
|
expression_string = static_cast<Identifier const&>(*m_callee).string();
|
2020-04-19 01:12:51 +01:00
|
|
|
} else {
|
2021-07-02 18:25:32 +02:00
|
|
|
expression_string = static_cast<MemberExpression const&>(*m_callee).to_string_approximation();
|
2020-04-19 01:12:51 +01:00
|
|
|
}
|
2021-07-02 18:25:32 +02:00
|
|
|
vm.throw_exception<TypeError>(global_object, ErrorType::IsNotAEvaluatedFrom, callee_value.to_string_without_side_effects(), call_type, expression_string);
|
|
|
|
} else {
|
|
|
|
vm.throw_exception<TypeError>(global_object, ErrorType::IsNotA, callee_value.to_string_without_side_effects(), call_type);
|
2020-04-01 18:31:24 +01:00
|
|
|
}
|
2021-07-02 18:25:32 +02:00
|
|
|
}
|
2020-04-01 18:31:24 +01:00
|
|
|
|
2021-07-02 18:25:32 +02:00
|
|
|
Value CallExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
|
|
|
auto& vm = interpreter.vm();
|
2021-09-14 06:56:31 +04:30
|
|
|
auto callee_reference = m_callee->to_reference(interpreter, global_object);
|
|
|
|
if (vm.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto [this_value, callee] = compute_this_and_callee(interpreter, global_object, callee_reference);
|
2021-07-02 18:25:32 +02:00
|
|
|
if (vm.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
VERIFY(!callee.is_empty());
|
|
|
|
|
2021-07-02 14:24:50 +02:00
|
|
|
MarkedValueList arg_list(vm.heap());
|
|
|
|
argument_list_evaluation(interpreter, global_object, m_arguments, arg_list);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-03-12 19:22:13 +08:00
|
|
|
|
2021-09-13 17:44:08 +01:00
|
|
|
if (!callee.is_function()) {
|
|
|
|
throw_type_error_for_callee(interpreter, global_object, callee, "function"sv);
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-07-02 18:25:32 +02:00
|
|
|
auto& function = callee.as_function();
|
|
|
|
|
2021-09-14 06:56:31 +04:30
|
|
|
if (&function == global_object.eval_function()
|
|
|
|
&& callee_reference.is_environment_reference()
|
|
|
|
&& callee_reference.name().is_string()
|
|
|
|
&& callee_reference.name().as_string() == vm.names.eval.as_string()) {
|
|
|
|
|
2021-07-02 14:24:50 +02:00
|
|
|
auto script_value = arg_list.size() == 0 ? js_undefined() : arg_list[0];
|
2021-09-21 22:16:08 +03:00
|
|
|
return TRY_OR_DISCARD(perform_eval(script_value, global_object, vm.in_strict_mode() ? CallerMode::Strict : CallerMode::NonStrict, EvalMode::Direct));
|
2021-06-19 20:13:53 -07:00
|
|
|
}
|
|
|
|
|
2021-09-23 20:56:28 +03:00
|
|
|
return TRY_OR_DISCARD(vm.call(function, this_value, move(arg_list)));
|
2021-07-02 19:30:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// 13.3.7.1 Runtime Semantics: Evaluation, https://tc39.es/ecma262/#sec-super-keyword-runtime-semantics-evaluation
|
|
|
|
// SuperCall : super Arguments
|
|
|
|
Value SuperCall::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
|
|
|
auto& vm = interpreter.vm();
|
|
|
|
|
|
|
|
// 1. Let newTarget be GetNewTarget().
|
|
|
|
auto new_target = vm.get_new_target();
|
|
|
|
if (vm.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
// 2. Assert: Type(newTarget) is Object.
|
|
|
|
VERIFY(new_target.is_function());
|
2020-06-08 13:31:21 -05:00
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
// 3. Let func be ! GetSuperConstructor().
|
|
|
|
auto* func = get_super_constructor(interpreter.vm());
|
|
|
|
VERIFY(!vm.exception());
|
2021-07-02 18:25:32 +02:00
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
// 4. Let argList be ? ArgumentListEvaluation of Arguments.
|
|
|
|
MarkedValueList arg_list(vm.heap());
|
|
|
|
argument_list_evaluation(interpreter, global_object, m_arguments, arg_list);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
// 5. If IsConstructor(func) is false, throw a TypeError exception.
|
|
|
|
// FIXME: This check is non-conforming.
|
|
|
|
if (!func || !func->is_function()) {
|
2021-07-02 18:25:32 +02:00
|
|
|
vm.throw_exception<TypeError>(global_object, ErrorType::NotAConstructor, "Super constructor");
|
|
|
|
return {};
|
2020-03-24 14:37:39 +01:00
|
|
|
}
|
2021-07-02 19:30:38 +02:00
|
|
|
|
|
|
|
// 6. Let result be ? Construct(func, argList, newTarget).
|
|
|
|
auto& function = new_target.as_function();
|
|
|
|
auto result = vm.construct(static_cast<FunctionObject&>(*func), function, move(arg_list));
|
2021-07-02 18:25:32 +02:00
|
|
|
if (vm.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
// 7. Let thisER be GetThisEnvironment().
|
|
|
|
auto& this_er = verify_cast<FunctionEnvironment>(get_this_environment(interpreter.vm()));
|
2020-03-15 15:01:10 +01:00
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
// 8. Perform ? thisER.BindThisValue(result).
|
|
|
|
this_er.bind_this_value(global_object, result);
|
2020-10-13 19:00:37 +02:00
|
|
|
if (vm.exception())
|
2020-04-06 20:24:45 +02:00
|
|
|
return {};
|
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
// 9. Let F be thisER.[[FunctionObject]].
|
|
|
|
// 10. Assert: F is an ECMAScript function object. (NOTE: This is implied by the strong C++ type.)
|
|
|
|
[[maybe_unused]] auto& f = this_er.function_object();
|
|
|
|
|
|
|
|
// 11. Perform ? InitializeInstanceElements(result, F).
|
2021-08-28 17:11:05 +02:00
|
|
|
VERIFY(result.is_object());
|
|
|
|
vm.initialize_instance_elements(result.as_object(), f);
|
2021-07-02 19:30:38 +02:00
|
|
|
|
|
|
|
// 12. Return result.
|
2020-03-15 15:01:10 +01:00
|
|
|
return result;
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
2021-06-11 01:38:30 +04:30
|
|
|
Value YieldExpression::execute(Interpreter&, GlobalObject&) const
|
|
|
|
{
|
|
|
|
// This should be transformed to a return.
|
|
|
|
VERIFY_NOT_REACHED();
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ReturnStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-07 19:42:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-06-08 20:57:54 +02:00
|
|
|
auto value = argument() ? argument()->execute(interpreter, global_object) : js_undefined();
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().unwind(ScopeType::Function);
|
2020-03-07 19:42:11 +01:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value IfStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-08 07:58:58 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto predicate_result = m_predicate->execute(interpreter, global_object);
|
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-09-27 16:56:58 +02:00
|
|
|
return interpreter.execute_statement(global_object, *m_consequent);
|
2020-03-21 18:40:17 +01:00
|
|
|
|
|
|
|
if (m_alternate)
|
2020-09-27 16:56:58 +02:00
|
|
|
return interpreter.execute_statement(global_object, *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
|
|
|
}
|
|
|
|
|
2021-07-04 17:22:27 +02:00
|
|
|
// 14.11.2 Runtime Semantics: Evaluation, https://tc39.es/ecma262/#sec-with-statement-runtime-semantics-evaluation
|
|
|
|
// WithStatement : with ( Expression ) Statement
|
2020-11-28 16:14:26 +01:00
|
|
|
Value WithStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-11-28 15:05:57 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-07-04 17:22:27 +02:00
|
|
|
// 1. Let value be the result of evaluating Expression.
|
|
|
|
auto value = m_object->execute(interpreter, global_object);
|
2020-11-28 16:14:26 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-07-04 17:22:27 +02:00
|
|
|
// 2. Let obj be ? ToObject(? GetValue(value)).
|
|
|
|
auto* object = value.to_object(global_object);
|
2020-11-28 16:14:26 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-07-04 17:22:27 +02:00
|
|
|
// 3. Let oldEnv be the running execution context's LexicalEnvironment.
|
|
|
|
auto* old_environment = interpreter.vm().running_execution_context().lexical_environment;
|
2020-11-28 16:14:26 +01:00
|
|
|
|
2021-07-04 17:22:27 +02:00
|
|
|
// 4. Let newEnv be NewObjectEnvironment(obj, true, oldEnv).
|
|
|
|
auto* new_environment = new_object_environment(*object, true, old_environment);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
// 5. Set the running execution context's LexicalEnvironment to newEnv.
|
|
|
|
interpreter.vm().running_execution_context().lexical_environment = new_environment;
|
|
|
|
|
|
|
|
// 6. Let C be the result of evaluating Statement.
|
|
|
|
auto result = interpreter.execute_statement(global_object, m_body).value_or(js_undefined());
|
|
|
|
|
|
|
|
// 7. Set the running execution context's LexicalEnvironment to oldEnv.
|
|
|
|
interpreter.vm().running_execution_context().lexical_environment = old_environment;
|
|
|
|
|
2021-09-08 20:46:11 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-07-04 17:22:27 +02:00
|
|
|
// 8. Return Completion(UpdateEmpty(C, undefined)).
|
|
|
|
return result;
|
2020-11-28 15:05:57 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value WhileStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-09 03:22:21 +08:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-03-16 09:12:34 +01:00
|
|
|
auto last_value = js_undefined();
|
2020-10-23 00:40:14 +01:00
|
|
|
for (;;) {
|
|
|
|
auto test_result = m_test->execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-10-23 00:40:14 +01:00
|
|
|
if (!test_result.to_boolean())
|
|
|
|
break;
|
2021-03-16 09:12:34 +01:00
|
|
|
last_value = interpreter.execute_statement(global_object, *m_body).value_or(last_value);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-10-08 13:12:31 -07:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-10-08 13:12:31 -07:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-10-08 13:12:31 -07:00
|
|
|
interpreter.vm().stop_unwind();
|
|
|
|
break;
|
|
|
|
} else {
|
2020-10-08 13:17:40 -07:00
|
|
|
return last_value;
|
2020-10-08 13:12:31 -07:00
|
|
|
}
|
|
|
|
}
|
2020-03-09 03:22:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return last_value;
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value DoWhileStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-04 21:29:23 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-03-16 09:12:34 +01:00
|
|
|
auto last_value = js_undefined();
|
2020-10-23 00:40:14 +01:00
|
|
|
for (;;) {
|
2020-04-04 21:29:23 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-03-16 09:12:34 +01:00
|
|
|
last_value = interpreter.execute_statement(global_object, *m_body).value_or(last_value);
|
2020-04-04 21:29:23 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-10-08 13:12:31 -07:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-10-08 13:12:31 -07:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-10-08 13:12:31 -07:00
|
|
|
interpreter.vm().stop_unwind();
|
|
|
|
break;
|
|
|
|
} else {
|
2020-10-08 13:17:40 -07:00
|
|
|
return last_value;
|
2020-10-08 13:12:31 -07:00
|
|
|
}
|
|
|
|
}
|
2020-10-23 00:40:14 +01:00
|
|
|
auto test_result = m_test->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
if (!test_result.to_boolean())
|
|
|
|
break;
|
|
|
|
}
|
2020-04-04 21:29:23 +02:00
|
|
|
|
|
|
|
return last_value;
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ForStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-12 23:12:12 +11:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-03-18 11:23:53 +01:00
|
|
|
RefPtr<BlockStatement> wrapper;
|
2020-03-14 13:56:49 +02:00
|
|
|
|
2021-06-11 02:13:06 +04:30
|
|
|
if (m_init && is<VariableDeclaration>(*m_init) && static_cast<VariableDeclaration const&>(*m_init).declaration_kind() != DeclarationKind::Var) {
|
2020-12-28 20:45:22 +03:30
|
|
|
wrapper = create_ast_node<BlockStatement>(source_range());
|
2020-05-05 01:58:53 +03:00
|
|
|
NonnullRefPtrVector<VariableDeclaration> decls;
|
2021-06-11 02:13:06 +04:30
|
|
|
decls.append(*static_cast<VariableDeclaration const*>(m_init.ptr()));
|
2020-05-05 01:58:53 +03:00
|
|
|
wrapper->add_variables(decls);
|
2020-12-08 18:21:55 +01:00
|
|
|
interpreter.enter_scope(*wrapper, ScopeType::Block, global_object);
|
2020-03-14 13:56:49 +02:00
|
|
|
}
|
|
|
|
|
2020-04-05 00:24:32 +02:00
|
|
|
auto wrapper_cleanup = ScopeGuard([&] {
|
|
|
|
if (wrapper)
|
2020-09-27 16:56:58 +02:00
|
|
|
interpreter.exit_scope(*wrapper);
|
2020-04-05 00:24:32 +02:00
|
|
|
});
|
|
|
|
|
2021-03-16 09:12:34 +01:00
|
|
|
auto last_value = js_undefined();
|
2020-03-27 15:35:35 +01:00
|
|
|
if (m_init) {
|
2020-06-08 20:57:54 +02:00
|
|
|
m_init->execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
2020-03-12 23:12:12 +11:00
|
|
|
|
|
|
|
if (m_test) {
|
2020-04-28 20:16:40 +01:00
|
|
|
while (true) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto test_result = m_test->execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-28 20:16:40 +01:00
|
|
|
if (!test_result.to_boolean())
|
|
|
|
break;
|
2021-03-16 09:12:34 +01:00
|
|
|
last_value = interpreter.execute_statement(global_object, *m_body).value_or(last_value);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-09-27 15:18:55 +02:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2020-04-05 00:22:42 +02:00
|
|
|
break;
|
|
|
|
} else {
|
2020-10-08 13:17:40 -07:00
|
|
|
return last_value;
|
2020-04-05 00:22:42 +02:00
|
|
|
}
|
|
|
|
}
|
2020-03-27 15:35:35 +01:00
|
|
|
if (m_update) {
|
2020-06-08 20:57:54 +02:00
|
|
|
m_update->execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
2020-03-12 23:12:12 +11:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
while (true) {
|
2021-03-16 09:12:34 +01:00
|
|
|
last_value = interpreter.execute_statement(global_object, *m_body).value_or(last_value);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-09-27 15:18:55 +02:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2020-04-05 00:22:42 +02:00
|
|
|
break;
|
|
|
|
} else {
|
2020-10-08 13:17:40 -07:00
|
|
|
return last_value;
|
2020-04-05 00:22:42 +02:00
|
|
|
}
|
|
|
|
}
|
2020-03-27 15:35:35 +01:00
|
|
|
if (m_update) {
|
2020-06-08 20:57:54 +02:00
|
|
|
m_update->execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
2020-03-12 23:12:12 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return last_value;
|
|
|
|
}
|
|
|
|
|
2021-06-11 02:13:06 +04:30
|
|
|
static Variant<NonnullRefPtr<Identifier>, NonnullRefPtr<BindingPattern>> variable_from_for_declaration(Interpreter& interpreter, GlobalObject& global_object, ASTNode const& node, RefPtr<BlockStatement> wrapper)
|
2020-04-21 19:21:26 +01:00
|
|
|
{
|
2021-01-01 19:34:07 +01:00
|
|
|
if (is<VariableDeclaration>(node)) {
|
2021-06-11 02:13:06 +04:30
|
|
|
auto& variable_declaration = static_cast<VariableDeclaration const&>(node);
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(!variable_declaration.declarations().is_empty());
|
2021-01-01 19:34:07 +01:00
|
|
|
if (variable_declaration.declaration_kind() != DeclarationKind::Var) {
|
|
|
|
wrapper = create_ast_node<BlockStatement>(node.source_range());
|
2020-12-08 18:21:55 +01:00
|
|
|
interpreter.enter_scope(*wrapper, ScopeType::Block, global_object);
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
2021-01-01 19:34:07 +01:00
|
|
|
variable_declaration.execute(interpreter, global_object);
|
2021-05-29 16:03:19 +04:30
|
|
|
return variable_declaration.declarations().first().target();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is<Identifier>(node)) {
|
2021-06-11 02:13:06 +04:30
|
|
|
return NonnullRefPtr(static_cast<Identifier const&>(node));
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
2021-05-29 16:03:19 +04:30
|
|
|
|
|
|
|
VERIFY_NOT_REACHED();
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ForInStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-21 19:21:26 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-02-26 00:37:27 +02:00
|
|
|
bool has_declaration = is<VariableDeclaration>(*m_lhs);
|
|
|
|
if (!has_declaration && !is<Identifier>(*m_lhs)) {
|
2020-04-21 19:21:26 +01:00
|
|
|
// FIXME: Implement "for (foo.bar in baz)", "for (foo[0] in bar)"
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
|
|
|
RefPtr<BlockStatement> wrapper;
|
2021-05-29 16:03:19 +04:30
|
|
|
auto target = variable_from_for_declaration(interpreter, global_object, m_lhs, wrapper);
|
2020-04-21 19:21:26 +01:00
|
|
|
auto wrapper_cleanup = ScopeGuard([&] {
|
|
|
|
if (wrapper)
|
2020-09-27 16:56:58 +02:00
|
|
|
interpreter.exit_scope(*wrapper);
|
2020-04-21 19:21:26 +01:00
|
|
|
});
|
|
|
|
auto last_value = js_undefined();
|
2020-06-08 20:57:54 +02:00
|
|
|
auto rhs_result = m_rhs->execute(interpreter, global_object);
|
2020-04-21 19:21:26 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-04-10 18:47:18 +02:00
|
|
|
if (rhs_result.is_nullish())
|
|
|
|
return {};
|
2020-09-27 18:36:49 +02:00
|
|
|
auto* object = rhs_result.to_object(global_object);
|
2020-04-21 19:21:26 +01:00
|
|
|
while (object) {
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
auto property_names = object->enumerable_own_property_names(Object::PropertyKind::Key);
|
2021-04-06 21:39:17 +02:00
|
|
|
for (auto& value : property_names) {
|
2021-05-29 16:03:19 +04:30
|
|
|
interpreter.vm().assign(target, value, global_object, has_declaration);
|
LibJS: Object index properties have descriptors; Handle sparse indices
This patch adds an IndexedProperties object for storing indexed
properties within an Object. This accomplishes two goals: indexed
properties now have an associated descriptor, and objects now gracefully
handle sparse properties.
The IndexedProperties class is a wrapper around two other classes, one
for simple indexed properties storage, and one for general indexed
property storage. Simple indexed property storage is the common-case,
and is simply a vector of properties which all have attributes of
default_attributes (writable, enumerable, and configurable).
General indexed property storage is for a collection of indexed
properties where EITHER one or more properties have attributes other
than default_attributes OR there is a property with a large index (in
particular, large is '200' or higher).
Indexed properties are now treated relatively the same as storage within
the various Object methods. Additionally, there is a custom iterator
class for IndexedProperties which makes iteration easy. The iterator
skips empty values by default, but can be configured otherwise.
Likewise, it evaluates getters by default, but can be set not to.
2020-05-27 11:35:09 -07:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-03-16 09:12:34 +01:00
|
|
|
last_value = interpreter.execute_statement(global_object, *m_body).value_or(last_value);
|
2020-04-21 19:21:26 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-09-27 15:18:55 +02:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2020-04-21 19:21:26 +01:00
|
|
|
break;
|
|
|
|
} else {
|
2020-10-08 13:17:40 -07:00
|
|
|
return last_value;
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-09-28 23:30:17 +01:00
|
|
|
object = TRY_OR_DISCARD(object->internal_get_prototype_of());
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
|
|
|
return last_value;
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ForOfStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-21 19:21:26 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-02-26 00:37:27 +02:00
|
|
|
bool has_declaration = is<VariableDeclaration>(*m_lhs);
|
|
|
|
if (!has_declaration && !is<Identifier>(*m_lhs)) {
|
2020-04-21 19:21:26 +01:00
|
|
|
// FIXME: Implement "for (foo.bar of baz)", "for (foo[0] of bar)"
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
|
|
|
RefPtr<BlockStatement> wrapper;
|
2021-05-29 16:03:19 +04:30
|
|
|
auto target = variable_from_for_declaration(interpreter, global_object, m_lhs, wrapper);
|
2020-04-21 19:21:26 +01:00
|
|
|
auto wrapper_cleanup = ScopeGuard([&] {
|
|
|
|
if (wrapper)
|
2020-09-27 16:56:58 +02:00
|
|
|
interpreter.exit_scope(*wrapper);
|
2020-04-21 19:21:26 +01:00
|
|
|
});
|
|
|
|
auto last_value = js_undefined();
|
2020-06-08 20:57:54 +02:00
|
|
|
auto rhs_result = m_rhs->execute(interpreter, global_object);
|
2020-04-21 19:21:26 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2020-09-08 14:15:13 +02:00
|
|
|
get_iterator_values(global_object, rhs_result, [&](Value value) {
|
2021-05-29 16:03:19 +04:30
|
|
|
interpreter.vm().assign(target, value, global_object, has_declaration);
|
2021-03-16 09:12:34 +01:00
|
|
|
last_value = interpreter.execute_statement(global_object, *m_body).value_or(last_value);
|
2020-04-21 19:21:26 +01:00
|
|
|
if (interpreter.exception())
|
2020-07-13 08:27:20 -07:00
|
|
|
return IterationDecision::Break;
|
2020-09-27 15:18:55 +02:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2020-07-13 08:27:20 -07:00
|
|
|
return IterationDecision::Break;
|
2020-04-21 19:21:26 +01:00
|
|
|
} else {
|
2020-07-13 08:27:20 -07:00
|
|
|
return IterationDecision::Break;
|
2020-04-21 19:21:26 +01:00
|
|
|
}
|
|
|
|
}
|
2020-07-13 08:27:20 -07:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2020-04-21 19:21:26 +01:00
|
|
|
return last_value;
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value BinaryExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-07 19:42:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto lhs_result = m_lhs->execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-06-08 20:57:54 +02:00
|
|
|
auto rhs_result = m_rhs->execute(interpreter, global_object);
|
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-09-27 19:52:47 +02:00
|
|
|
return add(global_object, lhs_result, rhs_result);
|
2020-04-05 12:56:53 +01:00
|
|
|
case BinaryOp::Subtraction:
|
2020-09-27 19:52:47 +02:00
|
|
|
return sub(global_object, lhs_result, rhs_result);
|
2020-04-05 12:56:53 +01:00
|
|
|
case BinaryOp::Multiplication:
|
2020-09-27 19:52:47 +02:00
|
|
|
return mul(global_object, lhs_result, rhs_result);
|
2020-04-05 12:56:53 +01:00
|
|
|
case BinaryOp::Division:
|
2020-09-27 19:52:47 +02:00
|
|
|
return div(global_object, lhs_result, rhs_result);
|
2020-04-04 21:17:34 +02:00
|
|
|
case BinaryOp::Modulo:
|
2020-09-27 19:52:47 +02:00
|
|
|
return mod(global_object, lhs_result, rhs_result);
|
2020-04-05 13:40:00 +01:00
|
|
|
case BinaryOp::Exponentiation:
|
2020-09-27 18:36:49 +02:00
|
|
|
return exp(global_object, lhs_result, rhs_result);
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::StrictlyEquals:
|
2021-09-23 23:43:28 +02:00
|
|
|
return Value(is_strictly_equal(lhs_result, rhs_result));
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::StrictlyInequals:
|
2021-09-23 23:43:28 +02:00
|
|
|
return Value(!is_strictly_equal(lhs_result, rhs_result));
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::LooselyEquals:
|
2021-09-23 23:46:36 +02:00
|
|
|
return Value(is_loosely_equal(global_object, lhs_result, rhs_result));
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::LooselyInequals:
|
2021-09-23 23:46:36 +02:00
|
|
|
return Value(!is_loosely_equal(global_object, lhs_result, rhs_result));
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::GreaterThan:
|
2020-09-27 19:52:47 +02:00
|
|
|
return greater_than(global_object, lhs_result, rhs_result);
|
2020-03-12 23:07:08 +11:00
|
|
|
case BinaryOp::GreaterThanEquals:
|
2020-09-27 19:52:47 +02:00
|
|
|
return greater_than_equals(global_object, lhs_result, rhs_result);
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::LessThan:
|
2020-09-27 19:52:47 +02:00
|
|
|
return less_than(global_object, lhs_result, rhs_result);
|
2020-03-12 23:07:08 +11:00
|
|
|
case BinaryOp::LessThanEquals:
|
2020-09-27 19:52:47 +02:00
|
|
|
return less_than_equals(global_object, lhs_result, rhs_result);
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::BitwiseAnd:
|
2020-09-27 19:52:47 +02:00
|
|
|
return bitwise_and(global_object, lhs_result, rhs_result);
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::BitwiseOr:
|
2020-09-27 19:52:47 +02:00
|
|
|
return bitwise_or(global_object, lhs_result, rhs_result);
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::BitwiseXor:
|
2020-09-27 19:52:47 +02:00
|
|
|
return bitwise_xor(global_object, lhs_result, rhs_result);
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::LeftShift:
|
2020-09-27 19:52:47 +02:00
|
|
|
return left_shift(global_object, lhs_result, rhs_result);
|
2020-03-10 11:35:05 +01:00
|
|
|
case BinaryOp::RightShift:
|
2020-09-27 19:52:47 +02:00
|
|
|
return right_shift(global_object, lhs_result, rhs_result);
|
2020-04-23 15:43:10 +01:00
|
|
|
case BinaryOp::UnsignedRightShift:
|
2020-09-27 19:52:47 +02:00
|
|
|
return unsigned_right_shift(global_object, lhs_result, rhs_result);
|
2020-04-23 16:06:01 +01:00
|
|
|
case BinaryOp::In:
|
2020-09-27 19:52:47 +02:00
|
|
|
return in(global_object, lhs_result, rhs_result);
|
2020-03-28 16:56:54 +01:00
|
|
|
case BinaryOp::InstanceOf:
|
2020-09-27 18:36:49 +02:00
|
|
|
return instance_of(global_object, lhs_result, rhs_result);
|
2020-03-08 07:53:02 +02:00
|
|
|
}
|
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-08 07:53:02 +02:00
|
|
|
}
|
2020-03-08 07:55:44 +02:00
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value LogicalExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-08 07:55:44 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto lhs_result = m_lhs->execute(interpreter, global_object);
|
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()) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto rhs_result = m_rhs->execute(interpreter, global_object);
|
2020-04-03 19:11:31 +02:00
|
|
|
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-06-08 20:57:54 +02:00
|
|
|
auto rhs_result = m_rhs->execute(interpreter, global_object);
|
2020-04-03 19:11:31 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-18 00:49:11 +01:00
|
|
|
return rhs_result;
|
|
|
|
}
|
|
|
|
case LogicalOp::NullishCoalescing:
|
2020-10-02 16:00:15 +02:00
|
|
|
if (lhs_result.is_nullish()) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto rhs_result = m_rhs->execute(interpreter, global_object);
|
2020-04-18 00:49:11 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
return rhs_result;
|
|
|
|
}
|
|
|
|
return lhs_result;
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Reference Expression::to_reference(Interpreter&, GlobalObject&) const
|
2020-04-27 12:10:16 +02:00
|
|
|
{
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-07-02 21:54:56 +02:00
|
|
|
Reference Identifier::to_reference(Interpreter& interpreter, GlobalObject&) const
|
2020-04-27 12:37:27 +02:00
|
|
|
{
|
2021-07-02 21:54:56 +02:00
|
|
|
return interpreter.vm().resolve_binding(string());
|
2020-04-27 12:37:27 +02:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Reference MemberExpression::to_reference(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-27 12:10:16 +02:00
|
|
|
{
|
2021-07-03 00:20:52 +02:00
|
|
|
// 13.3.7.1 Runtime Semantics: Evaluation
|
|
|
|
// SuperProperty : super [ Expression ]
|
|
|
|
// SuperProperty : super . IdentifierName
|
|
|
|
// https://tc39.es/ecma262/#sec-super-keyword-runtime-semantics-evaluation
|
|
|
|
if (is<SuperExpression>(object())) {
|
|
|
|
// 1. Let env be GetThisEnvironment().
|
|
|
|
auto& environment = get_this_environment(interpreter.vm());
|
|
|
|
// 2. Let actualThis be ? env.GetThisBinding().
|
|
|
|
auto actual_this = environment.get_this_binding(global_object);
|
|
|
|
|
|
|
|
StringOrSymbol property_key;
|
|
|
|
|
|
|
|
if (is_computed()) {
|
|
|
|
// SuperProperty : super [ Expression ]
|
|
|
|
|
|
|
|
// 3. Let propertyNameReference be the result of evaluating Expression.
|
|
|
|
// 4. Let propertyNameValue be ? GetValue(propertyNameReference).
|
|
|
|
auto property_name_value = m_property->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
// 5. Let propertyKey be ? ToPropertyKey(propertyNameValue).
|
|
|
|
property_key = property_name_value.to_property_key(global_object);
|
|
|
|
} else {
|
|
|
|
// SuperProperty : super . IdentifierName
|
|
|
|
|
|
|
|
// 3. Let propertyKey be StringValue of IdentifierName.
|
|
|
|
VERIFY(is<Identifier>(property()));
|
|
|
|
property_key = static_cast<Identifier const&>(property()).string();
|
|
|
|
}
|
|
|
|
|
|
|
|
// 6. If the code matched by this SuperProperty is strict mode code, let strict be true; else let strict be false.
|
|
|
|
bool strict = interpreter.vm().in_strict_mode();
|
|
|
|
|
|
|
|
// 7. Return ? MakeSuperPropertyReference(actualThis, propertyKey, strict).
|
2021-09-21 21:52:09 +03:00
|
|
|
return TRY_OR_DISCARD(make_super_property_reference(global_object, actual_this, property_key, strict));
|
2021-07-03 00:20:52 +02:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto object_value = m_object->execute(interpreter, global_object);
|
2020-05-28 17:48:25 +01:00
|
|
|
if (interpreter.exception())
|
2020-04-27 12:10:16 +02:00
|
|
|
return {};
|
2021-06-25 16:27:59 +02:00
|
|
|
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
// From here on equivalent to
|
|
|
|
// 13.3.4 EvaluatePropertyAccessWithIdentifierKey ( baseValue, identifierName, strict ), https://tc39.es/ecma262/#sec-evaluate-property-access-with-identifier-key
|
|
|
|
|
2021-09-15 20:52:21 +01:00
|
|
|
object_value = TRY_OR_DISCARD(require_object_coercible(global_object, object_value));
|
2021-06-25 16:27:59 +02:00
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto property_name = computed_property_name(interpreter, global_object);
|
2020-04-27 12:10:16 +02:00
|
|
|
if (!property_name.is_valid())
|
2021-06-25 16:27:59 +02:00
|
|
|
return Reference {};
|
|
|
|
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
auto strict = interpreter.vm().in_strict_mode();
|
2021-09-11 19:03:38 +02:00
|
|
|
return Reference { object_value, move(property_name), {}, strict };
|
2020-04-27 12:10:16 +02:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value UnaryExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-08 23:27:18 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-09-27 18:36:49 +02:00
|
|
|
auto& vm = interpreter.vm();
|
2020-04-26 13:53:40 +02:00
|
|
|
if (m_op == UnaryOp::Delete) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto reference = m_lhs->to_reference(interpreter, global_object);
|
2020-04-26 13:53:40 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-06-08 04:00:53 +03:00
|
|
|
return Value(reference.delete_(global_object));
|
2020-04-26 13:53:40 +02:00
|
|
|
}
|
|
|
|
|
2020-06-02 23:26:39 +02:00
|
|
|
Value lhs_result;
|
2021-01-01 19:34:07 +01:00
|
|
|
if (m_op == UnaryOp::Typeof && is<Identifier>(*m_lhs)) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto reference = m_lhs->to_reference(interpreter, global_object);
|
2020-06-02 23:26:39 +02:00
|
|
|
if (interpreter.exception()) {
|
|
|
|
return {};
|
|
|
|
}
|
2021-06-25 16:27:59 +02:00
|
|
|
if (reference.is_unresolvable()) {
|
|
|
|
lhs_result = js_undefined();
|
|
|
|
} else {
|
2021-06-25 17:19:01 +02:00
|
|
|
lhs_result = reference.get_value(global_object, false);
|
2020-06-02 23:26:39 +02:00
|
|
|
}
|
|
|
|
} else {
|
2020-06-08 20:57:54 +02:00
|
|
|
lhs_result = m_lhs->execute(interpreter, global_object);
|
2020-06-02 23:26:39 +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-09-27 19:52:47 +02:00
|
|
|
return bitwise_not(global_object, 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-09-27 19:52:47 +02:00
|
|
|
return unary_plus(global_object, lhs_result);
|
2020-04-02 17:58:39 +01:00
|
|
|
case UnaryOp::Minus:
|
2020-09-27 19:52:47 +02:00
|
|
|
return unary_minus(global_object, lhs_result);
|
2020-03-18 06:33:32 +11:00
|
|
|
case UnaryOp::Typeof:
|
2021-04-02 20:33:03 +02:00
|
|
|
return js_string(vm, lhs_result.typeof());
|
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:
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-08 23:27:18 +02:00
|
|
|
}
|
|
|
|
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-08 23:27:18 +02:00
|
|
|
}
|
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
Value SuperExpression::execute(Interpreter&, GlobalObject&) const
|
2020-06-08 13:31:21 -05:00
|
|
|
{
|
2021-07-02 19:30:38 +02:00
|
|
|
// The semantics for SuperExpression are handled in CallExpression and SuperCall.
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
Value ClassMethod::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-06-08 13:31:21 -05:00
|
|
|
return m_function->execute(interpreter, global_object);
|
|
|
|
}
|
|
|
|
|
2021-08-28 17:11:05 +02:00
|
|
|
Value ClassField::execute(Interpreter& interpreter, GlobalObject&) const
|
|
|
|
{
|
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
Value ClassExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-10-13 23:49:19 +02:00
|
|
|
auto& vm = interpreter.vm();
|
2020-06-08 13:31:21 -05:00
|
|
|
Value class_constructor_value = m_constructor->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
update_function_name(class_constructor_value, m_name);
|
|
|
|
|
2021-09-24 22:40:38 +02:00
|
|
|
VERIFY(class_constructor_value.is_function() && is<ECMAScriptFunctionObject>(class_constructor_value.as_function()));
|
|
|
|
auto* class_constructor = static_cast<ECMAScriptFunctionObject*>(&class_constructor_value.as_function());
|
2020-11-11 21:37:40 +00:00
|
|
|
class_constructor->set_is_class_constructor();
|
2020-06-08 13:31:21 -05:00
|
|
|
Value super_constructor = js_undefined();
|
|
|
|
if (!m_super_class.is_null()) {
|
|
|
|
super_constructor = m_super_class->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-08-25 12:52:32 +02:00
|
|
|
if (!super_constructor.is_function() && !super_constructor.is_null()) {
|
2021-01-23 23:49:04 +01:00
|
|
|
interpreter.vm().throw_exception<TypeError>(global_object, ErrorType::ClassExtendsValueNotAConstructorOrNull, super_constructor.to_string_without_side_effects());
|
2020-08-25 12:52:32 +02:00
|
|
|
return {};
|
|
|
|
}
|
2021-09-24 23:34:13 +02:00
|
|
|
class_constructor->set_constructor_kind(ECMAScriptFunctionObject::ConstructorKind::Derived);
|
2020-06-08 13:31:21 -05:00
|
|
|
|
|
|
|
Object* super_constructor_prototype = nullptr;
|
|
|
|
if (!super_constructor.is_null()) {
|
2021-07-04 22:55:45 +01:00
|
|
|
auto super_constructor_prototype_value = super_constructor.as_object().get(vm.names.prototype);
|
2020-06-08 13:31:21 -05:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-01-23 23:49:04 +01:00
|
|
|
if (!super_constructor_prototype_value.is_object() && !super_constructor_prototype_value.is_null()) {
|
|
|
|
interpreter.vm().throw_exception<TypeError>(global_object, ErrorType::ClassExtendsValueInvalidPrototype, super_constructor_prototype_value.to_string_without_side_effects());
|
|
|
|
return {};
|
|
|
|
}
|
2021-01-24 00:15:35 +01:00
|
|
|
if (super_constructor_prototype_value.is_object())
|
2021-01-23 23:49:04 +01:00
|
|
|
super_constructor_prototype = &super_constructor_prototype_value.as_object();
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
2021-06-16 20:52:30 +01:00
|
|
|
auto* prototype = Object::create(global_object, super_constructor_prototype);
|
2020-06-08 13:31:21 -05:00
|
|
|
|
2021-07-06 02:15:08 +03:00
|
|
|
prototype->define_direct_property(vm.names.constructor, class_constructor, 0);
|
2020-06-08 13:31:21 -05:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-07-06 02:15:08 +03:00
|
|
|
class_constructor->define_direct_property(vm.names.prototype, prototype, Attribute::Writable);
|
2020-06-08 13:31:21 -05:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-09-28 23:54:42 +01:00
|
|
|
TRY_OR_DISCARD(class_constructor->internal_set_prototype_of(super_constructor.is_null() ? global_object.function_prototype() : &super_constructor.as_object()));
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
|
|
|
|
2020-10-13 23:49:19 +02:00
|
|
|
auto class_prototype = class_constructor->get(vm.names.prototype);
|
2020-06-08 13:31:21 -05:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2020-08-25 12:52:32 +02:00
|
|
|
if (!class_prototype.is_object()) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().throw_exception<TypeError>(global_object, ErrorType::NotAnObject, "Class prototype");
|
2020-08-25 12:52:32 +02:00
|
|
|
return {};
|
|
|
|
}
|
2021-08-28 17:11:05 +02:00
|
|
|
for (auto const& method : m_methods) {
|
2020-06-08 13:31:21 -05:00
|
|
|
auto method_value = method.execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-09-24 23:49:24 +02:00
|
|
|
auto& method_function = static_cast<ECMAScriptFunctionObject&>(method_value.as_function());
|
2020-06-08 13:31:21 -05:00
|
|
|
|
|
|
|
auto key = method.key().execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-06-17 01:32:08 +03:00
|
|
|
auto property_key = key.to_property_key(global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
auto& target = method.is_static() ? *class_constructor : class_prototype.as_object();
|
|
|
|
method_function.set_home_object(&target);
|
|
|
|
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
switch (method.kind()) {
|
|
|
|
case ClassMethod::Kind::Method:
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
target.define_property_or_throw(property_key, { .value = method_value, .writable = true, .enumerable = false, .configurable = true });
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
break;
|
|
|
|
case ClassMethod::Kind::Getter:
|
|
|
|
update_function_name(method_value, String::formatted("get {}", get_function_name(global_object, key)));
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
target.define_property_or_throw(property_key, { .get = &method_function, .enumerable = true, .configurable = true });
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
break;
|
|
|
|
case ClassMethod::Kind::Setter:
|
|
|
|
update_function_name(method_value, String::formatted("set {}", get_function_name(global_object, key)));
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
target.define_property_or_throw(property_key, { .set = &method_function, .enumerable = true, .configurable = true });
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
VERIFY_NOT_REACHED();
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-08-28 17:11:05 +02:00
|
|
|
for (auto& field : m_fields) {
|
|
|
|
auto key = field.key().execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
auto property_key = key.to_property_key(global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-09-24 23:49:24 +02:00
|
|
|
ECMAScriptFunctionObject* initializer = nullptr;
|
2021-08-28 17:11:05 +02:00
|
|
|
if (field.initializer()) {
|
|
|
|
auto copy_initializer = field.initializer();
|
|
|
|
auto body = create_ast_node<ExpressionStatement>(field.initializer()->source_range(), copy_initializer.release_nonnull());
|
|
|
|
// FIXME: A potential optimization is not creating the functions here since these are never directly accessible.
|
2021-09-24 22:40:38 +02:00
|
|
|
initializer = ECMAScriptFunctionObject::create(interpreter.global_object(), property_key.to_display_string(), *body, {}, 0, interpreter.lexical_environment(), FunctionKind::Regular, false);
|
2021-08-28 17:11:05 +02:00
|
|
|
initializer->set_home_object(field.is_static() ? class_constructor : &class_prototype.as_object());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (field.is_static()) {
|
|
|
|
Value field_value = js_undefined();
|
2021-09-23 20:56:28 +03:00
|
|
|
if (initializer)
|
|
|
|
field_value = TRY_OR_DISCARD(interpreter.vm().call(*initializer, class_constructor_value));
|
2021-08-28 17:11:05 +02:00
|
|
|
|
|
|
|
class_constructor->create_data_property_or_throw(property_key, field_value);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
} else {
|
|
|
|
class_constructor->add_field(property_key, initializer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
return class_constructor;
|
|
|
|
}
|
|
|
|
|
|
|
|
Value ClassDeclaration::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
Value class_constructor = m_class_expression->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-07-01 12:24:46 +02:00
|
|
|
interpreter.lexical_environment()->put_into_environment(m_class_expression->name(), { class_constructor, DeclarationKind::Let });
|
2020-06-08 13:31:21 -05:00
|
|
|
|
2021-03-16 09:12:34 +01:00
|
|
|
return {};
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
|
|
|
|
2020-03-07 19:42:11 +01:00
|
|
|
static void print_indent(int indent)
|
|
|
|
{
|
2020-12-06 16:55:19 +00:00
|
|
|
out("{}", String::repeated(' ', indent * 2));
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ASTNode::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", class_name());
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Variables)");
|
2020-04-13 16:42:54 +02:00
|
|
|
for (auto& variable : m_variables)
|
|
|
|
variable.dump(indent + 2);
|
|
|
|
}
|
|
|
|
if (!m_children.is_empty()) {
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Children)");
|
2020-04-13 16:42:54 +02:00
|
|
|
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;
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::StrictlyEquals:
|
2020-03-08 07:53:02 +02:00
|
|
|
op_string = "===";
|
|
|
|
break;
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::StrictlyInequals:
|
2020-03-08 23:27:18 +02:00
|
|
|
op_string = "!==";
|
|
|
|
break;
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::LooselyEquals:
|
2020-03-16 00:23:38 +02:00
|
|
|
op_string = "==";
|
|
|
|
break;
|
2021-09-24 00:06:10 +02:00
|
|
|
case BinaryOp::LooselyInequals:
|
2020-03-16 00:23:38 +02:00
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", class_name());
|
2020-03-08 07:53:02 +02:00
|
|
|
m_lhs->dump(indent + 1);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", op_string);
|
2020-03-08 07:53:02 +02:00
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", 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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", 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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", class_name());
|
2020-03-08 23:27:18 +02:00
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", op_string);
|
2020-03-08 23:27:18 +02:00
|
|
|
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);
|
2021-01-01 19:34:07 +01:00
|
|
|
if (is<NewExpression>(*this))
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("CallExpression [new]");
|
|
|
|
else
|
|
|
|
outln("CallExpression");
|
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)
|
2020-05-05 22:36:24 -07:00
|
|
|
argument.value->dump(indent + 1);
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|
|
|
|
|
2021-07-02 19:30:38 +02:00
|
|
|
void SuperCall::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
|
|
|
outln("SuperCall");
|
|
|
|
for (auto& argument : m_arguments)
|
|
|
|
argument.value->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
void ClassDeclaration::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
m_class_expression->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClassExpression::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("ClassExpression: \"{}\"", m_name);
|
2020-06-08 13:31:21 -05:00
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Constructor)");
|
2020-06-08 13:31:21 -05:00
|
|
|
m_constructor->dump(indent + 1);
|
|
|
|
|
|
|
|
if (!m_super_class.is_null()) {
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Super Class)");
|
2020-06-08 13:31:21 -05:00
|
|
|
m_super_class->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Methods)");
|
2020-06-08 13:31:21 -05:00
|
|
|
for (auto& method : m_methods)
|
|
|
|
method.dump(indent + 1);
|
2021-08-28 17:11:05 +02:00
|
|
|
|
|
|
|
print_indent(indent);
|
|
|
|
outln("(Fields)");
|
|
|
|
for (auto& field : m_fields)
|
|
|
|
field.dump(indent + 1);
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void ClassMethod::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Key)");
|
2020-06-08 13:31:21 -05:00
|
|
|
m_key->dump(indent + 1);
|
|
|
|
|
|
|
|
const char* kind_string = nullptr;
|
|
|
|
switch (m_kind) {
|
|
|
|
case Kind::Method:
|
|
|
|
kind_string = "Method";
|
|
|
|
break;
|
|
|
|
case Kind::Getter:
|
|
|
|
kind_string = "Getter";
|
|
|
|
break;
|
|
|
|
case Kind::Setter:
|
|
|
|
kind_string = "Setter";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("Kind: {}", kind_string);
|
2020-06-08 13:31:21 -05:00
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("Static: {}", m_is_static);
|
2020-06-08 13:31:21 -05:00
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Function)");
|
2020-06-08 13:31:21 -05:00
|
|
|
m_function->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2021-08-28 17:11:05 +02:00
|
|
|
void ClassField::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
print_indent(indent);
|
|
|
|
outln("(Key)");
|
|
|
|
m_key->dump(indent + 1);
|
|
|
|
|
|
|
|
print_indent(indent);
|
|
|
|
outln("Static: {}", m_is_static);
|
|
|
|
|
|
|
|
if (m_initializer) {
|
|
|
|
print_indent(indent);
|
|
|
|
outln("(Initializer)");
|
|
|
|
m_initializer->dump(indent + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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-12-06 16:55:19 +00:00
|
|
|
outln("StringLiteral \"{}\"", m_value);
|
2020-03-12 12:19:11 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
void SuperExpression::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("super");
|
2020-06-08 13:31:21 -05:00
|
|
|
}
|
|
|
|
|
2020-03-12 12:19:11 +01:00
|
|
|
void NumericLiteral::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("NumericLiteral {}", m_value);
|
2020-03-12 12:19:11 +01:00
|
|
|
}
|
|
|
|
|
2020-06-06 01:14:10 +01:00
|
|
|
void BigIntLiteral::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("BigIntLiteral {}", m_value);
|
2020-06-06 01:14:10 +01:00
|
|
|
}
|
|
|
|
|
2020-03-12 12:19:11 +01:00
|
|
|
void BooleanLiteral::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("BooleanLiteral {}", m_value);
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("null");
|
2020-03-15 23:32:34 +02:00
|
|
|
}
|
|
|
|
|
2021-05-29 16:03:19 +04:30
|
|
|
void BindingPattern::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
|
|
|
outln("BindingPattern {}", kind == Kind::Array ? "Array" : "Object");
|
2021-06-12 18:04:28 -07:00
|
|
|
|
|
|
|
for (auto& entry : entries) {
|
2021-05-29 16:03:19 +04:30
|
|
|
print_indent(indent + 1);
|
2021-06-12 18:04:28 -07:00
|
|
|
outln("(Property)");
|
|
|
|
|
|
|
|
if (kind == Kind::Object) {
|
|
|
|
print_indent(indent + 2);
|
|
|
|
outln("(Identifier)");
|
|
|
|
if (entry.name.has<NonnullRefPtr<Identifier>>()) {
|
|
|
|
entry.name.get<NonnullRefPtr<Identifier>>()->dump(indent + 3);
|
|
|
|
} else {
|
|
|
|
entry.name.get<NonnullRefPtr<Expression>>()->dump(indent + 3);
|
|
|
|
}
|
|
|
|
} else if (entry.is_elision()) {
|
2021-05-29 16:03:19 +04:30
|
|
|
print_indent(indent + 2);
|
2021-06-12 18:04:28 -07:00
|
|
|
outln("(Elision)");
|
|
|
|
continue;
|
2021-05-29 16:03:19 +04:30
|
|
|
}
|
|
|
|
|
2021-06-12 18:04:28 -07:00
|
|
|
print_indent(indent + 2);
|
|
|
|
outln("(Pattern{})", entry.is_rest ? " rest=true" : "");
|
|
|
|
if (entry.alias.has<NonnullRefPtr<Identifier>>()) {
|
|
|
|
entry.alias.get<NonnullRefPtr<Identifier>>()->dump(indent + 3);
|
|
|
|
} else if (entry.alias.has<NonnullRefPtr<BindingPattern>>()) {
|
|
|
|
entry.alias.get<NonnullRefPtr<BindingPattern>>()->dump(indent + 3);
|
2021-09-18 01:11:32 +02:00
|
|
|
} else if (entry.alias.has<NonnullRefPtr<MemberExpression>>()) {
|
|
|
|
entry.alias.get<NonnullRefPtr<MemberExpression>>()->dump(indent + 3);
|
2021-05-29 16:03:19 +04:30
|
|
|
} else {
|
2021-06-12 18:04:28 -07:00
|
|
|
print_indent(indent + 3);
|
|
|
|
outln("<empty>");
|
2021-05-29 16:03:19 +04:30
|
|
|
}
|
|
|
|
|
2021-06-12 18:04:28 -07:00
|
|
|
if (entry.initializer) {
|
|
|
|
print_indent(indent + 2);
|
|
|
|
outln("(Initializer)");
|
|
|
|
entry.initializer->dump(indent + 3);
|
|
|
|
}
|
2021-05-29 16:03:19 +04:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-11 02:13:06 +04:30
|
|
|
void FunctionNode::dump(int indent, String const& class_name) const
|
2020-03-07 19:42:11 +01:00
|
|
|
{
|
|
|
|
print_indent(indent);
|
2021-06-11 03:38:05 +04:30
|
|
|
outln("{}{} '{}'", class_name, m_kind == FunctionKind::Generator ? "*" : "", name());
|
2020-05-02 11:46:39 -07:00
|
|
|
if (!m_parameters.is_empty()) {
|
|
|
|
print_indent(indent + 1);
|
2021-06-03 22:49:37 +02:00
|
|
|
outln("(Parameters)");
|
2020-05-02 11:46:39 -07:00
|
|
|
|
|
|
|
for (auto& parameter : m_parameters) {
|
|
|
|
print_indent(indent + 2);
|
2020-05-04 16:05:13 +01:00
|
|
|
if (parameter.is_rest)
|
2020-12-06 16:55:19 +00:00
|
|
|
out("...");
|
2021-05-29 16:03:19 +04:30
|
|
|
parameter.binding.visit(
|
2021-06-11 02:13:06 +04:30
|
|
|
[&](FlyString const& name) {
|
2021-05-29 16:03:19 +04:30
|
|
|
outln("{}", name);
|
|
|
|
},
|
2021-06-11 02:13:06 +04:30
|
|
|
[&](BindingPattern const& pattern) {
|
2021-05-29 16:03:19 +04:30
|
|
|
pattern.dump(indent + 2);
|
|
|
|
});
|
2020-05-04 16:05:13 +01:00
|
|
|
if (parameter.default_value)
|
2020-05-02 11:46:39 -07:00
|
|
|
parameter.default_value->dump(indent + 3);
|
|
|
|
}
|
|
|
|
}
|
2020-04-13 16:42:54 +02:00
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Body)");
|
2020-04-13 16:42:54 +02:00
|
|
|
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());
|
|
|
|
}
|
|
|
|
|
2021-06-11 01:38:30 +04:30
|
|
|
void YieldExpression::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
if (argument())
|
|
|
|
argument()->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("If");
|
2020-03-08 07:58:58 +02:00
|
|
|
predicate().dump(indent + 1);
|
|
|
|
consequent().dump(indent + 1);
|
2020-03-21 18:40:17 +01:00
|
|
|
if (alternate()) {
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("Else");
|
2020-03-21 18:40:17 +01:00
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("While");
|
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-11-28 15:05:57 +01:00
|
|
|
void WithStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("Object");
|
2020-11-28 15:05:57 +01:00
|
|
|
object().dump(indent + 2);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("Body");
|
2020-11-28 15:05:57 +01:00
|
|
|
body().dump(indent + 2);
|
|
|
|
}
|
|
|
|
|
2020-04-04 21:29:23 +02:00
|
|
|
void DoWhileStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("DoWhile");
|
2020-04-04 21:29:23 +02:00
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("For");
|
2020-03-12 23:12:12 +11:00
|
|
|
if (init())
|
|
|
|
init()->dump(indent + 1);
|
|
|
|
if (test())
|
|
|
|
test()->dump(indent + 1);
|
|
|
|
if (update())
|
|
|
|
update()->dump(indent + 1);
|
|
|
|
body().dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2020-04-21 19:21:26 +01:00
|
|
|
void ForInStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("ForIn");
|
2020-04-21 19:21:26 +01:00
|
|
|
lhs().dump(indent + 1);
|
|
|
|
rhs().dump(indent + 1);
|
|
|
|
body().dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ForOfStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("ForOf");
|
2020-04-21 19:21:26 +01:00
|
|
|
lhs().dump(indent + 1);
|
|
|
|
rhs().dump(indent + 1);
|
|
|
|
body().dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2020-06-08 21:25:16 +02:00
|
|
|
Value Identifier::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-09 21:13:55 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-09-27 15:18:55 +02:00
|
|
|
auto value = interpreter.vm().get_variable(string(), global_object);
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-08-25 12:52:32 +02:00
|
|
|
if (value.is_empty()) {
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
interpreter.vm().throw_exception<ReferenceError>(global_object, ErrorType::UnknownIdentifier, string());
|
2020-08-25 12:52:32 +02:00
|
|
|
return {};
|
|
|
|
}
|
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);
|
2021-06-22 22:20:17 +02:00
|
|
|
outln("Identifier \"{}\"", m_string);
|
2020-03-09 21:13:55 +01:00
|
|
|
}
|
|
|
|
|
2020-04-26 23:05:37 -07:00
|
|
|
void SpreadExpression::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
m_target->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value SpreadExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-26 23:05:37 -07:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-06-08 20:57:54 +02:00
|
|
|
return m_target->execute(interpreter, global_object);
|
2020-04-26 23:05:37 -07:00
|
|
|
}
|
|
|
|
|
2020-09-29 16:41:28 +02:00
|
|
|
Value ThisExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-13 00:42:14 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2021-06-25 11:40:03 +02:00
|
|
|
return interpreter.vm().resolve_this_binding(global_object);
|
2020-04-13 00:42:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ThisExpression::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value AssignmentExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-09 21:13:55 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-07-11 01:16:17 +04:30
|
|
|
#define EXECUTE_LHS() \
|
|
|
|
do { \
|
|
|
|
if (auto* ptr = m_lhs.get_pointer<NonnullRefPtr<Expression>>()) { \
|
|
|
|
lhs_result = (*ptr)->execute(interpreter, global_object); \
|
|
|
|
if (interpreter.exception()) \
|
|
|
|
return {}; \
|
|
|
|
} \
|
|
|
|
} while (0)
|
|
|
|
|
2020-10-05 12:30:08 +01:00
|
|
|
#define EXECUTE_LHS_AND_RHS() \
|
|
|
|
do { \
|
2021-07-11 01:16:17 +04:30
|
|
|
EXECUTE_LHS(); \
|
2020-10-05 12:30:08 +01:00
|
|
|
rhs_result = m_rhs->execute(interpreter, global_object); \
|
|
|
|
if (interpreter.exception()) \
|
|
|
|
return {}; \
|
|
|
|
} while (0)
|
2020-03-09 21:13:55 +01:00
|
|
|
|
2020-04-06 20:24:45 +02:00
|
|
|
Value lhs_result;
|
2020-10-05 12:30:08 +01:00
|
|
|
Value rhs_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-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = add(global_object, 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-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = sub(global_object, 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-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = mul(global_object, 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-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = div(global_object, lhs_result, rhs_result);
|
2020-03-12 23:09:15 +11:00
|
|
|
break;
|
2020-05-04 23:07:05 +01:00
|
|
|
case AssignmentOp::ModuloAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = mod(global_object, lhs_result, rhs_result);
|
2020-05-04 23:07:05 +01:00
|
|
|
break;
|
2020-05-04 23:03:35 +01:00
|
|
|
case AssignmentOp::ExponentiationAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 18:36:49 +02:00
|
|
|
rhs_result = exp(global_object, lhs_result, rhs_result);
|
2020-05-04 23:03:35 +01:00
|
|
|
break;
|
2020-05-04 22:34:45 +01:00
|
|
|
case AssignmentOp::BitwiseAndAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = bitwise_and(global_object, lhs_result, rhs_result);
|
2020-05-04 22:34:45 +01:00
|
|
|
break;
|
|
|
|
case AssignmentOp::BitwiseOrAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = bitwise_or(global_object, lhs_result, rhs_result);
|
2020-05-04 22:34:45 +01:00
|
|
|
break;
|
|
|
|
case AssignmentOp::BitwiseXorAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = bitwise_xor(global_object, lhs_result, rhs_result);
|
2020-05-04 22:34:45 +01:00
|
|
|
break;
|
2020-04-23 13:36:14 +01:00
|
|
|
case AssignmentOp::LeftShiftAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = left_shift(global_object, lhs_result, rhs_result);
|
2020-04-23 13:36:14 +01:00
|
|
|
break;
|
2020-04-23 13:45:19 +01:00
|
|
|
case AssignmentOp::RightShiftAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = right_shift(global_object, lhs_result, rhs_result);
|
2020-04-23 13:45:19 +01:00
|
|
|
break;
|
2020-04-23 15:43:10 +01:00
|
|
|
case AssignmentOp::UnsignedRightShiftAssignment:
|
2020-10-05 12:30:08 +01:00
|
|
|
EXECUTE_LHS_AND_RHS();
|
2020-09-27 19:52:47 +02:00
|
|
|
rhs_result = unsigned_right_shift(global_object, lhs_result, rhs_result);
|
2020-04-23 15:43:10 +01:00
|
|
|
break;
|
2020-10-05 16:49:43 +01:00
|
|
|
case AssignmentOp::AndAssignment:
|
2021-07-11 01:16:17 +04:30
|
|
|
EXECUTE_LHS();
|
2020-10-05 16:49:43 +01:00
|
|
|
if (!lhs_result.to_boolean())
|
|
|
|
return lhs_result;
|
|
|
|
rhs_result = m_rhs->execute(interpreter, global_object);
|
|
|
|
break;
|
|
|
|
case AssignmentOp::OrAssignment:
|
2021-07-11 01:16:17 +04:30
|
|
|
EXECUTE_LHS();
|
2020-10-05 16:49:43 +01:00
|
|
|
if (lhs_result.to_boolean())
|
|
|
|
return lhs_result;
|
|
|
|
rhs_result = m_rhs->execute(interpreter, global_object);
|
|
|
|
break;
|
|
|
|
case AssignmentOp::NullishAssignment:
|
2021-07-11 01:16:17 +04:30
|
|
|
EXECUTE_LHS();
|
2020-10-05 16:49:43 +01:00
|
|
|
if (!lhs_result.is_nullish())
|
|
|
|
return lhs_result;
|
|
|
|
rhs_result = m_rhs->execute(interpreter, global_object);
|
|
|
|
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
|
|
|
|
2021-07-11 01:16:17 +04:30
|
|
|
return m_lhs.visit(
|
|
|
|
[&](NonnullRefPtr<Expression>& lhs) -> JS::Value {
|
|
|
|
auto reference = lhs->to_reference(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-27 12:56:09 +02:00
|
|
|
|
2021-07-11 01:16:17 +04:30
|
|
|
if (m_op == AssignmentOp::Assignment) {
|
|
|
|
rhs_result = m_rhs->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
2020-10-05 12:30:08 +01:00
|
|
|
|
2021-07-11 01:16:17 +04:30
|
|
|
if (reference.is_unresolvable()) {
|
|
|
|
interpreter.vm().throw_exception<ReferenceError>(global_object, ErrorType::InvalidLeftHandAssignment);
|
|
|
|
return {};
|
|
|
|
}
|
2021-03-21 17:14:20 +01:00
|
|
|
|
2021-07-11 01:16:17 +04:30
|
|
|
reference.put_value(global_object, rhs_result);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-04-02 20:33:03 +02:00
|
|
|
|
2021-07-11 01:16:17 +04:30
|
|
|
return rhs_result;
|
|
|
|
},
|
|
|
|
[&](NonnullRefPtr<BindingPattern>& pattern) -> JS::Value {
|
|
|
|
VERIFY(m_op == AssignmentOp::Assignment);
|
|
|
|
|
|
|
|
rhs_result = m_rhs->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
interpreter.vm().assign(pattern, rhs_result, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
return rhs_result;
|
|
|
|
});
|
2020-03-09 21:13:55 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value UpdateExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-12 13:45:45 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto reference = m_argument->to_reference(interpreter, global_object);
|
2021-06-25 16:27:59 +02:00
|
|
|
|
2020-04-28 14:44:48 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-06-25 17:19:01 +02:00
|
|
|
auto old_value = reference.get_value(global_object);
|
2020-04-21 23:27:11 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-09-27 18:36:49 +02:00
|
|
|
old_value = old_value.to_numeric(global_object);
|
2020-05-18 00:28:00 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-03-12 13:45:45 +02:00
|
|
|
|
2020-06-06 01:14:10 +01:00
|
|
|
Value new_value;
|
2020-03-12 13:45:45 +02:00
|
|
|
switch (m_op) {
|
|
|
|
case UpdateOp::Increment:
|
2020-06-06 01:14:10 +01:00
|
|
|
if (old_value.is_number())
|
|
|
|
new_value = Value(old_value.as_double() + 1);
|
|
|
|
else
|
2020-09-27 20:00:38 +02:00
|
|
|
new_value = js_bigint(interpreter.heap(), old_value.as_bigint().big_integer().plus(Crypto::SignedBigInteger { 1 }));
|
2020-03-12 13:45:45 +02:00
|
|
|
break;
|
|
|
|
case UpdateOp::Decrement:
|
2020-06-06 01:14:10 +01:00
|
|
|
if (old_value.is_number())
|
|
|
|
new_value = Value(old_value.as_double() - 1);
|
|
|
|
else
|
2020-09-27 20:00:38 +02:00
|
|
|
new_value = js_bigint(interpreter.heap(), old_value.as_bigint().big_integer().minus(Crypto::SignedBigInteger { 1 }));
|
2020-03-12 14:24:34 +02:00
|
|
|
break;
|
2020-04-21 23:27:11 +01:00
|
|
|
default:
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-12 13:45:45 +02:00
|
|
|
}
|
|
|
|
|
2021-06-25 17:19:01 +02:00
|
|
|
reference.put_value(global_object, new_value);
|
2020-04-28 14:44:48 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-21 23:27:11 +01:00
|
|
|
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-05-04 23:07:05 +01:00
|
|
|
case AssignmentOp::ModuloAssignment:
|
|
|
|
op_string = "%=";
|
|
|
|
break;
|
2020-05-04 23:03:35 +01:00
|
|
|
case AssignmentOp::ExponentiationAssignment:
|
|
|
|
op_string = "**=";
|
|
|
|
break;
|
2020-05-04 22:34:45 +01:00
|
|
|
case AssignmentOp::BitwiseAndAssignment:
|
|
|
|
op_string = "&=";
|
|
|
|
break;
|
|
|
|
case AssignmentOp::BitwiseOrAssignment:
|
|
|
|
op_string = "|=";
|
|
|
|
break;
|
|
|
|
case AssignmentOp::BitwiseXorAssignment:
|
|
|
|
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-10-05 16:49:43 +01:00
|
|
|
case AssignmentOp::AndAssignment:
|
|
|
|
op_string = "&&=";
|
|
|
|
break;
|
|
|
|
case AssignmentOp::OrAssignment:
|
|
|
|
op_string = "||=";
|
|
|
|
break;
|
|
|
|
case AssignmentOp::NullishAssignment:
|
|
|
|
op_string = "\?\?=";
|
|
|
|
break;
|
2020-03-09 21:13:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", op_string);
|
2021-07-11 01:16:17 +04:30
|
|
|
m_lhs.visit([&](auto& lhs) { lhs->dump(indent + 1); });
|
2020-03-09 21:13:55 +01:00
|
|
|
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);
|
2020-10-19 00:17:46 +01:00
|
|
|
if (m_prefixed) {
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", op_string);
|
2020-10-19 00:17:46 +01:00
|
|
|
}
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{}", op_string);
|
2020-03-14 20:44:57 +02:00
|
|
|
}
|
2020-03-12 13:45:45 +02:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value VariableDeclaration::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-09 21:13:55 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-04-04 21:46:25 +02:00
|
|
|
for (auto& declarator : m_declarations) {
|
|
|
|
if (auto* init = declarator.init()) {
|
2021-06-21 19:23:06 +02:00
|
|
|
auto initializer_result = init->execute(interpreter, global_object);
|
2020-04-04 21:46:25 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-05-29 16:03:19 +04:30
|
|
|
declarator.target().visit(
|
2021-06-11 02:13:06 +04:30
|
|
|
[&](NonnullRefPtr<Identifier> const& id) {
|
2021-05-29 16:03:19 +04:30
|
|
|
auto variable_name = id->string();
|
|
|
|
if (is<ClassExpression>(*init))
|
2021-06-21 19:23:06 +02:00
|
|
|
update_function_name(initializer_result, variable_name);
|
|
|
|
interpreter.vm().set_variable(variable_name, initializer_result, global_object, true);
|
2021-05-29 16:03:19 +04:30
|
|
|
},
|
2021-06-11 02:13:06 +04:30
|
|
|
[&](NonnullRefPtr<BindingPattern> const& pattern) {
|
2021-06-21 19:23:06 +02:00
|
|
|
interpreter.vm().assign(pattern, initializer_result, global_object, true);
|
2021-05-29 16:03:19 +04:30
|
|
|
});
|
2020-04-04 21:46:25 +02:00
|
|
|
}
|
2020-03-09 21:13:55 +01:00
|
|
|
}
|
2021-03-16 09:12:34 +01:00
|
|
|
return {};
|
2020-03-09 21:13:55 +01:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value VariableDeclarator::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-04-04 21:46:25 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-10-18 17:44:55 +01:00
|
|
|
// NOTE: VariableDeclarator execution is handled by VariableDeclaration.
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-04-04 21:46:25 +02:00
|
|
|
}
|
|
|
|
|
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-12-06 16:55:19 +00:00
|
|
|
outln("{}", 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);
|
2021-05-29 16:03:19 +04:30
|
|
|
m_target.visit([indent](const auto& value) { value->dump(indent + 1); });
|
2020-04-04 21:46:25 +02:00
|
|
|
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-12-28 20:45:22 +03:30
|
|
|
Value ObjectProperty::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-04-23 19:37:53 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-04-23 19:37:53 +01:00
|
|
|
// NOTE: ObjectProperty execution is handled by ObjectExpression.
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-04-23 19:37:53 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ObjectExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-09 21:28:31 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-06-16 20:52:30 +01:00
|
|
|
auto* object = Object::create(global_object, global_object.object_prototype());
|
2020-04-23 19:37:53 +01:00
|
|
|
for (auto& property : m_properties) {
|
2020-07-07 21:38:46 -07:00
|
|
|
auto key = property.key().execute(interpreter, global_object);
|
2020-04-23 19:37:53 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-27 21:52:47 -07:00
|
|
|
|
2020-05-21 17:28:28 -07:00
|
|
|
if (property.type() == ObjectProperty::Type::Spread) {
|
2021-07-05 18:58:51 +01:00
|
|
|
if (key.is_object() && is<Array>(key.as_object())) {
|
2020-07-07 21:38:46 -07:00
|
|
|
auto& array_to_spread = static_cast<Array&>(key.as_object());
|
LibJS: Object index properties have descriptors; Handle sparse indices
This patch adds an IndexedProperties object for storing indexed
properties within an Object. This accomplishes two goals: indexed
properties now have an associated descriptor, and objects now gracefully
handle sparse properties.
The IndexedProperties class is a wrapper around two other classes, one
for simple indexed properties storage, and one for general indexed
property storage. Simple indexed property storage is the common-case,
and is simply a vector of properties which all have attributes of
default_attributes (writable, enumerable, and configurable).
General indexed property storage is for a collection of indexed
properties where EITHER one or more properties have attributes other
than default_attributes OR there is a property with a large index (in
particular, large is '200' or higher).
Indexed properties are now treated relatively the same as storage within
the various Object methods. Additionally, there is a custom iterator
class for IndexedProperties which makes iteration easy. The iterator
skips empty values by default, but can be configured otherwise.
Likewise, it evaluates getters by default, but can be set not to.
2020-05-27 11:35:09 -07:00
|
|
|
for (auto& entry : array_to_spread.indexed_properties()) {
|
LibJS: Rewrite most of Object for spec compliance :^)
This is a huge patch, I know. In hindsight this perhaps could've been
done slightly more incremental, but I started and then fixed everything
until it worked, and here we are. I tried splitting of some completely
unrelated changes into separate commits, however. Anyway.
This is a rewrite of most of Object, and by extension large parts of
Array, Proxy, Reflect, String, TypedArray, and some other things.
What we already had worked fine for about 90% of things, but getting the
last 10% right proved to be increasingly difficult with the current code
that sort of grew organically and is only very loosely based on the
spec - this became especially obvious when we started fixing a large
number of test262 failures.
Key changes include:
- 1:1 matching function names and parameters of all object-related
functions, to avoid ambiguity. Previously we had things like put(),
which the spec doesn't have - as a result it wasn't always clear which
need to be used.
- Better separation between object abstract operations and internal
methods - the former are always the same, the latter can be overridden
(and are therefore virtual). The internal methods (i.e. [[Foo]] in the
spec) are now prefixed with 'internal_' for clarity - again, it was
previously not always clear which AO a certain method represents,
get() could've been both Get and [[Get]] (I don't know which one it
was closer to right now).
Note that some of the old names have been kept until all code relying
on them is updated, but they are now simple wrappers around the
closest matching standard abstract operation.
- Simplifications of the storage layer: functions that write values to
storage are now prefixed with 'storage_' to make their purpose clear,
and as they are not part of the spec they should not contain any steps
specified by it. Much functionality is now covered by the layers above
it and was removed (e.g. handling of accessors, attribute checks).
- PropertyAttributes has been greatly simplified, and is being replaced
by PropertyDescriptor - a concept similar to the current
implementation, but more aligned with the actual spec. See the commit
message of the previous commit where it was introduced for details.
- As a bonus, and since I had to look at the spec a whole lot anyway, I
introduced more inline comments with the exact steps from the spec -
this makes it super easy to verify correctness.
- East-const all the things.
As a result of all of this, things are much more correct but a bit
slower now. Retaining speed wasn't a consideration at all, I have done
no profiling of the new code - there might be low hanging fruits, which
we can then harvest separately.
Special thanks to Idan for helping me with this by tracking down bugs,
updating everything outside of LibJS to work with these changes (LibWeb,
Spreadsheet, HackStudio), as well as providing countless patches to fix
regressions I introduced - there still are very few (we got it down to
5), but we also get many new passing test262 tests in return. :^)
Co-authored-by: Idan Horowitz <idan.horowitz@gmail.com>
2021-07-04 18:14:16 +01:00
|
|
|
auto value = array_to_spread.get(entry.index());
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
object->indexed_properties().put(entry.index(), value);
|
LibJS: Object index properties have descriptors; Handle sparse indices
This patch adds an IndexedProperties object for storing indexed
properties within an Object. This accomplishes two goals: indexed
properties now have an associated descriptor, and objects now gracefully
handle sparse properties.
The IndexedProperties class is a wrapper around two other classes, one
for simple indexed properties storage, and one for general indexed
property storage. Simple indexed property storage is the common-case,
and is simply a vector of properties which all have attributes of
default_attributes (writable, enumerable, and configurable).
General indexed property storage is for a collection of indexed
properties where EITHER one or more properties have attributes other
than default_attributes OR there is a property with a large index (in
particular, large is '200' or higher).
Indexed properties are now treated relatively the same as storage within
the various Object methods. Additionally, there is a custom iterator
class for IndexedProperties which makes iteration easy. The iterator
skips empty values by default, but can be configured otherwise.
Likewise, it evaluates getters by default, but can be set not to.
2020-05-27 11:35:09 -07:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-27 21:52:47 -07:00
|
|
|
}
|
2020-07-07 21:38:46 -07:00
|
|
|
} else if (key.is_object()) {
|
|
|
|
auto& obj_to_spread = key.as_object();
|
2020-04-27 21:52:47 -07:00
|
|
|
|
2020-04-28 19:19:31 -07:00
|
|
|
for (auto& it : obj_to_spread.shape().property_table_ordered()) {
|
2020-06-07 10:53:14 -07:00
|
|
|
if (it.value.attributes.is_enumerable()) {
|
2021-07-06 02:15:08 +03:00
|
|
|
object->define_direct_property(it.key, obj_to_spread.get(it.key), JS::default_attributes);
|
2020-06-07 10:53:14 -07:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
2020-04-27 21:52:47 -07:00
|
|
|
}
|
2020-07-07 21:38:46 -07:00
|
|
|
} else if (key.is_string()) {
|
|
|
|
auto& str_to_spread = key.as_string().string();
|
2020-04-27 21:52:47 -07:00
|
|
|
|
|
|
|
for (size_t i = 0; i < str_to_spread.length(); i++) {
|
2021-07-06 02:15:08 +03:00
|
|
|
object->define_direct_property(i, js_string(interpreter.heap(), str_to_spread.substring(i, 1)), JS::default_attributes);
|
2020-06-07 10:53:14 -07:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-27 21:52:47 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto value = property.value().execute(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-05-21 17:28:28 -07:00
|
|
|
|
2020-06-08 13:31:21 -05:00
|
|
|
if (value.is_function() && property.is_method())
|
2021-09-24 23:49:24 +02:00
|
|
|
static_cast<ECMAScriptFunctionObject&>(value.as_function()).set_home_object(object);
|
2020-06-08 13:31:21 -05:00
|
|
|
|
2020-09-27 18:36:49 +02:00
|
|
|
String name = get_function_name(global_object, key);
|
2020-05-21 17:28:28 -07:00
|
|
|
if (property.type() == ObjectProperty::Type::Getter) {
|
2020-10-04 15:18:52 +01:00
|
|
|
name = String::formatted("get {}", name);
|
2020-05-21 17:28:28 -07:00
|
|
|
} else if (property.type() == ObjectProperty::Type::Setter) {
|
2020-10-04 15:18:52 +01:00
|
|
|
name = String::formatted("set {}", name);
|
2020-05-21 17:28:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
update_function_name(value, name);
|
|
|
|
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
switch (property.type()) {
|
|
|
|
case ObjectProperty::Type::Getter:
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(value.is_function());
|
2021-07-06 02:15:08 +03:00
|
|
|
object->define_direct_accessor(PropertyName::from_value(global_object, key), &value.as_function(), nullptr, Attribute::Configurable | Attribute::Enumerable);
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
break;
|
|
|
|
case ObjectProperty::Type::Setter:
|
|
|
|
VERIFY(value.is_function());
|
2021-07-06 02:15:08 +03:00
|
|
|
object->define_direct_accessor(PropertyName::from_value(global_object, key), nullptr, &value.as_function(), Attribute::Configurable | Attribute::Enumerable);
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
break;
|
|
|
|
case ObjectProperty::Type::KeyValue:
|
2021-07-06 02:15:08 +03:00
|
|
|
object->define_direct_property(PropertyName::from_value(global_object, key), value, JS::default_attributes);
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
break;
|
|
|
|
case ObjectProperty::Type::Spread:
|
|
|
|
default:
|
|
|
|
VERIFY_NOT_REACHED();
|
2020-05-21 17:28:28 -07:00
|
|
|
}
|
LibJS: Update Object::define_accessor() to take both getter and setter
This replaces the current 'function plus boolean indicating the type'
API, which makes it easier to set both getter and setter at once.
This was already possible before but required two calls of this
function, which wasn't intuitive:
define_accessor(name, getter, true, ...);
define_accessor(name, setter, false, ...);
Which now becomes:
define_accessor(name, getter, setter, ...);
2021-04-10 18:35:29 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
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);
|
2021-04-24 13:49:43 +02:00
|
|
|
outln("{}(computed={})", class_name(), is_computed());
|
2020-03-11 18:58:19 +01:00
|
|
|
m_object->dump(indent + 1);
|
|
|
|
m_property->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
PropertyName MemberExpression::computed_property_name(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-20 20:51:03 +01:00
|
|
|
{
|
2021-06-24 21:13:09 +02:00
|
|
|
if (!is_computed())
|
|
|
|
return verify_cast<Identifier>(*m_property).string();
|
|
|
|
|
2020-10-07 19:54:36 +01:00
|
|
|
auto value = m_property->execute(interpreter, global_object);
|
2020-05-15 13:39:24 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY(!value.is_empty());
|
2020-10-07 19:54:36 +01:00
|
|
|
return PropertyName::from_value(global_object, value);
|
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>";
|
2021-01-01 19:34:07 +01:00
|
|
|
if (is<Identifier>(*m_object))
|
2021-06-11 02:13:06 +04:30
|
|
|
object_string = static_cast<Identifier const&>(*m_object).string();
|
2020-04-19 01:12:51 +01:00
|
|
|
if (is_computed())
|
2020-10-04 15:18:52 +01:00
|
|
|
return String::formatted("{}[<computed>]", object_string);
|
2021-06-24 21:13:09 +02:00
|
|
|
return String::formatted("{}.{}", object_string, verify_cast<Identifier>(*m_property).string());
|
2020-04-19 01:12:51 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value MemberExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-11 18:58:19 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-04-14 22:14:54 +02:00
|
|
|
auto reference = to_reference(interpreter, global_object);
|
2020-03-27 15:35:35 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2021-06-25 17:19:01 +02:00
|
|
|
return reference.get_value(global_object);
|
2020-03-11 18:58:19 +01:00
|
|
|
}
|
|
|
|
|
2021-09-14 06:56:31 +04:30
|
|
|
void OptionalChain::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
|
|
|
outln("{}", class_name());
|
|
|
|
m_base->dump(indent + 1);
|
|
|
|
for (auto& reference : m_references) {
|
|
|
|
reference.visit(
|
|
|
|
[&](Call const& call) {
|
|
|
|
print_indent(indent + 1);
|
|
|
|
outln("Call({})", call.mode == Mode::Optional ? "Optional" : "Not Optional");
|
|
|
|
for (auto& argument : call.arguments)
|
|
|
|
argument.value->dump(indent + 2);
|
|
|
|
},
|
|
|
|
[&](ComputedReference const& ref) {
|
|
|
|
print_indent(indent + 1);
|
|
|
|
outln("ComputedReference({})", ref.mode == Mode::Optional ? "Optional" : "Not Optional");
|
|
|
|
ref.expression->dump(indent + 2);
|
|
|
|
},
|
|
|
|
[&](MemberReference const& ref) {
|
|
|
|
print_indent(indent + 1);
|
|
|
|
outln("MemberReference({})", ref.mode == Mode::Optional ? "Optional" : "Not Optional");
|
|
|
|
ref.identifier->dump(indent + 2);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Optional<OptionalChain::ReferenceAndValue> OptionalChain::to_reference_and_value(JS::Interpreter& interpreter, JS::GlobalObject& global_object) const
|
|
|
|
{
|
|
|
|
// Note: This is wrapped in an optional to allow base_reference = ...
|
|
|
|
Optional<JS::Reference> base_reference = m_base->to_reference(interpreter, global_object);
|
|
|
|
auto base = base_reference->is_unresolvable() ? m_base->execute(interpreter, global_object) : base_reference->get_value(global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
for (auto& reference : m_references) {
|
|
|
|
auto is_optional = reference.visit([](auto& ref) { return ref.mode; }) == Mode::Optional;
|
|
|
|
if (is_optional && base.is_nullish())
|
|
|
|
return ReferenceAndValue { {}, js_undefined() };
|
|
|
|
|
|
|
|
auto expression = reference.visit(
|
|
|
|
[&](Call const& call) -> NonnullRefPtr<Expression> {
|
|
|
|
return create_ast_node<CallExpression>(source_range(),
|
|
|
|
create_ast_node<SyntheticReferenceExpression>(source_range(), *base_reference, base),
|
|
|
|
call.arguments);
|
|
|
|
},
|
|
|
|
[&](ComputedReference const& ref) -> NonnullRefPtr<Expression> {
|
|
|
|
return create_ast_node<MemberExpression>(source_range(),
|
|
|
|
create_ast_node<SyntheticReferenceExpression>(source_range(), *base_reference, base),
|
|
|
|
ref.expression,
|
|
|
|
true);
|
|
|
|
},
|
|
|
|
[&](MemberReference const& ref) -> NonnullRefPtr<Expression> {
|
|
|
|
return create_ast_node<MemberExpression>(source_range(),
|
|
|
|
create_ast_node<SyntheticReferenceExpression>(source_range(), *base_reference, base),
|
|
|
|
ref.identifier,
|
|
|
|
false);
|
|
|
|
});
|
|
|
|
if (is<CallExpression>(*expression)) {
|
|
|
|
base_reference = JS::Reference {};
|
|
|
|
base = expression->execute(interpreter, global_object);
|
|
|
|
} else {
|
|
|
|
base_reference = expression->to_reference(interpreter, global_object);
|
|
|
|
base = base_reference->get_value(global_object);
|
|
|
|
}
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
return ReferenceAndValue { base_reference.release_value(), base };
|
|
|
|
}
|
|
|
|
|
|
|
|
Value OptionalChain::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
|
|
|
if (auto result = to_reference_and_value(interpreter, global_object); result.has_value())
|
|
|
|
return result.release_value().value;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
JS::Reference OptionalChain::to_reference(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
|
|
|
if (auto result = to_reference_and_value(interpreter, global_object); result.has_value())
|
|
|
|
return result.release_value().reference;
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-11-02 21:27:42 +00:00
|
|
|
void MetaProperty::dump(int indent) const
|
|
|
|
{
|
|
|
|
String name;
|
|
|
|
if (m_type == MetaProperty::Type::NewTarget)
|
|
|
|
name = "new.target";
|
|
|
|
else if (m_type == MetaProperty::Type::ImportMeta)
|
|
|
|
name = "import.meta";
|
|
|
|
else
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-11-02 21:27:42 +00:00
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("{} {}", class_name(), name);
|
2020-11-02 21:27:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Value MetaProperty::execute(Interpreter& interpreter, GlobalObject&) const
|
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-11-02 21:27:42 +00:00
|
|
|
if (m_type == MetaProperty::Type::NewTarget)
|
|
|
|
return interpreter.vm().get_new_target().value_or(js_undefined());
|
|
|
|
if (m_type == MetaProperty::Type::ImportMeta)
|
|
|
|
TODO();
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-11-02 21:27:42 +00:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value StringLiteral::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-12 12:19:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-09-27 20:03:42 +02:00
|
|
|
return js_string(interpreter.heap(), m_value);
|
2020-03-12 12:19:11 +01:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value NumericLiteral::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-12 12:19:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-03-12 12:19:11 +01:00
|
|
|
return Value(m_value);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value BigIntLiteral::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-06-06 01:14:10 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2021-06-14 02:19:23 +03:00
|
|
|
Crypto::SignedBigInteger integer;
|
|
|
|
if (m_value[0] == '0' && m_value.length() >= 3) {
|
|
|
|
if (m_value[1] == 'x' || m_value[1] == 'X') {
|
2021-06-29 17:51:52 +03:00
|
|
|
return js_bigint(interpreter.heap(), Crypto::SignedBigInteger::from_base(16, m_value.substring(2, m_value.length() - 3)));
|
2021-06-14 02:19:23 +03:00
|
|
|
} else if (m_value[1] == 'o' || m_value[1] == 'O') {
|
2021-06-29 17:51:52 +03:00
|
|
|
return js_bigint(interpreter.heap(), Crypto::SignedBigInteger::from_base(8, m_value.substring(2, m_value.length() - 3)));
|
2021-06-14 02:19:23 +03:00
|
|
|
} else if (m_value[1] == 'b' || m_value[1] == 'B') {
|
2021-06-29 17:51:52 +03:00
|
|
|
return js_bigint(interpreter.heap(), Crypto::SignedBigInteger::from_base(2, m_value.substring(2, m_value.length() - 3)));
|
2021-06-14 02:19:23 +03:00
|
|
|
}
|
|
|
|
}
|
2021-06-29 17:51:52 +03:00
|
|
|
return js_bigint(interpreter.heap(), Crypto::SignedBigInteger::from_base(10, m_value.substring(0, m_value.length() - 1)));
|
2020-06-06 01:14:10 +01:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value BooleanLiteral::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-12 12:19:11 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-03-12 12:19:11 +01:00
|
|
|
return Value(m_value);
|
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value NullLiteral::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-15 23:32:34 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-03-15 23:32:34 +02:00
|
|
|
return js_null();
|
|
|
|
}
|
|
|
|
|
2020-06-03 16:05:49 -07:00
|
|
|
void RegExpLiteral::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2021-05-10 11:56:08 +01:00
|
|
|
outln("{} (/{}/{})", class_name(), pattern(), flags());
|
2020-06-03 16:05:49 -07:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value RegExpLiteral::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-06-03 16:05:49 -07:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2021-07-29 10:34:37 -04:00
|
|
|
|
|
|
|
Regex<ECMA262> regex(parsed_regex(), parsed_pattern(), parsed_flags());
|
|
|
|
return RegExpObject::create(global_object, move(regex), pattern(), flags());
|
2020-06-03 16:05:49 -07:00
|
|
|
}
|
|
|
|
|
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);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("<empty>");
|
2020-04-15 20:09:06 +01:00
|
|
|
}
|
2020-03-20 20:29:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ArrayExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-20 20:29:57 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-07-04 01:36:44 +03:00
|
|
|
auto* array = Array::create(global_object, 0);
|
2021-09-06 03:29:52 +04:30
|
|
|
array->indexed_properties();
|
|
|
|
size_t index = 0;
|
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) {
|
2020-06-08 20:57:54 +02:00
|
|
|
value = element->execute(interpreter, global_object);
|
2020-04-15 20:09:06 +01:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-04-26 23:05:37 -07:00
|
|
|
|
2021-01-01 19:34:07 +01:00
|
|
|
if (is<SpreadExpression>(*element)) {
|
2020-09-08 14:15:13 +02:00
|
|
|
get_iterator_values(global_object, value, [&](Value iterator_value) {
|
2021-09-06 03:29:52 +04:30
|
|
|
array->indexed_properties().put(index++, iterator_value, default_attributes);
|
2020-07-13 08:27:20 -07:00
|
|
|
return IterationDecision::Continue;
|
|
|
|
});
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
continue;
|
2020-04-26 23:05:37 -07:00
|
|
|
}
|
2020-04-15 20:09:06 +01:00
|
|
|
}
|
2021-09-06 03:29:52 +04:30
|
|
|
array->indexed_properties().put(index++, value, default_attributes);
|
2020-03-20 20:29:57 +01:00
|
|
|
}
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
void TemplateLiteral::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
2020-05-06 10:17:35 +01:00
|
|
|
for (auto& expression : m_expressions)
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
expression.dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value TemplateLiteral::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
StringBuilder string_builder;
|
|
|
|
|
2020-05-06 10:17:35 +01:00
|
|
|
for (auto& expression : m_expressions) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto expr = expression.execute(interpreter, global_object);
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
2020-09-27 18:36:49 +02:00
|
|
|
auto string = expr.to_string(global_object);
|
2020-05-15 13:39:24 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
string_builder.append(string);
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
}
|
|
|
|
|
2020-09-27 20:03:42 +02:00
|
|
|
return js_string(interpreter.heap(), string_builder.build());
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
}
|
|
|
|
|
2020-05-06 10:17:35 +01:00
|
|
|
void TaggedTemplateLiteral::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Tag)");
|
2020-05-06 10:17:35 +01:00
|
|
|
m_tag->dump(indent + 2);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Template Literal)");
|
2020-05-06 10:17:35 +01:00
|
|
|
m_template_literal->dump(indent + 2);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value TaggedTemplateLiteral::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-05-06 10:17:35 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-10-13 23:49:19 +02:00
|
|
|
auto& vm = interpreter.vm();
|
2020-06-08 20:57:54 +02:00
|
|
|
auto tag = m_tag->execute(interpreter, global_object);
|
2020-10-13 23:49:19 +02:00
|
|
|
if (vm.exception())
|
2020-05-06 10:17:35 +01:00
|
|
|
return {};
|
|
|
|
if (!tag.is_function()) {
|
2020-10-13 23:49:19 +02:00
|
|
|
vm.throw_exception<TypeError>(global_object, ErrorType::NotAFunction, tag.to_string_without_side_effects());
|
2020-05-06 10:17:35 +01:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
auto& tag_function = tag.as_function();
|
|
|
|
auto& expressions = m_template_literal->expressions();
|
2021-07-04 01:36:44 +03:00
|
|
|
auto* strings = Array::create(global_object, 0);
|
2020-10-13 23:49:19 +02:00
|
|
|
MarkedValueList arguments(vm.heap());
|
2020-05-06 10:17:35 +01:00
|
|
|
arguments.append(strings);
|
|
|
|
for (size_t i = 0; i < expressions.size(); ++i) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto value = expressions[i].execute(interpreter, global_object);
|
2020-10-13 23:49:19 +02:00
|
|
|
if (vm.exception())
|
2020-05-06 10:17:35 +01:00
|
|
|
return {};
|
|
|
|
// tag`${foo}` -> "", foo, "" -> tag(["", ""], foo)
|
|
|
|
// tag`foo${bar}baz${qux}` -> "foo", bar, "baz", qux, "" -> tag(["foo", "baz", ""], bar, qux)
|
LibJS: Object index properties have descriptors; Handle sparse indices
This patch adds an IndexedProperties object for storing indexed
properties within an Object. This accomplishes two goals: indexed
properties now have an associated descriptor, and objects now gracefully
handle sparse properties.
The IndexedProperties class is a wrapper around two other classes, one
for simple indexed properties storage, and one for general indexed
property storage. Simple indexed property storage is the common-case,
and is simply a vector of properties which all have attributes of
default_attributes (writable, enumerable, and configurable).
General indexed property storage is for a collection of indexed
properties where EITHER one or more properties have attributes other
than default_attributes OR there is a property with a large index (in
particular, large is '200' or higher).
Indexed properties are now treated relatively the same as storage within
the various Object methods. Additionally, there is a custom iterator
class for IndexedProperties which makes iteration easy. The iterator
skips empty values by default, but can be configured otherwise.
Likewise, it evaluates getters by default, but can be set not to.
2020-05-27 11:35:09 -07:00
|
|
|
if (i % 2 == 0) {
|
|
|
|
strings->indexed_properties().append(value);
|
|
|
|
} else {
|
2020-05-06 10:17:35 +01:00
|
|
|
arguments.append(value);
|
LibJS: Object index properties have descriptors; Handle sparse indices
This patch adds an IndexedProperties object for storing indexed
properties within an Object. This accomplishes two goals: indexed
properties now have an associated descriptor, and objects now gracefully
handle sparse properties.
The IndexedProperties class is a wrapper around two other classes, one
for simple indexed properties storage, and one for general indexed
property storage. Simple indexed property storage is the common-case,
and is simply a vector of properties which all have attributes of
default_attributes (writable, enumerable, and configurable).
General indexed property storage is for a collection of indexed
properties where EITHER one or more properties have attributes other
than default_attributes OR there is a property with a large index (in
particular, large is '200' or higher).
Indexed properties are now treated relatively the same as storage within
the various Object methods. Additionally, there is a custom iterator
class for IndexedProperties which makes iteration easy. The iterator
skips empty values by default, but can be configured otherwise.
Likewise, it evaluates getters by default, but can be set not to.
2020-05-27 11:35:09 -07:00
|
|
|
}
|
2020-05-06 10:17:35 +01:00
|
|
|
}
|
2020-05-06 16:34:14 -07:00
|
|
|
|
2021-07-04 01:36:44 +03:00
|
|
|
auto* raw_strings = Array::create(global_object, 0);
|
2020-05-06 16:34:14 -07:00
|
|
|
for (auto& raw_string : m_template_literal->raw_strings()) {
|
2020-06-08 20:57:54 +02:00
|
|
|
auto value = raw_string.execute(interpreter, global_object);
|
2020-10-13 23:49:19 +02:00
|
|
|
if (vm.exception())
|
2020-05-06 16:34:14 -07:00
|
|
|
return {};
|
LibJS: Object index properties have descriptors; Handle sparse indices
This patch adds an IndexedProperties object for storing indexed
properties within an Object. This accomplishes two goals: indexed
properties now have an associated descriptor, and objects now gracefully
handle sparse properties.
The IndexedProperties class is a wrapper around two other classes, one
for simple indexed properties storage, and one for general indexed
property storage. Simple indexed property storage is the common-case,
and is simply a vector of properties which all have attributes of
default_attributes (writable, enumerable, and configurable).
General indexed property storage is for a collection of indexed
properties where EITHER one or more properties have attributes other
than default_attributes OR there is a property with a large index (in
particular, large is '200' or higher).
Indexed properties are now treated relatively the same as storage within
the various Object methods. Additionally, there is a custom iterator
class for IndexedProperties which makes iteration easy. The iterator
skips empty values by default, but can be configured otherwise.
Likewise, it evaluates getters by default, but can be set not to.
2020-05-27 11:35:09 -07:00
|
|
|
raw_strings->indexed_properties().append(value);
|
2020-05-06 16:34:14 -07:00
|
|
|
}
|
2021-07-06 02:15:08 +03:00
|
|
|
strings->define_direct_property(vm.names.raw, raw_strings, 0);
|
2021-09-23 20:56:28 +03:00
|
|
|
return TRY_OR_DISCARD(vm.call(tag_function, js_undefined(), move(arguments)));
|
2020-05-06 10:17:35 +01:00
|
|
|
}
|
|
|
|
|
2020-03-24 14:03:55 +01:00
|
|
|
void TryStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Block)");
|
2020-03-24 14:03:55 +01:00
|
|
|
block().dump(indent + 1);
|
|
|
|
|
|
|
|
if (handler()) {
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Handler)");
|
2020-03-24 14:03:55 +01:00
|
|
|
handler()->dump(indent + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (finalizer()) {
|
|
|
|
print_indent(indent);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Finalizer)");
|
2020-03-24 14:03:55 +01:00
|
|
|
finalizer()->dump(indent + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CatchClause::dump(int indent) const
|
|
|
|
{
|
|
|
|
print_indent(indent);
|
2021-07-11 15:15:38 +04:30
|
|
|
m_parameter.visit(
|
|
|
|
[&](FlyString const& parameter) {
|
|
|
|
if (parameter.is_null())
|
|
|
|
outln("CatchClause");
|
|
|
|
else
|
|
|
|
outln("CatchClause ({})", parameter);
|
|
|
|
},
|
|
|
|
[&](NonnullRefPtr<BindingPattern> const& pattern) {
|
|
|
|
outln("CatchClause");
|
|
|
|
print_indent(indent);
|
|
|
|
outln("(Parameter)");
|
|
|
|
pattern->dump(indent + 2);
|
|
|
|
});
|
|
|
|
|
2020-03-24 14:03:55 +01:00
|
|
|
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-06-08 20:57:54 +02:00
|
|
|
Value TryStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-24 14:03:55 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2021-04-03 14:56:54 +02:00
|
|
|
auto result = interpreter.execute_statement(global_object, m_block, ScopeType::Try);
|
2020-03-24 14:37:39 +01:00
|
|
|
if (auto* exception = interpreter.exception()) {
|
|
|
|
if (m_handler) {
|
2020-09-21 15:28:09 +02:00
|
|
|
interpreter.vm().clear_exception();
|
2020-12-08 18:12:39 +01:00
|
|
|
|
|
|
|
HashMap<FlyString, Variable> parameters;
|
2021-07-11 15:15:38 +04:30
|
|
|
m_handler->parameter().visit(
|
|
|
|
[&](FlyString const& parameter) {
|
|
|
|
parameters.set(parameter, Variable { exception->value(), DeclarationKind::Var });
|
|
|
|
},
|
|
|
|
[&](NonnullRefPtr<BindingPattern> const& pattern) {
|
|
|
|
pattern->for_each_bound_name([&](auto& name) {
|
|
|
|
parameters.set(name, Variable { Value {}, DeclarationKind::Var });
|
|
|
|
});
|
|
|
|
});
|
2021-07-01 12:24:46 +02:00
|
|
|
auto* catch_scope = interpreter.heap().allocate<DeclarativeEnvironment>(global_object, move(parameters), interpreter.vm().running_execution_context().lexical_environment);
|
|
|
|
TemporaryChange<Environment*> scope_change(interpreter.vm().running_execution_context().lexical_environment, catch_scope);
|
2021-07-11 15:15:38 +04:30
|
|
|
|
|
|
|
if (auto* pattern = m_handler->parameter().get_pointer<NonnullRefPtr<BindingPattern>>())
|
|
|
|
interpreter.vm().assign(*pattern, exception->value(), global_object, true);
|
|
|
|
if (interpreter.exception())
|
|
|
|
result = js_undefined();
|
|
|
|
else
|
|
|
|
result = interpreter.execute_statement(global_object, m_handler->body());
|
2020-03-24 14:37:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-11 23:30:00 +01:00
|
|
|
if (m_finalizer) {
|
|
|
|
// Keep, if any, and then clear the current exception so we can
|
|
|
|
// execute() the finalizer without an exception in our way.
|
|
|
|
auto* previous_exception = interpreter.exception();
|
2020-09-21 15:28:09 +02:00
|
|
|
interpreter.vm().clear_exception();
|
2021-04-13 00:57:17 +02:00
|
|
|
|
|
|
|
// Remember what scope type we were unwinding to, and temporarily
|
|
|
|
// clear it as well (e.g. return from handler).
|
|
|
|
auto unwind_until = interpreter.vm().unwind_until();
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-04-13 00:57:17 +02:00
|
|
|
|
|
|
|
auto finalizer_result = m_finalizer->execute(interpreter, global_object);
|
|
|
|
if (interpreter.vm().should_unwind()) {
|
|
|
|
// This was NOT a 'normal' completion (e.g. return from finalizer).
|
|
|
|
result = finalizer_result;
|
|
|
|
} else {
|
|
|
|
// Continue unwinding to whatever we found ourselves unwinding
|
|
|
|
// to when the finalizer was entered (e.g. return from handler,
|
|
|
|
// which is unaffected by normal completion from finalizer).
|
|
|
|
interpreter.vm().unwind(unwind_until);
|
|
|
|
|
|
|
|
// If we previously had an exception and the finalizer didn't
|
|
|
|
// throw a new one, restore the old one.
|
|
|
|
if (previous_exception && !interpreter.exception())
|
2021-04-13 01:02:38 +02:00
|
|
|
interpreter.vm().set_exception(*previous_exception);
|
2021-04-13 00:57:17 +02:00
|
|
|
}
|
2020-09-11 23:30:00 +01:00
|
|
|
}
|
2020-03-24 14:37:39 +01:00
|
|
|
|
2021-04-13 00:58:54 +02:00
|
|
|
return result.value_or(js_undefined());
|
2020-03-24 14:03:55 +01:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value CatchClause::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-24 14:03:55 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-03-24 14:37:39 +01:00
|
|
|
// NOTE: CatchClause execution is handled by TryStatement.
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-24 14:03:55 +01:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ThrowStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-24 22:03:50 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-06-08 20:57:54 +02:00
|
|
|
auto value = m_argument->execute(interpreter, global_object);
|
2020-09-27 15:18:55 +02:00
|
|
|
if (interpreter.vm().exception())
|
2020-03-27 15:35:35 +01:00
|
|
|
return {};
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().throw_exception(global_object, value);
|
2020-08-25 12:52:32 +02:00
|
|
|
return {};
|
2020-03-24 22:03:50 +01:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value SwitchStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-03-29 13:09:54 +02:00
|
|
|
{
|
2021-09-26 16:28:50 +02:00
|
|
|
// FIXME: This needs a massive refactoring, ideally once we start using continue, break, and return completions.
|
|
|
|
// Instead of having an optional test expression, SwitchCase should be split into CaseClause and DefaultClause.
|
|
|
|
// https://tc39.es/ecma262/#sec-switch-statement
|
|
|
|
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto discriminant_result = m_discriminant->execute(interpreter, global_object);
|
2020-03-29 14:34:25 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
|
2021-06-06 20:58:10 +02:00
|
|
|
auto last_value = js_undefined();
|
2020-03-29 14:34:25 +02:00
|
|
|
|
2021-09-26 16:28:50 +02:00
|
|
|
auto execute_switch_case = [&](auto const& switch_case) -> Optional<Value> {
|
2020-03-29 14:34:25 +02:00
|
|
|
for (auto& statement : switch_case.consequent()) {
|
2021-06-06 20:58:10 +02:00
|
|
|
auto value = statement.execute(interpreter, global_object);
|
|
|
|
if (!value.is_empty())
|
|
|
|
last_value = value;
|
2020-03-29 14:34:25 +02:00
|
|
|
if (interpreter.exception())
|
2021-09-26 16:28:50 +02:00
|
|
|
return Value {};
|
2020-09-27 15:18:55 +02:00
|
|
|
if (interpreter.vm().should_unwind()) {
|
2021-09-26 18:16:06 +02:00
|
|
|
if (interpreter.vm().should_unwind_until(ScopeType::Continuable, m_labels)) {
|
2020-10-18 18:01:12 +01:00
|
|
|
// No stop_unwind(), the outer loop will handle that - we just need to break out of the switch/case.
|
2021-06-06 20:58:10 +02:00
|
|
|
return last_value;
|
2021-09-26 18:16:06 +02:00
|
|
|
} else if (interpreter.vm().should_unwind_until(ScopeType::Breakable, m_labels)) {
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().stop_unwind();
|
2021-06-06 20:58:10 +02:00
|
|
|
return last_value;
|
2020-10-18 17:44:55 +01:00
|
|
|
} else {
|
|
|
|
return last_value;
|
2020-04-05 00:09:48 +02:00
|
|
|
}
|
|
|
|
}
|
2020-03-29 14:34:25 +02:00
|
|
|
}
|
2021-09-26 16:28:50 +02:00
|
|
|
return {};
|
|
|
|
};
|
|
|
|
|
|
|
|
bool falling_through = false;
|
|
|
|
SwitchCase const* default_switch_case = nullptr;
|
|
|
|
for (auto& switch_case : m_cases) {
|
|
|
|
if (!switch_case.test()) {
|
|
|
|
default_switch_case = &switch_case;
|
|
|
|
falling_through = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!falling_through) {
|
|
|
|
auto test_result = switch_case.test()->execute(interpreter, global_object);
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
if (!is_strictly_equal(discriminant_result, test_result))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
falling_through = true;
|
|
|
|
if (auto result = execute_switch_case(switch_case); result.has_value())
|
|
|
|
return *result;
|
|
|
|
}
|
|
|
|
if (default_switch_case) {
|
|
|
|
if (auto result = execute_switch_case(*default_switch_case); result.has_value())
|
|
|
|
return *result;
|
2020-03-29 14:34:25 +02:00
|
|
|
}
|
2021-09-26 16:28:50 +02:00
|
|
|
|
2021-06-06 20:58:10 +02:00
|
|
|
return last_value;
|
2020-03-29 13:09:54 +02:00
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value SwitchCase::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-29 13:09:54 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-10-18 17:44:55 +01:00
|
|
|
// NOTE: SwitchCase execution is handled by SwitchStatement.
|
2021-02-23 20:42:32 +01:00
|
|
|
VERIFY_NOT_REACHED();
|
2020-03-29 13:09:54 +02:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value BreakStatement::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-03-29 13:09:54 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().unwind(ScopeType::Breakable, m_target_label);
|
2021-03-16 09:12:34 +01:00
|
|
|
return {};
|
2020-03-29 13:09:54 +02:00
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ContinueStatement::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-04-05 00:22:42 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-09-27 15:18:55 +02:00
|
|
|
interpreter.vm().unwind(ScopeType::Continuable, m_target_label);
|
2021-03-16 09:12:34 +01:00
|
|
|
return {};
|
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);
|
2020-05-04 19:01:24 +01:00
|
|
|
print_indent(indent + 1);
|
2020-03-29 13:09:54 +02:00
|
|
|
if (m_test) {
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Test)");
|
2020-05-04 19:01:24 +01:00
|
|
|
m_test->dump(indent + 2);
|
2020-03-29 13:09:54 +02:00
|
|
|
} else {
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Default)");
|
2020-03-29 13:09:54 +02:00
|
|
|
}
|
2020-05-04 19:01:24 +01:00
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Consequent)");
|
2020-05-04 19:01:24 +01:00
|
|
|
for (auto& statement : m_consequent)
|
|
|
|
statement.dump(indent + 2);
|
2020-03-29 13:09:54 +02:00
|
|
|
}
|
2020-04-01 18:31:24 +01:00
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value ConditionalExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-03 12:14:28 +02:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
auto test_result = m_test->execute(interpreter, global_object);
|
2020-04-03 12:14:28 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
Value result;
|
|
|
|
if (test_result.to_boolean()) {
|
2020-06-08 20:57:54 +02:00
|
|
|
result = m_consequent->execute(interpreter, global_object);
|
2020-04-03 12:14:28 +02:00
|
|
|
} else {
|
2020-06-08 20:57:54 +02:00
|
|
|
result = m_alternate->execute(interpreter, global_object);
|
2020-04-03 12:14:28 +02:00
|
|
|
}
|
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConditionalExpression::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Test)");
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
m_test->dump(indent + 2);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Consequent)");
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
m_consequent->dump(indent + 2);
|
|
|
|
print_indent(indent + 1);
|
2020-12-06 16:55:19 +00:00
|
|
|
outln("(Alternate)");
|
LibJS: Add template literals
Adds fully functioning template literals. Because template literals
contain expressions, most of the work has to be done in the Lexer rather
than the Parser. And because of the complexity of template literals
(expressions, nesting, escapes, etc), the Lexer needs to have some
template-related state.
When entering a new template literal, a TemplateLiteralStart token is
emitted. When inside a literal, all text will be parsed up until a '${'
or '`' (or EOF, but that's a syntax error) is seen, and then a
TemplateLiteralExprStart token is emitted. At this point, the Lexer
proceeds as normal, however it keeps track of the number of opening
and closing curly braces it has seen in order to determine the close
of the expression. Once it finds a matching curly brace for the '${',
a TemplateLiteralExprEnd token is emitted and the state is updated
accordingly.
When the Lexer is inside of a template literal, but not an expression,
and sees a '`', this must be the closing grave: a TemplateLiteralEnd
token is emitted.
The state required to correctly parse template strings consists of a
vector (for nesting) of two pieces of information: whether or not we
are in a template expression (as opposed to a template string); and
the count of the number of unmatched open curly braces we have seen
(only applicable if the Lexer is currently in a template expression).
TODO: Add support for template literal newlines in the JS REPL (this will
cause a syntax error currently):
> `foo
> bar`
'foo
bar'
2020-05-03 15:41:14 -07:00
|
|
|
m_alternate->dump(indent + 2);
|
2020-04-03 12:14:28 +02:00
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2020-06-08 20:57:54 +02:00
|
|
|
Value SequenceExpression::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
2020-04-07 15:11:05 +02:00
|
|
|
{
|
2021-03-21 17:38:42 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-12-28 20:45:22 +03:30
|
|
|
|
2020-04-07 15:11:05 +02:00
|
|
|
Value last_value;
|
|
|
|
for (auto& expression : m_expressions) {
|
2020-06-08 20:57:54 +02:00
|
|
|
last_value = expression.execute(interpreter, global_object);
|
2020-04-07 15:11:05 +02:00
|
|
|
if (interpreter.exception())
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
return last_value;
|
|
|
|
}
|
|
|
|
|
2020-12-28 20:45:22 +03:30
|
|
|
Value DebuggerStatement::execute(Interpreter& interpreter, GlobalObject&) const
|
2020-04-30 17:26:27 +01:00
|
|
|
{
|
2021-03-16 10:51:55 +01:00
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
2020-07-03 22:46:12 -07:00
|
|
|
// Sorry, no JavaScript debugger available (yet)!
|
2021-03-16 09:12:34 +01:00
|
|
|
return {};
|
2020-04-30 17:26:27 +01:00
|
|
|
}
|
|
|
|
|
2020-04-13 16:42:54 +02:00
|
|
|
void ScopeNode::add_variables(NonnullRefPtrVector<VariableDeclaration> variables)
|
|
|
|
{
|
2021-06-12 13:24:45 +02:00
|
|
|
m_variables.extend(move(variables));
|
2020-04-13 16:42:54 +02:00
|
|
|
}
|
|
|
|
|
2020-06-04 14:48:36 +02:00
|
|
|
void ScopeNode::add_functions(NonnullRefPtrVector<FunctionDeclaration> functions)
|
|
|
|
{
|
2021-06-12 13:24:45 +02:00
|
|
|
m_functions.extend(move(functions));
|
2020-06-04 14:48:36 +02:00
|
|
|
}
|
|
|
|
|
2021-07-05 21:45:34 +02:00
|
|
|
void ScopeNode::add_hoisted_function(NonnullRefPtr<FunctionDeclaration> hoisted_function)
|
2021-07-04 03:15:52 +02:00
|
|
|
{
|
2021-07-05 21:45:34 +02:00
|
|
|
m_hoisted_functions.append(hoisted_function);
|
2021-07-04 03:15:52 +02:00
|
|
|
}
|
2021-08-14 17:42:30 +02:00
|
|
|
|
|
|
|
Value ImportStatement::execute(Interpreter& interpreter, GlobalObject&) const
|
|
|
|
{
|
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
|
|
|
dbgln("Modules are not fully supported yet!");
|
|
|
|
TODO();
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Value ExportStatement::execute(Interpreter& interpreter, GlobalObject& global_object) const
|
|
|
|
{
|
|
|
|
InterpreterNodeScope node_scope { interpreter, *this };
|
|
|
|
if (m_statement)
|
|
|
|
return m_statement->execute(interpreter, global_object);
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
void ExportStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
print_indent(indent + 1);
|
|
|
|
outln("(ExportEntries)");
|
|
|
|
|
|
|
|
auto string_or_null = [](String const& string) -> String {
|
|
|
|
if (string.is_empty()) {
|
|
|
|
return "null";
|
|
|
|
}
|
|
|
|
return String::formatted("\"{}\"", string);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (auto& entry : m_entries) {
|
|
|
|
print_indent(indent + 2);
|
|
|
|
outln("ModuleRequest: {}, ImportName: {}, LocalName: {}, ExportName: {}", string_or_null(entry.module_request), entry.kind == ExportEntry::ModuleRequest ? string_or_null(entry.local_or_import_name) : "null", entry.kind != ExportEntry::ModuleRequest ? string_or_null(entry.local_or_import_name) : "null", string_or_null(entry.export_name));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ImportStatement::dump(int indent) const
|
|
|
|
{
|
|
|
|
ASTNode::dump(indent);
|
|
|
|
print_indent(indent + 1);
|
|
|
|
if (m_entries.is_empty()) {
|
|
|
|
// direct from "module" import
|
|
|
|
outln("Entire module '{}'", m_module_request);
|
|
|
|
} else {
|
|
|
|
outln("(ExportEntries) from {}", m_module_request);
|
|
|
|
|
|
|
|
for (auto& entry : m_entries) {
|
|
|
|
print_indent(indent + 2);
|
|
|
|
outln("ImportName: {}, LocalName: {}", entry.import_name, entry.local_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ExportStatement::has_export(StringView export_name) const
|
|
|
|
{
|
|
|
|
return any_of(m_entries.begin(), m_entries.end(), [&](auto& entry) {
|
|
|
|
return entry.export_name == export_name;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ImportStatement::has_bound_name(StringView name) const
|
|
|
|
{
|
|
|
|
return any_of(m_entries.begin(), m_entries.end(), [&](auto& entry) {
|
|
|
|
return entry.local_name == name;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-03-07 19:42:11 +01:00
|
|
|
}
|