| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-05-29 12:38:28 +02:00
										 |  |  |  * Copyright (c) 2020, Stephan Unverwerth <s.unverwerth@serenityos.org> | 
					
						
							| 
									
										
										
										
											2021-10-03 14:16:10 +02:00
										 |  |  |  * Copyright (c) 2021, David Tuin <davidot@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-08 10:49:08 -07:00
										 |  |  | #include <AK/HashTable.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  | #include <AK/NonnullRefPtr.h>
 | 
					
						
							| 
									
										
										
										
											2020-05-26 12:57:50 +01:00
										 |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2020-05-27 22:22:08 -07:00
										 |  |  | #include <LibJS/AST.h>
 | 
					
						
							|  |  |  | #include <LibJS/Lexer.h>
 | 
					
						
							| 
									
										
										
										
											2022-01-15 17:26:06 +01:00
										 |  |  | #include <LibJS/Runtime/FunctionConstructor.h>
 | 
					
						
							| 
									
										
										
										
											2020-12-28 20:45:22 +03:30
										 |  |  | #include <LibJS/SourceRange.h>
 | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace JS { | 
					
						
							| 
									
										
										
										
											2020-03-12 23:02:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum class Associativity { | 
					
						
							|  |  |  |     Left, | 
					
						
							|  |  |  |     Right | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 17:56:49 +01:00
										 |  |  | struct FunctionNodeParseOptions { | 
					
						
							|  |  |  |     enum { | 
					
						
							|  |  |  |         CheckForFunctionAndName = 1 << 0, | 
					
						
							|  |  |  |         AllowSuperPropertyLookup = 1 << 1, | 
					
						
							|  |  |  |         AllowSuperConstructorCall = 1 << 2, | 
					
						
							| 
									
										
										
										
											2020-10-20 18:32:51 +01:00
										 |  |  |         IsGetterFunction = 1 << 3, | 
					
						
							|  |  |  |         IsSetterFunction = 1 << 4, | 
					
						
							| 
									
										
										
										
											2020-10-25 11:14:04 +00:00
										 |  |  |         IsArrowFunction = 1 << 5, | 
					
						
							| 
									
										
										
										
											2021-06-14 14:52:59 +04:30
										 |  |  |         IsGeneratorFunction = 1 << 6, | 
					
						
							| 
									
										
										
										
											2021-11-09 20:39:22 +02:00
										 |  |  |         IsAsyncFunction = 1 << 7, | 
					
						
							| 
									
										
										
										
											2020-10-20 17:56:49 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-22 12:44:56 +02:00
										 |  |  | class ScopePusher; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | class Parser { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2021-08-14 17:30:37 +02:00
										 |  |  |     explicit Parser(Lexer lexer, Program::Type program_type = Program::Type::Script); | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-19 20:13:53 -07:00
										 |  |  |     NonnullRefPtr<Program> parse_program(bool starts_in_strict_mode = false); | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-19 11:52:56 +01:00
										 |  |  |     template<typename FunctionNodeType> | 
					
						
							| 
									
										
										
										
											2020-10-20 17:56:49 +01:00
										 |  |  |     NonnullRefPtr<FunctionNodeType> parse_function_node(u8 parse_options = FunctionNodeParseOptions::CheckForFunctionAndName); | 
					
						
							| 
									
										
										
										
											2021-05-29 16:03:19 +04:30
										 |  |  |     Vector<FunctionNode::Parameter> parse_formal_parameters(int& function_length, u8 parse_options = 0); | 
					
						
							| 
									
										
										
										
											2021-09-18 01:11:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     enum class AllowDuplicates { | 
					
						
							|  |  |  |         Yes, | 
					
						
							|  |  |  |         No | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     enum class AllowMemberExpressions { | 
					
						
							|  |  |  |         Yes, | 
					
						
							|  |  |  |         No | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     RefPtr<BindingPattern> parse_binding_pattern(AllowDuplicates is_var_declaration = AllowDuplicates::No, AllowMemberExpressions allow_member_expressions = AllowMemberExpressions::No); | 
					
						
							| 
									
										
										
										
											2020-03-19 11:52:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-14 15:46:41 +04:30
										 |  |  |     struct PrimaryExpressionParseResult { | 
					
						
							|  |  |  |         NonnullRefPtr<Expression> result; | 
					
						
							|  |  |  |         bool should_continue_parsing_as_expression { true }; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-22 23:30:07 +01:00
										 |  |  |     NonnullRefPtr<Declaration> parse_declaration(); | 
					
						
							| 
									
										
										
										
											2021-07-25 01:01:22 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     enum class AllowLabelledFunction { | 
					
						
							|  |  |  |         No, | 
					
						
							|  |  |  |         Yes | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NonnullRefPtr<Statement> parse_statement(AllowLabelledFunction allow_labelled_function = AllowLabelledFunction::No); | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |     NonnullRefPtr<BlockStatement> parse_block_statement(); | 
					
						
							| 
									
										
										
										
											2021-10-08 12:43:38 +02:00
										 |  |  |     NonnullRefPtr<FunctionBody> parse_function_body(Vector<FunctionDeclaration::Parameter> const& parameters, FunctionKind function_kind, bool& contains_direct_call_to_eval); | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |     NonnullRefPtr<ReturnStatement> parse_return_statement(); | 
					
						
							| 
									
										
										
										
											2020-10-30 19:08:45 +00:00
										 |  |  |     NonnullRefPtr<VariableDeclaration> parse_variable_declaration(bool for_loop_variable_declaration = false); | 
					
						
							| 
									
										
										
										
											2020-04-21 19:21:26 +01:00
										 |  |  |     NonnullRefPtr<Statement> parse_for_statement(); | 
					
						
							| 
									
										
										
										
											2021-11-23 16:09:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     enum class IsForAwaitLoop { | 
					
						
							|  |  |  |         No, | 
					
						
							|  |  |  |         Yes | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NonnullRefPtr<Statement> parse_for_in_of_statement(NonnullRefPtr<ASTNode> lhs, IsForAwaitLoop is_await); | 
					
						
							| 
									
										
										
										
											2020-03-21 18:40:17 +01:00
										 |  |  |     NonnullRefPtr<IfStatement> parse_if_statement(); | 
					
						
							| 
									
										
										
										
											2020-03-24 22:03:50 +01:00
										 |  |  |     NonnullRefPtr<ThrowStatement> parse_throw_statement(); | 
					
						
							| 
									
										
										
										
											2020-03-24 14:03:55 +01:00
										 |  |  |     NonnullRefPtr<TryStatement> parse_try_statement(); | 
					
						
							|  |  |  |     NonnullRefPtr<CatchClause> parse_catch_clause(); | 
					
						
							| 
									
										
										
										
											2020-03-29 13:09:54 +02:00
										 |  |  |     NonnullRefPtr<SwitchStatement> parse_switch_statement(); | 
					
						
							|  |  |  |     NonnullRefPtr<SwitchCase> parse_switch_case(); | 
					
						
							|  |  |  |     NonnullRefPtr<BreakStatement> parse_break_statement(); | 
					
						
							| 
									
										
										
										
											2020-04-05 00:22:42 +02:00
										 |  |  |     NonnullRefPtr<ContinueStatement> parse_continue_statement(); | 
					
						
							| 
									
										
										
										
											2020-04-04 21:29:23 +02:00
										 |  |  |     NonnullRefPtr<DoWhileStatement> parse_do_while_statement(); | 
					
						
							| 
									
										
										
										
											2020-04-21 19:27:57 +01:00
										 |  |  |     NonnullRefPtr<WhileStatement> parse_while_statement(); | 
					
						
							| 
									
										
										
										
											2020-11-28 15:05:57 +01:00
										 |  |  |     NonnullRefPtr<WithStatement> parse_with_statement(); | 
					
						
							| 
									
										
										
										
											2020-04-30 17:26:27 +01:00
										 |  |  |     NonnullRefPtr<DebuggerStatement> parse_debugger_statement(); | 
					
						
							| 
									
										
										
										
											2020-04-03 12:14:28 +02:00
										 |  |  |     NonnullRefPtr<ConditionalExpression> parse_conditional_expression(NonnullRefPtr<Expression> test); | 
					
						
							| 
									
										
										
										
											2021-09-14 06:56:31 +04:30
										 |  |  |     NonnullRefPtr<OptionalChain> parse_optional_chain(NonnullRefPtr<Expression> base); | 
					
						
							| 
									
										
										
										
											2021-02-24 09:58:24 +01:00
										 |  |  |     NonnullRefPtr<Expression> parse_expression(int min_precedence, Associativity associate = Associativity::Right, const Vector<TokenType>& forbidden = {}); | 
					
						
							| 
									
										
										
										
											2021-06-14 15:46:41 +04:30
										 |  |  |     PrimaryExpressionParseResult parse_primary_expression(); | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |     NonnullRefPtr<Expression> parse_unary_prefixed_expression(); | 
					
						
							| 
									
										
										
										
											2020-06-03 16:05:49 -07:00
										 |  |  |     NonnullRefPtr<RegExpLiteral> parse_regexp_literal(); | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |     NonnullRefPtr<ObjectExpression> parse_object_expression(); | 
					
						
							| 
									
										
										
										
											2020-03-20 20:29:57 +01:00
										 |  |  |     NonnullRefPtr<ArrayExpression> parse_array_expression(); | 
					
						
							| 
									
										
										
										
											2021-02-24 09:58:24 +01:00
										 |  |  |     NonnullRefPtr<StringLiteral> parse_string_literal(const Token& token, bool in_template_literal = false); | 
					
						
							| 
									
										
										
										
											2020-05-06 16:34:14 -07:00
										 |  |  |     NonnullRefPtr<TemplateLiteral> parse_template_literal(bool is_tagged); | 
					
						
							| 
									
										
										
										
											2020-03-18 11:23:53 +01:00
										 |  |  |     NonnullRefPtr<Expression> parse_secondary_expression(NonnullRefPtr<Expression>, int min_precedence, Associativity associate = Associativity::Right); | 
					
						
							| 
									
										
										
										
											2021-09-03 19:11:51 +02:00
										 |  |  |     NonnullRefPtr<Expression> parse_call_expression(NonnullRefPtr<Expression>); | 
					
						
							| 
									
										
										
										
											2020-03-28 16:33:52 +01:00
										 |  |  |     NonnullRefPtr<NewExpression> parse_new_expression(); | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  |     NonnullRefPtr<ClassDeclaration> parse_class_declaration(); | 
					
						
							|  |  |  |     NonnullRefPtr<ClassExpression> parse_class_expression(bool expect_class_name); | 
					
						
							| 
									
										
										
										
											2021-06-11 01:38:30 +04:30
										 |  |  |     NonnullRefPtr<YieldExpression> parse_yield_expression(); | 
					
						
							| 
									
										
										
										
											2021-11-09 22:52:21 +02:00
										 |  |  |     NonnullRefPtr<AwaitExpression> parse_await_expression(); | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  |     NonnullRefPtr<Expression> parse_property_key(); | 
					
						
							| 
									
										
										
										
											2020-10-04 23:58:57 +01:00
										 |  |  |     NonnullRefPtr<AssignmentExpression> parse_assignment_expression(AssignmentOp, NonnullRefPtr<Expression> lhs, int min_precedence, Associativity); | 
					
						
							| 
									
										
										
										
											2021-06-12 18:04:28 -07:00
										 |  |  |     NonnullRefPtr<Identifier> parse_identifier(); | 
					
						
							| 
									
										
										
										
											2021-08-14 17:42:30 +02:00
										 |  |  |     NonnullRefPtr<ImportStatement> parse_import_statement(Program& program); | 
					
						
							|  |  |  |     NonnullRefPtr<ExportStatement> parse_export_statement(Program& program); | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-15 00:47:16 +01:00
										 |  |  |     RefPtr<FunctionExpression> try_parse_arrow_function_expression(bool expect_parens, bool is_async = false); | 
					
						
							| 
									
										
											  
											
												LibJS: Replace the custom unwind mechanism with completions :^)
This includes:
- Parsing proper LabelledStatements with try_parse_labelled_statement()
- Removing LabelableStatement
- Implementing the LoopEvaluation semantics via loop_evaluation() in
  each IterationStatement subclass; and IterationStatement evaluation
  via {For,ForIn,ForOf,ForAwaitOf,While,DoWhile}Statement::execute()
- Updating ReturnStatement, BreakStatement and ContinueStatement to
  return the appropriate completion types
- Basically reimplementing TryStatement and SwitchStatement according to
  the spec, using completions
- Honoring result completion types in AsyncBlockStart and
  OrdinaryCallEvaluateBody
- Removing any uses of the VM unwind mechanism - most importantly,
  VM::throw_exception() now exclusively sets an exception and no longer
  triggers any unwinding mechanism.
  However, we already did a good job updating all of LibWeb and userland
  applications to not use it, and the few remaining uses elsewhere don't
  rely on unwinding AFAICT.
											
										 
											2022-01-05 19:11:16 +01:00
										 |  |  |     RefPtr<LabelledStatement> try_parse_labelled_statement(AllowLabelledFunction allow_function); | 
					
						
							| 
									
										
										
										
											2020-11-02 21:27:42 +00:00
										 |  |  |     RefPtr<MetaProperty> try_parse_new_target_expression(); | 
					
						
							| 
									
										
										
										
											2021-11-26 23:45:10 +01:00
										 |  |  |     RefPtr<MetaProperty> try_parse_import_meta_expression(); | 
					
						
							|  |  |  |     NonnullRefPtr<ImportCall> parse_import_call(); | 
					
						
							| 
									
										
										
										
											2020-11-02 21:27:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-14 06:56:31 +04:30
										 |  |  |     Vector<CallExpression::Argument> parse_arguments(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |     struct Error { | 
					
						
							|  |  |  |         String message; | 
					
						
							|  |  |  |         Optional<Position> position; | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         String to_string() const | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |             if (!position.has_value()) | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  |                 return message; | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |             return String::formatted("{} (line: {}, column: {})", message, position.value().line, position.value().column); | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-26 12:57:50 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-11 00:55:02 +01:00
										 |  |  |         String source_location_hint(StringView source, const char spacer = ' ', const char indicator = '^') const | 
					
						
							| 
									
										
										
										
											2020-05-26 12:57:50 +01:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |             if (!position.has_value()) | 
					
						
							| 
									
										
										
										
											2020-05-26 12:57:50 +01:00
										 |  |  |                 return {}; | 
					
						
							| 
									
										
										
										
											2021-07-20 17:53:48 +01:00
										 |  |  |             // We need to modify the source to match what the lexer considers one line - normalizing
 | 
					
						
							|  |  |  |             // line terminators to \n is easier than splitting using all different LT characters.
 | 
					
						
							| 
									
										
										
										
											2021-09-11 02:15:44 +03:00
										 |  |  |             String source_string = source.replace("\r\n", "\n").replace("\r", "\n").replace(LINE_SEPARATOR_STRING, "\n").replace(PARAGRAPH_SEPARATOR_STRING, "\n"); | 
					
						
							| 
									
										
										
										
											2021-07-20 17:53:48 +01:00
										 |  |  |             StringBuilder builder; | 
					
						
							|  |  |  |             builder.append(source_string.split_view('\n', true)[position.value().line - 1]); | 
					
						
							| 
									
										
										
										
											2020-05-26 12:57:50 +01:00
										 |  |  |             builder.append('\n'); | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |             for (size_t i = 0; i < position.value().column - 1; ++i) | 
					
						
							| 
									
										
										
										
											2020-05-26 12:57:50 +01:00
										 |  |  |                 builder.append(spacer); | 
					
						
							|  |  |  |             builder.append(indicator); | 
					
						
							|  |  |  |             return builder.build(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |     bool has_errors() const { return m_state.errors.size(); } | 
					
						
							|  |  |  |     const Vector<Error>& errors() const { return m_state.errors; } | 
					
						
							| 
									
										
										
										
											2021-07-19 17:56:21 +02:00
										 |  |  |     void print_errors(bool print_hint = true) const | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |         for (auto& error : m_state.errors) { | 
					
						
							| 
									
										
										
										
											2021-07-19 17:56:21 +02:00
										 |  |  |             if (print_hint) { | 
					
						
							|  |  |  |                 auto hint = error.source_location_hint(m_state.lexer.source()); | 
					
						
							|  |  |  |                 if (!hint.is_empty()) | 
					
						
							|  |  |  |                     warnln("{}", hint); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-12-06 14:50:39 +00:00
										 |  |  |             warnln("SyntaxError: {}", error.to_string()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-05-14 16:26:01 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-11 22:41:51 +02:00
										 |  |  |     struct TokenMemoization { | 
					
						
							|  |  |  |         bool try_parse_arrow_function_expression_failed; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-15 17:26:06 +01:00
										 |  |  |     // Needs to mess with m_state, and we're not going to expose a non-const getter for that :^)
 | 
					
						
							|  |  |  |     friend ThrowCompletionOr<ECMAScriptFunctionObject*> FunctionConstructor::create_dynamic_function(GlobalObject&, FunctionObject&, FunctionObject*, FunctionKind, MarkedValueList const&); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2020-04-13 16:42:54 +02:00
										 |  |  |     friend class ScopePusher; | 
					
						
							| 
									
										
										
										
											2021-11-27 00:01:23 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void parse_script(Program& program, bool starts_in_strict_mode); | 
					
						
							|  |  |  |     void parse_module(Program& program); | 
					
						
							| 
									
										
										
										
											2020-04-13 16:42:54 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-12 23:02:41 +01:00
										 |  |  |     Associativity operator_associativity(TokenType) const; | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     bool match_expression() const; | 
					
						
							| 
									
										
										
										
											2020-03-14 20:45:51 +02:00
										 |  |  |     bool match_unary_prefixed_expression() const; | 
					
						
							| 
									
										
										
										
											2021-02-24 09:58:24 +01:00
										 |  |  |     bool match_secondary_expression(const Vector<TokenType>& forbidden = {}) const; | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     bool match_statement() const; | 
					
						
							| 
									
										
										
										
											2021-08-14 17:42:30 +02:00
										 |  |  |     bool match_export_or_import() const; | 
					
						
							| 
									
										
										
										
											2021-12-20 15:29:25 +01:00
										 |  |  |     bool match_assert_clause() const; | 
					
						
							| 
									
										
										
										
											2021-10-08 00:38:24 +02:00
										 |  |  |     bool match_declaration() const; | 
					
						
							|  |  |  |     bool try_match_let_declaration() const; | 
					
						
							|  |  |  |     bool match_variable_declaration() const; | 
					
						
							| 
									
										
										
										
											2021-07-11 15:34:55 +04:30
										 |  |  |     bool match_identifier() const; | 
					
						
							| 
									
										
										
										
											2020-04-18 20:31:27 +02:00
										 |  |  |     bool match_identifier_name() const; | 
					
						
							| 
									
										
										
										
											2020-06-08 13:31:21 -05:00
										 |  |  |     bool match_property_key() const; | 
					
						
							| 
									
										
										
										
											2021-10-12 22:45:52 +02:00
										 |  |  |     bool is_private_identifier_valid() const; | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     bool match(TokenType type) const; | 
					
						
							|  |  |  |     bool done() const; | 
					
						
							|  |  |  |     void expected(const char* what); | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |     void syntax_error(const String& message, Optional<Position> = {}); | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     Token consume(); | 
					
						
							| 
									
										
										
										
											2021-07-11 15:34:55 +04:30
										 |  |  |     Token consume_identifier(); | 
					
						
							|  |  |  |     Token consume_identifier_reference(); | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  |     Token consume(TokenType type); | 
					
						
							| 
									
										
										
										
											2020-10-19 18:01:28 +01:00
										 |  |  |     Token consume_and_validate_numeric_literal(); | 
					
						
							| 
									
										
										
										
											2020-04-17 15:05:58 +02:00
										 |  |  |     void consume_or_insert_semicolon(); | 
					
						
							| 
									
										
										
										
											2020-03-30 08:24:43 -05:00
										 |  |  |     void save_state(); | 
					
						
							|  |  |  |     void load_state(); | 
					
						
							| 
									
										
										
										
											2020-12-29 16:47:39 +03:30
										 |  |  |     void discard_saved_state(); | 
					
						
							| 
									
										
										
										
											2020-11-02 21:03:19 +00:00
										 |  |  |     Position position() const; | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 01:11:32 +02:00
										 |  |  |     RefPtr<BindingPattern> synthesize_binding_pattern(Expression const& expression); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-08 00:38:24 +02:00
										 |  |  |     Token next_token() const; | 
					
						
							| 
									
										
										
										
											2021-08-28 17:04:37 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-12 01:27:35 +02:00
										 |  |  |     void check_identifier_name_for_assignment_validity(StringView, bool force_strict = false); | 
					
						
							| 
									
										
										
										
											2021-07-11 15:34:55 +04:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-11 22:41:51 +02:00
										 |  |  |     bool try_parse_arrow_function_expression_failed_at_position(const Position&) const; | 
					
						
							|  |  |  |     void set_try_parse_arrow_function_expression_failed_at_position(const Position&, bool); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 23:02:50 +02:00
										 |  |  |     bool match_invalid_escaped_keyword() const; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-22 12:44:56 +02:00
										 |  |  |     bool parse_directive(ScopeNode& body); | 
					
						
							|  |  |  |     void parse_statement_list(ScopeNode& output_node, AllowLabelledFunction allow_labelled_functions = AllowLabelledFunction::No); | 
					
						
							| 
									
										
										
										
											2021-12-20 15:29:25 +01:00
										 |  |  |     void parse_assert_clause(ModuleRequest& request); | 
					
						
							| 
									
										
										
										
											2021-09-22 12:44:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-28 20:45:22 +03:30
										 |  |  |     struct RulePosition { | 
					
						
							| 
									
										
										
										
											2020-12-29 08:42:02 +03:30
										 |  |  |         AK_MAKE_NONCOPYABLE(RulePosition); | 
					
						
							|  |  |  |         AK_MAKE_NONMOVABLE(RulePosition); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     public: | 
					
						
							| 
									
										
										
										
											2020-12-28 20:45:22 +03:30
										 |  |  |         RulePosition(Parser& parser, Position position) | 
					
						
							|  |  |  |             : m_parser(parser) | 
					
						
							|  |  |  |             , m_position(position) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2020-12-29 16:47:39 +03:30
										 |  |  |             m_parser.m_rule_starts.append(position); | 
					
						
							| 
									
										
										
										
											2020-12-28 20:45:22 +03:30
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ~RulePosition() | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2020-12-29 16:47:39 +03:30
										 |  |  |             auto last = m_parser.m_rule_starts.take_last(); | 
					
						
							| 
									
										
										
										
											2021-02-23 20:42:32 +01:00
										 |  |  |             VERIFY(last.line == m_position.line); | 
					
						
							|  |  |  |             VERIFY(last.column == m_position.column); | 
					
						
							| 
									
										
										
										
											2020-12-28 20:45:22 +03:30
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Position& position() const { return m_position; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     private: | 
					
						
							|  |  |  |         Parser& m_parser; | 
					
						
							|  |  |  |         Position m_position; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [[nodiscard]] RulePosition push_start() { return { *this, position() }; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-30 08:24:43 -05:00
										 |  |  |     struct ParserState { | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |         Lexer lexer; | 
					
						
							|  |  |  |         Token current_token; | 
					
						
							|  |  |  |         Vector<Error> errors; | 
					
						
							| 
									
										
										
										
											2021-10-08 01:55:24 +02:00
										 |  |  |         ScopePusher* current_scope_pusher { nullptr }; | 
					
						
							| 
									
										
										
										
											2021-06-14 09:30:43 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-18 23:01:54 +02:00
										 |  |  |         HashMap<StringView, Optional<Position>> labels_in_scope; | 
					
						
							| 
									
										
										
										
											2021-10-12 22:45:52 +02:00
										 |  |  |         HashTable<StringView>* referenced_private_names { nullptr }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |         bool strict_mode { false }; | 
					
						
							|  |  |  |         bool allow_super_property_lookup { false }; | 
					
						
							|  |  |  |         bool allow_super_constructor_call { false }; | 
					
						
							|  |  |  |         bool in_function_context { false }; | 
					
						
							| 
									
										
										
										
											2021-08-21 11:31:36 +02:00
										 |  |  |         bool in_formal_parameter_context { false }; | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |         bool in_generator_function_context { false }; | 
					
						
							| 
									
										
										
										
											2021-11-26 23:50:32 +01:00
										 |  |  |         bool await_expression_is_valid { false }; | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |         bool in_arrow_function_context { false }; | 
					
						
							|  |  |  |         bool in_break_context { false }; | 
					
						
							|  |  |  |         bool in_continue_context { false }; | 
					
						
							|  |  |  |         bool string_legacy_octal_escape_sequence_in_scope { false }; | 
					
						
							| 
									
										
										
										
											2021-08-28 17:11:05 +02:00
										 |  |  |         bool in_class_field_initializer { false }; | 
					
						
							| 
									
										
										
										
											2021-11-09 22:52:21 +02:00
										 |  |  |         bool in_class_static_init_block { false }; | 
					
						
							| 
									
										
											  
											
												LibJS: Add an optimization to avoid needless arguments object creation
This gives FunctionNode a "might need arguments object" boolean flag and
sets it based on the simplest possible heuristic for this: if we
encounter an identifier called "arguments" or "eval" up to the next
(nested) function declaration or expression, we won't need an arguments
object. Otherwise, we *might* need one - the final decision is made in
the FunctionDeclarationInstantiation AO.
Now, this is obviously not perfect. Even if you avoid eval, something
like `foo.arguments` will still trigger a false positive - but it's a
start and already massively cuts down on needlessly allocated objects,
especially in real-world code that is often minified, and so a full
"arguments" identifier will be an actual arguments object more often
than not.
To illustrate the actual impact of this change, here's the number of
allocated arguments objects during a full test-js run:
Before:
- Unmapped arguments objects: 78765
- Mapped arguments objects: 2455
After:
- Unmapped arguments objects: 18
- Mapped arguments objects: 37
This results in a ~5% speedup of test-js on my Linux host machine, and
about 3.5% on i686 Serenity in QEMU (warm runs, average of 5).
The following microbenchmark (calling an empty function 1M times) runs
25% faster on Linux and 45% on Serenity:
    function foo() {}
    for (var i = 0; i < 1_000_000; ++i)
        foo();
test262 reports no changes in either direction, apart from a speedup :^)
											
										 
											2021-10-05 08:44:58 +01:00
										 |  |  |         bool function_might_need_arguments_object { false }; | 
					
						
							| 
									
										
										
										
											2020-03-30 08:24:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-14 17:30:37 +02:00
										 |  |  |         ParserState(Lexer, Program::Type); | 
					
						
							| 
									
										
										
										
											2020-03-30 08:24:43 -05:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-11 22:41:51 +02:00
										 |  |  |     class PositionKeyTraits { | 
					
						
							|  |  |  |     public: | 
					
						
							|  |  |  |         static int hash(const Position& position) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return int_hash(position.line) ^ int_hash(position.column); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         static bool equals(const Position& a, const Position& b) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             return a.column == b.column && a.line == b.line; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-29 16:47:39 +03:30
										 |  |  |     Vector<Position> m_rule_starts; | 
					
						
							| 
									
										
										
										
											2021-06-19 14:43:09 +02:00
										 |  |  |     ParserState m_state; | 
					
						
							| 
									
										
										
										
											2021-02-28 10:42:34 +01:00
										 |  |  |     FlyString m_filename; | 
					
						
							| 
									
										
										
										
											2020-05-02 11:46:39 -07:00
										 |  |  |     Vector<ParserState> m_saved_state; | 
					
						
							| 
									
										
										
										
											2021-04-11 22:41:51 +02:00
										 |  |  |     HashMap<Position, TokenMemoization, PositionKeyTraits> m_token_memoizations; | 
					
						
							| 
									
										
										
										
											2021-08-14 17:30:37 +02:00
										 |  |  |     Program::Type m_program_type; | 
					
						
							| 
									
										
										
										
											2020-03-11 19:27:43 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | } |