summaryrefslogtreecommitdiff
path: root/Source/JavaScriptCore/parser/Parser.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Source/JavaScriptCore/parser/Parser.cpp')
-rw-r--r--Source/JavaScriptCore/parser/Parser.cpp3839
1 files changed, 3131 insertions, 708 deletions
diff --git a/Source/JavaScriptCore/parser/Parser.cpp b/Source/JavaScriptCore/parser/Parser.cpp
index 297666e72..3b79d0aa7 100644
--- a/Source/JavaScriptCore/parser/Parser.cpp
+++ b/Source/JavaScriptCore/parser/Parser.cpp
@@ -1,7 +1,7 @@
/*
* Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
* Copyright (C) 2001 Peter Kelly (pmk@post.com)
- * Copyright (C) 2003, 2006, 2007, 2008, 2009, 2010, 2013 Apple Inc. All rights reserved.
+ * Copyright (C) 2003, 2006-2010, 2013, 2016 Apple Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
@@ -24,40 +24,33 @@
#include "Parser.h"
#include "ASTBuilder.h"
-#include "CodeBlock.h"
-#include "Debugger.h"
-#include "JSCJSValueInlines.h"
-#include "Lexer.h"
-#include "NodeInfo.h"
-#include "SourceProvider.h"
+#include "DebuggerParseData.h"
+#include "JSCInlines.h"
#include "VM.h"
#include <utility>
-#include <wtf/HashFunctions.h>
-#include <wtf/OwnPtr.h>
+#include <wtf/SetForScope.h>
#include <wtf/StringPrintStream.h>
-#include <wtf/WTFThreadData.h>
-
#define updateErrorMessage(shouldPrintToken, ...) do {\
propagateError(); \
logError(shouldPrintToken, __VA_ARGS__); \
} while (0)
-#define propagateError() do { if (hasError()) return 0; } while (0)
+#define propagateError() do { if (UNLIKELY(hasError())) return 0; } while (0)
#define internalFailWithMessage(shouldPrintToken, ...) do { updateErrorMessage(shouldPrintToken, __VA_ARGS__); return 0; } while (0)
#define handleErrorToken() do { if (m_token.m_type == EOFTOK || m_token.m_type & ErrorTokenFlag) { failDueToUnexpectedToken(); } } while (0)
#define failWithMessage(...) do { { handleErrorToken(); updateErrorMessage(true, __VA_ARGS__); } return 0; } while (0)
#define failWithStackOverflow() do { updateErrorMessage(false, "Stack exhausted"); m_hasStackOverflow = true; return 0; } while (0)
#define failIfFalse(cond, ...) do { if (!(cond)) { handleErrorToken(); internalFailWithMessage(true, __VA_ARGS__); } } while (0)
-#define failIfTrue(cond, ...) do { if ((cond)) { handleErrorToken(); internalFailWithMessage(true, __VA_ARGS__); } } while (0)
+#define failIfTrue(cond, ...) do { if (cond) { handleErrorToken(); internalFailWithMessage(true, __VA_ARGS__); } } while (0)
#define failIfTrueIfStrict(cond, ...) do { if ((cond) && strictMode()) internalFailWithMessage(false, __VA_ARGS__); } while (0)
#define failIfFalseIfStrict(cond, ...) do { if ((!(cond)) && strictMode()) internalFailWithMessage(false, __VA_ARGS__); } while (0)
#define consumeOrFail(tokenType, ...) do { if (!consume(tokenType)) { handleErrorToken(); internalFailWithMessage(true, __VA_ARGS__); } } while (0)
#define consumeOrFailWithFlags(tokenType, flags, ...) do { if (!consume(tokenType, flags)) { handleErrorToken(); internalFailWithMessage(true, __VA_ARGS__); } } while (0)
#define matchOrFail(tokenType, ...) do { if (!match(tokenType)) { handleErrorToken(); internalFailWithMessage(true, __VA_ARGS__); } } while (0)
-#define failIfStackOverflow() do { if (!canRecurse()) failWithStackOverflow(); } while (0)
+#define failIfStackOverflow() do { if (UNLIKELY(!canRecurse())) failWithStackOverflow(); } while (0)
#define semanticFail(...) do { internalFailWithMessage(false, __VA_ARGS__); } while (0)
-#define semanticFailIfTrue(cond, ...) do { if ((cond)) internalFailWithMessage(false, __VA_ARGS__); } while (0)
+#define semanticFailIfTrue(cond, ...) do { if (cond) internalFailWithMessage(false, __VA_ARGS__); } while (0)
#define semanticFailIfFalse(cond, ...) do { if (!(cond)) internalFailWithMessage(false, __VA_ARGS__); } while (0)
#define regexFail(failure) do { setErrorMessage(failure); return 0; } while (0)
#define failDueToUnexpectedToken() do {\
@@ -69,6 +62,10 @@
consumeOrFail(token, "Expected '", tokenString, "' to ", operation, " a ", production);\
} while (0)
+#define handleProductionOrFail2(token, tokenString, operation, production) do {\
+ consumeOrFail(token, "Expected '", tokenString, "' to ", operation, " an ", production);\
+} while (0)
+
#define semanticFailureDueToKeyword(...) do { \
if (strictMode() && m_token.m_type == RESERVED_IF_STRICT) \
semanticFail("Cannot use the reserved word '", getToken(), "' as a ", __VA_ARGS__, " in strict mode"); \
@@ -76,6 +73,8 @@
semanticFail("Cannot use the reserved word '", getToken(), "' as a ", __VA_ARGS__); \
if (m_token.m_type & KeywordTokenFlag) \
semanticFail("Cannot use the keyword '", getToken(), "' as a ", __VA_ARGS__); \
+ if (isDisallowedIdentifierAwait(m_token)) \
+ semanticFail("Can't use 'await' as a ", __VA_ARGS__, " ", disallowedIdentifierAwaitReason()); \
} while (0)
using namespace std;
@@ -89,53 +88,11 @@ void Parser<LexerType>::logError(bool)
return;
StringPrintStream stream;
printUnexpectedTokenText(stream);
- setErrorMessage(stream.toString());
-}
-
-template <typename LexerType> template <typename A>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1)
-{
- if (hasError())
- return;
- StringPrintStream stream;
- if (shouldPrintToken) {
- printUnexpectedTokenText(stream);
- stream.print(". ");
- }
- stream.print(value1, ".");
- setErrorMessage(stream.toString());
-}
-
-template <typename LexerType> template <typename A, typename B>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B& value2)
-{
- if (hasError())
- return;
- StringPrintStream stream;
- if (shouldPrintToken) {
- printUnexpectedTokenText(stream);
- stream.print(". ");
- }
- stream.print(value1, value2, ".");
- setErrorMessage(stream.toString());
-}
-
-template <typename LexerType> template <typename A, typename B, typename C>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B& value2, const C& value3)
-{
- if (hasError())
- return;
- StringPrintStream stream;
- if (shouldPrintToken) {
- printUnexpectedTokenText(stream);
- stream.print(". ");
- }
- stream.print(value1, value2, value3, ".");
- setErrorMessage(stream.toString());
+ setErrorMessage(stream.toStringWithLatin1Fallback());
}
-template <typename LexerType> template <typename A, typename B, typename C, typename D>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B& value2, const C& value3, const D& value4)
+template <typename LexerType> template <typename... Args>
+void Parser<LexerType>::logError(bool shouldPrintToken, Args&&... args)
{
if (hasError())
return;
@@ -144,109 +101,118 @@ void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B
printUnexpectedTokenText(stream);
stream.print(". ");
}
- stream.print(value1, value2, value3, value4, ".");
- setErrorMessage(stream.toString());
-}
-
-template <typename LexerType> template <typename A, typename B, typename C, typename D, typename E>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B& value2, const C& value3, const D& value4, const E& value5)
-{
- if (hasError())
- return;
- StringPrintStream stream;
- if (shouldPrintToken) {
- printUnexpectedTokenText(stream);
- stream.print(". ");
- }
- stream.print(value1, value2, value3, value4, value5, ".");
- setErrorMessage(stream.toString());
-}
-
-template <typename LexerType> template <typename A, typename B, typename C, typename D, typename E, typename F>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B& value2, const C& value3, const D& value4, const E& value5, const F& value6)
-{
- if (hasError())
- return;
- StringPrintStream stream;
- if (shouldPrintToken) {
- printUnexpectedTokenText(stream);
- stream.print(". ");
- }
- stream.print(value1, value2, value3, value4, value5, value6, ".");
- setErrorMessage(stream.toString());
-}
-
-template <typename LexerType> template <typename A, typename B, typename C, typename D, typename E, typename F, typename G>
-void Parser<LexerType>::logError(bool shouldPrintToken, const A& value1, const B& value2, const C& value3, const D& value4, const E& value5, const F& value6, const G& value7)
-{
- if (hasError())
- return;
- StringPrintStream stream;
- if (shouldPrintToken) {
- printUnexpectedTokenText(stream);
- stream.print(". ");
- }
- stream.print(value1, value2, value3, value4, value5, value6, value7, ".");
- setErrorMessage(stream.toString());
+ stream.print(std::forward<Args>(args)..., ".");
+ setErrorMessage(stream.toStringWithLatin1Fallback());
}
template <typename LexerType>
-Parser<LexerType>::Parser(VM* vm, const SourceCode& source, FunctionParameters* parameters, const Identifier& name, JSParserStrictness strictness, JSParserMode parserMode)
+Parser<LexerType>::Parser(VM* vm, const SourceCode& source, JSParserBuiltinMode builtinMode, JSParserStrictMode strictMode, JSParserScriptMode scriptMode, SourceParseMode parseMode, SuperBinding superBinding, ConstructorKind defaultConstructorKind, DerivedContextType derivedContextType, bool isEvalContext, EvalContextType evalContextType, DebuggerParseData* debuggerParseData)
: m_vm(vm)
, m_source(&source)
, m_hasStackOverflow(false)
, m_allowsIn(true)
- , m_assignmentCount(0)
- , m_nonLHSCount(0)
, m_syntaxAlreadyValidated(source.provider()->isValid())
, m_statementDepth(0)
- , m_nonTrivialExpressionCount(0)
- , m_lastIdentifier(0)
- , m_lastFunctionName(nullptr)
, m_sourceElements(0)
+ , m_parsingBuiltin(builtinMode == JSParserBuiltinMode::Builtin)
+ , m_scriptMode(scriptMode)
+ , m_superBinding(superBinding)
+ , m_defaultConstructorKind(defaultConstructorKind)
+ , m_immediateParentAllowsFunctionDeclarationInStatement(false)
+ , m_debuggerParseData(debuggerParseData)
{
- m_lexer = adoptPtr(new LexerType(vm));
- m_arena = m_vm->parserArena.get();
- m_lexer->setCode(source, m_arena);
- m_token.m_location.line = source.firstLine();
+ m_lexer = std::make_unique<LexerType>(vm, builtinMode, scriptMode);
+ m_lexer->setCode(source, &m_parserArena);
+ m_token.m_location.line = source.firstLine().oneBasedInt();
m_token.m_location.startOffset = source.startOffset();
m_token.m_location.endOffset = source.startOffset();
m_token.m_location.lineStartOffset = source.startOffset();
m_functionCache = vm->addSourceProviderCache(source.provider());
+ m_expressionErrorClassifier = nullptr;
+
ScopeRef scope = pushScope();
- if (parserMode == JSParseFunctionCode)
- scope->setIsFunction();
- if (strictness == JSParseStrict)
- scope->setStrictMode();
- if (parameters) {
- for (unsigned i = 0; i < parameters->size(); i++) {
- auto parameter = parameters->at(i);
- if (!parameter->isBindingNode())
- continue;
- scope->declareParameter(&static_cast<BindingNode*>(parameter)->boundProperty());
- }
+ scope->setSourceParseMode(parseMode);
+ scope->setIsEvalContext(isEvalContext);
+ if (isEvalContext)
+ scope->setEvalContextType(evalContextType);
+
+ if (derivedContextType == DerivedContextType::DerivedConstructorContext) {
+ scope->setConstructorKind(ConstructorKind::Extends);
+ scope->setExpectedSuperBinding(SuperBinding::Needed);
}
- if (!name.isNull())
- scope->declareCallee(&name);
+
+ if (derivedContextType == DerivedContextType::DerivedMethodContext)
+ scope->setExpectedSuperBinding(SuperBinding::Needed);
+
+ if (strictMode == JSParserStrictMode::Strict)
+ scope->setStrictMode();
+
+ if (parseMode == SourceParseMode::ModuleAnalyzeMode || parseMode == SourceParseMode::ModuleEvaluateMode)
+ m_moduleScopeData = ModuleScopeData::create();
+
next();
}
+class Scope::MaybeParseAsGeneratorForScope : public SetForScope<bool> {
+public:
+ MaybeParseAsGeneratorForScope(ScopeRef& scope, bool shouldParseAsGenerator)
+ : SetForScope<bool>(scope->m_isGenerator, shouldParseAsGenerator) { }
+};
+
template <typename LexerType>
Parser<LexerType>::~Parser()
{
}
template <typename LexerType>
-String Parser<LexerType>::parseInner()
+String Parser<LexerType>::parseInner(const Identifier& calleeName, SourceParseMode parseMode)
{
String parseError = String();
-
- ASTBuilder context(const_cast<VM*>(m_vm), const_cast<SourceCode*>(m_source));
- if (m_lexer->isReparsing())
- m_statementDepth--;
+
+ ASTBuilder context(const_cast<VM*>(m_vm), m_parserArena, const_cast<SourceCode*>(m_source));
ScopeRef scope = currentScope();
- SourceElements* sourceElements = parseSourceElements(context, CheckForStrictMode);
- if (!sourceElements || !consume(EOFTOK)) {
+ scope->setIsLexicalScope();
+ SetForScope<FunctionParsePhase> functionParsePhasePoisoner(m_parserState.functionParsePhase, FunctionParsePhase::Body);
+
+ bool isArrowFunctionBodyExpression = parseMode == SourceParseMode::AsyncArrowFunctionBodyMode && !match(OPENBRACE);
+ if (m_lexer->isReparsingFunction()) {
+ ParserFunctionInfo<ASTBuilder> functionInfo;
+ if (isGeneratorOrAsyncFunctionBodyParseMode(parseMode))
+ m_parameters = createGeneratorParameters(context, functionInfo.parameterCount);
+ else
+ m_parameters = parseFunctionParameters(context, parseMode, functionInfo);
+
+ if (SourceParseModeSet(SourceParseMode::ArrowFunctionMode, SourceParseMode::AsyncArrowFunctionMode).contains(parseMode) && !hasError()) {
+ // The only way we could have an error wile reparsing is if we run out of stack space.
+ RELEASE_ASSERT(match(ARROWFUNCTION));
+ next();
+ isArrowFunctionBodyExpression = !match(OPENBRACE);
+ }
+ }
+
+ if (!calleeName.isNull())
+ scope->declareCallee(&calleeName);
+
+ if (m_lexer->isReparsingFunction())
+ m_statementDepth--;
+
+ SourceElements* sourceElements = nullptr;
+ // The only way we can error this early is if we reparse a function and we run out of stack space.
+ if (!hasError()) {
+ if (isAsyncFunctionWrapperParseMode(parseMode))
+ sourceElements = parseAsyncFunctionSourceElements(context, parseMode, isArrowFunctionBodyExpression, CheckForStrictMode);
+ else if (isArrowFunctionBodyExpression)
+ sourceElements = parseArrowFunctionSingleExpressionBodySourceElements(context);
+ else if (isModuleParseMode(parseMode))
+ sourceElements = parseModuleSourceElements(context, parseMode);
+ else if (parseMode == SourceParseMode::GeneratorWrapperFunctionMode)
+ sourceElements = parseGeneratorFunctionSourceElements(context, calleeName, CheckForStrictMode);
+ else
+ sourceElements = parseSourceElements(context, CheckForStrictMode);
+ }
+
+ bool validEnding = consume(EOFTOK);
+ if (!sourceElements || !validEnding) {
if (hasError())
parseError = m_errorMessage;
else
@@ -254,35 +220,93 @@ String Parser<LexerType>::parseInner()
}
IdentifierSet capturedVariables;
- bool modifiedParameter = false;
- scope->getCapturedVariables(capturedVariables, modifiedParameter);
+ UniquedStringImplPtrSet sloppyModeHoistedFunctions;
+ scope->getSloppyModeHoistedFunctions(sloppyModeHoistedFunctions);
+ scope->getCapturedVars(capturedVariables);
+
+ VariableEnvironment& varDeclarations = scope->declaredVariables();
+ for (auto& entry : capturedVariables)
+ varDeclarations.markVariableAsCaptured(entry);
+
+ if (SourceParseModeSet(SourceParseMode::GeneratorWrapperFunctionMode).contains(parseMode) || isAsyncFunctionWrapperParseMode(parseMode)) {
+ if (scope->usedVariablesContains(m_vm->propertyNames->arguments.impl()))
+ context.propagateArgumentsUse();
+ }
+
CodeFeatures features = context.features();
if (scope->strictMode())
features |= StrictModeFeature;
if (scope->shadowsArguments())
features |= ShadowsArgumentsFeature;
- if (modifiedParameter)
- features |= ModifiedParameterFeature;
- didFinishParsing(sourceElements, context.varDeclarations(), context.funcDeclarations(), features,
- context.numConstants(), capturedVariables);
+#ifndef NDEBUG
+ if (m_parsingBuiltin && isProgramParseMode(parseMode)) {
+ VariableEnvironment& lexicalVariables = scope->lexicalVariables();
+ const HashSet<UniquedStringImpl*>& closedVariableCandidates = scope->closedVariableCandidates();
+ for (UniquedStringImpl* candidate : closedVariableCandidates) {
+ if (!lexicalVariables.contains(candidate) && !varDeclarations.contains(candidate) && !candidate->isSymbol()) {
+ dataLog("Bad global capture in builtin: '", candidate, "'\n");
+ dataLog(m_source->view());
+ CRASH();
+ }
+ }
+ }
+#endif // NDEBUG
+ didFinishParsing(sourceElements, scope->takeFunctionDeclarations(), varDeclarations, WTFMove(sloppyModeHoistedFunctions), features, context.numConstants());
return parseError;
}
template <typename LexerType>
-void Parser<LexerType>::didFinishParsing(SourceElements* sourceElements, ParserArenaData<DeclarationStacks::VarStack>* varStack,
- ParserArenaData<DeclarationStacks::FunctionStack>* funcStack, CodeFeatures features, int numConstants, IdentifierSet& capturedVars)
+void Parser<LexerType>::didFinishParsing(SourceElements* sourceElements, DeclarationStacks::FunctionStack&& funcStack,
+ VariableEnvironment& varDeclarations, UniquedStringImplPtrSet&& sloppyModeHoistedFunctions, CodeFeatures features, int numConstants)
{
m_sourceElements = sourceElements;
- m_varDeclarations = varStack;
- m_funcDeclarations = funcStack;
- m_capturedVariables.swap(capturedVars);
+ m_funcDeclarations = WTFMove(funcStack);
+ m_varDeclarations.swap(varDeclarations);
m_features = features;
+ m_sloppyModeHoistedFunctions = WTFMove(sloppyModeHoistedFunctions);
m_numConstants = numConstants;
}
template <typename LexerType>
+bool Parser<LexerType>::isArrowFunctionParameters()
+{
+ if (match(OPENPAREN)) {
+ SavePoint saveArrowFunctionPoint = createSavePoint();
+ next();
+ bool isArrowFunction = false;
+ if (match(CLOSEPAREN)) {
+ next();
+ isArrowFunction = match(ARROWFUNCTION);
+ } else {
+ SyntaxChecker syntaxChecker(const_cast<VM*>(m_vm), m_lexer.get());
+ // We make fake scope, otherwise parseFormalParameters will add variable to current scope that lead to errors
+ AutoPopScopeRef fakeScope(this, pushScope());
+ fakeScope->setSourceParseMode(SourceParseMode::ArrowFunctionMode);
+
+ unsigned parametersCount = 0;
+ bool isArrowFunctionParameterList = true;
+ isArrowFunction = parseFormalParameters(syntaxChecker, syntaxChecker.createFormalParameterList(), isArrowFunctionParameterList, parametersCount) && consume(CLOSEPAREN) && match(ARROWFUNCTION);
+ propagateError();
+ popScope(fakeScope, syntaxChecker.NeedsFreeVariableInfo);
+ }
+ restoreSavePoint(saveArrowFunctionPoint);
+ return isArrowFunction;
+ }
+
+ if (matchSpecIdentifier()) {
+ SavePoint saveArrowFunctionPoint = createSavePoint();
+ next();
+ bool isArrowFunction = match(ARROWFUNCTION);
+ restoreSavePoint(saveArrowFunctionPoint);
+ return isArrowFunction;
+ }
+
+ return false;
+}
+
+template <typename LexerType>
bool Parser<LexerType>::allowAutomaticSemicolon()
{
return match(CLOSEBRACE) || match(EOFTOK) || m_lexer->prevTerminator();
@@ -293,37 +317,47 @@ template <class TreeBuilder> TreeSourceElements Parser<LexerType>::parseSourceEl
{
const unsigned lengthOfUseStrictLiteral = 12; // "use strict".length
TreeSourceElements sourceElements = context.createSourceElements();
- bool seenNonDirective = false;
const Identifier* directive = 0;
unsigned directiveLiteralLength = 0;
auto savePoint = createSavePoint();
- bool hasSetStrict = false;
- while (TreeStatement statement = parseStatement(context, directive, &directiveLiteralLength)) {
- if (mode == CheckForStrictMode && !seenNonDirective) {
+ bool shouldCheckForUseStrict = mode == CheckForStrictMode;
+
+ while (TreeStatement statement = parseStatementListItem(context, directive, &directiveLiteralLength)) {
+ if (shouldCheckForUseStrict) {
if (directive) {
// "use strict" must be the exact literal without escape sequences or line continuation.
- if (!hasSetStrict && directiveLiteralLength == lengthOfUseStrictLiteral && m_vm->propertyNames->useStrictIdentifier == *directive) {
+ if (directiveLiteralLength == lengthOfUseStrictLiteral && m_vm->propertyNames->useStrictIdentifier == *directive) {
setStrictMode();
- hasSetStrict = true;
+ shouldCheckForUseStrict = false; // We saw "use strict", there is no need to keep checking for it.
if (!isValidStrictMode()) {
- if (m_lastFunctionName) {
- if (m_vm->propertyNames->arguments == *m_lastFunctionName)
+ if (m_parserState.lastFunctionName) {
+ if (m_vm->propertyNames->arguments == *m_parserState.lastFunctionName)
semanticFail("Cannot name a function 'arguments' in strict mode");
- if (m_vm->propertyNames->eval == *m_lastFunctionName)
+ if (m_vm->propertyNames->eval == *m_parserState.lastFunctionName)
semanticFail("Cannot name a function 'eval' in strict mode");
}
if (hasDeclaredVariable(m_vm->propertyNames->arguments))
semanticFail("Cannot declare a variable named 'arguments' in strict mode");
if (hasDeclaredVariable(m_vm->propertyNames->eval))
semanticFail("Cannot declare a variable named 'eval' in strict mode");
+ semanticFailIfTrue(currentScope()->hasNonSimpleParameterList(), "'use strict' directive not allowed inside a function with a non-simple parameter list");
semanticFailIfFalse(isValidStrictMode(), "Invalid parameters or function name in strict mode");
}
+ // Since strict mode is changed, restoring lexer state by calling next() may cause errors.
restoreSavePoint(savePoint);
propagateError();
continue;
}
- } else
- seenNonDirective = true;
+
+ // We saw a directive, but it wasn't "use strict". We reset our state to
+ // see if the next statement we parse is also a directive.
+ directive = nullptr;
+ } else {
+ // We saw a statement that wasn't in the form of a directive. The spec says that "use strict"
+ // is only allowed as the first statement, or after a sequence of directives before it, but
+ // not after non-directive statements.
+ shouldCheckForUseStrict = false;
+ }
}
context.appendStatement(sourceElements, statement);
}
@@ -333,35 +367,270 @@ template <class TreeBuilder> TreeSourceElements Parser<LexerType>::parseSourceEl
}
template <typename LexerType>
-template <class TreeBuilder> TreeStatement Parser<LexerType>::parseVarDeclaration(TreeBuilder& context)
+template <class TreeBuilder> TreeSourceElements Parser<LexerType>::parseModuleSourceElements(TreeBuilder& context, SourceParseMode parseMode)
{
- ASSERT(match(VAR));
- JSTokenLocation location(tokenLocation());
- int start = tokenLine();
- int end = 0;
- int scratch;
- TreeDeconstructionPattern scratch1 = 0;
- TreeExpression scratch2 = 0;
- JSTextPosition scratch3;
- TreeExpression varDecls = parseVarDeclarationList(context, scratch, scratch1, scratch2, scratch3, scratch3, scratch3);
+ TreeSourceElements sourceElements = context.createSourceElements();
+ SyntaxChecker syntaxChecker(const_cast<VM*>(m_vm), m_lexer.get());
+
+ while (true) {
+ TreeStatement statement = 0;
+ switch (m_token.m_type) {
+ case EXPORT:
+ statement = parseExportDeclaration(context);
+ if (statement)
+ recordPauseLocation(context.breakpointLocation(statement));
+ break;
+
+ case IMPORT: {
+ SavePoint savePoint = createSavePoint();
+ next();
+ bool isImportDeclaration = !match(OPENPAREN);
+ restoreSavePoint(savePoint);
+ if (isImportDeclaration) {
+ statement = parseImportDeclaration(context);
+ if (statement)
+ recordPauseLocation(context.breakpointLocation(statement));
+ break;
+ }
+
+ // This is `import("...")` call case.
+ FALLTHROUGH;
+ }
+
+ default: {
+ const Identifier* directive = 0;
+ unsigned directiveLiteralLength = 0;
+ if (parseMode == SourceParseMode::ModuleAnalyzeMode) {
+ if (!parseStatementListItem(syntaxChecker, directive, &directiveLiteralLength))
+ goto end;
+ continue;
+ }
+ statement = parseStatementListItem(context, directive, &directiveLiteralLength);
+ break;
+ }
+ }
+
+ if (!statement)
+ goto end;
+ context.appendStatement(sourceElements, statement);
+ }
+
+end:
propagateError();
- failIfFalse(autoSemiColon(), "Expected ';' after var declaration");
-
- return context.createVarStatement(location, varDecls, start, end);
+
+ for (const auto& pair : m_moduleScopeData->exportedBindings()) {
+ const auto& uid = pair.key;
+ if (currentScope()->hasDeclaredVariable(uid)) {
+ currentScope()->declaredVariables().markVariableAsExported(uid);
+ continue;
+ }
+
+ if (currentScope()->hasLexicallyDeclaredVariable(uid)) {
+ currentScope()->lexicalVariables().markVariableAsExported(uid);
+ continue;
+ }
+
+ semanticFail("Exported binding '", uid.get(), "' needs to refer to a top-level declared variable");
+ }
+
+ return sourceElements;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeSourceElements Parser<LexerType>::parseGeneratorFunctionSourceElements(TreeBuilder& context, const Identifier& name, SourceElementsMode mode)
+{
+ auto sourceElements = context.createSourceElements();
+
+ unsigned functionKeywordStart = tokenStart();
+ JSTokenLocation startLocation(tokenLocation());
+ JSTextPosition start = tokenStartPosition();
+ unsigned startColumn = tokenColumn();
+ int functionNameStart = m_token.m_location.startOffset;
+ int parametersStart = m_token.m_location.startOffset;
+
+ ParserFunctionInfo<TreeBuilder> info;
+ info.name = &m_vm->propertyNames->nullIdentifier;
+ createGeneratorParameters(context, info.parameterCount);
+ info.startOffset = parametersStart;
+ info.startLine = tokenLine();
+
+ {
+ AutoPopScopeRef generatorBodyScope(this, pushScope());
+ generatorBodyScope->setSourceParseMode(SourceParseMode::GeneratorBodyMode);
+ generatorBodyScope->setConstructorKind(ConstructorKind::None);
+ generatorBodyScope->setExpectedSuperBinding(m_superBinding);
+
+ SyntaxChecker generatorFunctionContext(const_cast<VM*>(m_vm), m_lexer.get());
+ failIfFalse(parseSourceElements(generatorFunctionContext, mode), "Cannot parse the body of a generator");
+ popScope(generatorBodyScope, TreeBuilder::NeedsFreeVariableInfo);
+ }
+ info.body = context.createFunctionMetadata(startLocation, tokenLocation(), startColumn, tokenColumn(), functionKeywordStart, functionNameStart, parametersStart, strictMode(), ConstructorKind::None, m_superBinding, info.parameterCount, SourceParseMode::GeneratorBodyMode, false);
+
+ info.endLine = tokenLine();
+ info.endOffset = m_token.m_data.offset;
+ info.parametersStartColumn = startColumn;
+
+ auto functionExpr = context.createGeneratorFunctionBody(startLocation, info, name);
+ auto statement = context.createExprStatement(startLocation, functionExpr, start, m_lastTokenEndPosition.line);
+ context.appendStatement(sourceElements, statement);
+
+ return sourceElements;
}
template <typename LexerType>
-template <class TreeBuilder> TreeStatement Parser<LexerType>::parseConstDeclaration(TreeBuilder& context)
+template <class TreeBuilder> TreeSourceElements Parser<LexerType>::parseAsyncFunctionSourceElements(TreeBuilder& context, SourceParseMode parseMode, bool isArrowFunctionBodyExpression, SourceElementsMode mode)
{
- ASSERT(match(CONSTTOKEN));
+ ASSERT(isAsyncFunctionWrapperParseMode(parseMode));
+ auto sourceElements = context.createSourceElements();
+
+ unsigned functionKeywordStart = tokenStart();
+ JSTokenLocation startLocation(tokenLocation());
+ JSTextPosition start = tokenStartPosition();
+ unsigned startColumn = tokenColumn();
+ int functionNameStart = m_token.m_location.startOffset;
+ int parametersStart = m_token.m_location.startOffset;
+
+ ParserFunctionInfo<TreeBuilder> info;
+ info.name = &m_vm->propertyNames->nullIdentifier;
+ createGeneratorParameters(context, info.parameterCount);
+ info.startOffset = parametersStart;
+ info.startLine = tokenLine();
+ SourceParseMode innerParseMode = parseMode == SourceParseMode::AsyncArrowFunctionMode
+ ? SourceParseMode::AsyncArrowFunctionBodyMode
+ : SourceParseMode::AsyncFunctionBodyMode;
+ {
+ AutoPopScopeRef asyncFunctionBodyScope(this, pushScope());
+ asyncFunctionBodyScope->setSourceParseMode(innerParseMode);
+ SyntaxChecker syntaxChecker(const_cast<VM*>(m_vm), m_lexer.get());
+ if (isArrowFunctionBodyExpression) {
+ if (m_debuggerParseData)
+ failIfFalse(parseArrowFunctionSingleExpressionBodySourceElements(context), "Cannot parse the body of async arrow function");
+ else
+ failIfFalse(parseArrowFunctionSingleExpressionBodySourceElements(syntaxChecker), "Cannot parse the body of async arrow function");
+ } else {
+ if (m_debuggerParseData)
+ failIfFalse(parseSourceElements(context, mode), "Cannot parse the body of async function");
+ else
+ failIfFalse(parseSourceElements(syntaxChecker, mode), "Cannot parse the body of async function");
+ }
+ popScope(asyncFunctionBodyScope, TreeBuilder::NeedsFreeVariableInfo);
+ }
+ info.body = context.createFunctionMetadata(startLocation, tokenLocation(), startColumn, tokenColumn(), functionKeywordStart, functionNameStart, parametersStart, strictMode(), ConstructorKind::None, m_superBinding, info.parameterCount, innerParseMode, isArrowFunctionBodyExpression);
+
+ info.endLine = tokenLine();
+ info.endOffset = isArrowFunctionBodyExpression ? tokenLocation().endOffset : m_token.m_data.offset;
+ info.parametersStartColumn = startColumn;
+
+ auto functionExpr = context.createAsyncFunctionBody(startLocation, info, innerParseMode);
+ auto statement = context.createExprStatement(startLocation, functionExpr, start, m_lastTokenEndPosition.line);
+ context.appendStatement(sourceElements, statement);
+
+ return sourceElements;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseStatementListItem(TreeBuilder& context, const Identifier*& directive, unsigned* directiveLiteralLength)
+{
+ // The grammar is documented here:
+ // http://www.ecma-international.org/ecma-262/6.0/index.html#sec-statements
+ DepthManager statementDepth(&m_statementDepth);
+ m_statementDepth++;
+ TreeStatement result = 0;
+ bool shouldSetEndOffset = true;
+ bool shouldSetPauseLocation = false;
+
+ switch (m_token.m_type) {
+ case CONSTTOKEN:
+ result = parseVariableDeclaration(context, DeclarationType::ConstDeclaration);
+ shouldSetPauseLocation = true;
+ break;
+ case LET: {
+ bool shouldParseVariableDeclaration = true;
+ if (!strictMode()) {
+ SavePoint savePoint = createSavePoint();
+ next();
+ // Intentionally use `isIdentifierOrAnyContextualKeyword(m_token)` and don't use `matchSpecIdentifier()`.
+ // We would like to fall into parseVariableDeclaration path even if "yield" is not treated as an Identifier.
+ // For example, under a generator context, matchSpecIdentifier() for "yield" returns `false`.
+ // But we would like to enter parseVariableDeclaration and raise an error under the context of parseVariableDeclaration
+ // to raise consistent errors between "var", "const" and "let".
+ if (!isIdentifierOrAnyContextualKeyword(m_token) && !match(OPENBRACE) && !match(OPENBRACKET))
+ shouldParseVariableDeclaration = false;
+ restoreSavePoint(savePoint);
+ }
+ if (shouldParseVariableDeclaration)
+ result = parseVariableDeclaration(context, DeclarationType::LetDeclaration);
+ else {
+ bool allowFunctionDeclarationAsStatement = true;
+ result = parseExpressionOrLabelStatement(context, allowFunctionDeclarationAsStatement);
+ }
+ shouldSetPauseLocation = !context.shouldSkipPauseLocation(result);
+ break;
+ }
+ case CLASSTOKEN:
+ result = parseClassDeclaration(context);
+ break;
+ case FUNCTION:
+ result = parseFunctionDeclaration(context);
+ break;
+ case ASYNC: {
+ // Eagerly parse as AsyncFunctionDeclaration. This is the uncommon case,
+ // but could be mistakenly parsed as an AsyncFunctionExpression.
+ SavePoint savePoint = createSavePoint();
+ next();
+ if (UNLIKELY(match(FUNCTION) && !m_lexer->prevTerminator())) {
+ result = parseAsyncFunctionDeclaration(context);
+ break;
+ }
+ restoreSavePoint(savePoint);
+ FALLTHROUGH;
+ }
+ case IDENT:
+ case AWAIT:
+ case YIELD: {
+ // This is a convenient place to notice labeled statements
+ // (even though we also parse them as normal statements)
+ // because we allow the following type of code in sloppy mode:
+ // ``` function foo() { label: function bar() { } } ```
+ bool allowFunctionDeclarationAsStatement = true;
+ result = parseExpressionOrLabelStatement(context, allowFunctionDeclarationAsStatement);
+ shouldSetPauseLocation = !context.shouldSkipPauseLocation(result);
+ break;
+ }
+ default:
+ m_statementDepth--; // parseStatement() increments the depth.
+ result = parseStatement(context, directive, directiveLiteralLength);
+ shouldSetEndOffset = false;
+ break;
+ }
+
+ if (result) {
+ if (shouldSetEndOffset)
+ context.setEndOffset(result, m_lastTokenEndPosition.offset);
+ if (shouldSetPauseLocation)
+ recordPauseLocation(context.breakpointLocation(result));
+ }
+
+ return result;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseVariableDeclaration(TreeBuilder& context, DeclarationType declarationType, ExportType exportType)
+{
+ ASSERT(match(VAR) || match(LET) || match(CONSTTOKEN));
JSTokenLocation location(tokenLocation());
int start = tokenLine();
int end = 0;
- TreeConstDeclList constDecls = parseConstDeclarationList(context);
+ int scratch;
+ TreeDestructuringPattern scratch1 = 0;
+ TreeExpression scratch2 = 0;
+ JSTextPosition scratch3;
+ bool scratchBool;
+ TreeExpression variableDecls = parseVariableDeclarationList(context, scratch, scratch1, scratch2, scratch3, scratch3, scratch3, VarDeclarationContext, declarationType, exportType, scratchBool);
propagateError();
- failIfFalse(autoSemiColon(), "Expected ';' after const declaration");
+ failIfFalse(autoSemiColon(), "Expected ';' after variable declaration");
- return context.createConstStatement(location, constDecls, start, end);
+ return context.createDeclarationStatement(location, variableDecls, start, end);
}
template <typename LexerType>
@@ -382,6 +651,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseDoWhileStatem
semanticFailIfTrue(match(CLOSEPAREN), "Must provide an expression as a do-while loop condition");
TreeExpression expr = parseExpression(context);
failIfFalse(expr, "Unable to parse do-while loop condition");
+ recordPauseLocation(context.breakpointLocation(expr));
handleProductionOrFail(CLOSEPAREN, ")", "end", "do-while loop condition");
if (match(SEMICOLON))
next(); // Always performs automatic semicolon insertion.
@@ -400,9 +670,10 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseWhileStatemen
semanticFailIfTrue(match(CLOSEPAREN), "Must provide an expression as a while loop condition");
TreeExpression expr = parseExpression(context);
failIfFalse(expr, "Unable to parse while loop condition");
+ recordPauseLocation(context.breakpointLocation(expr));
int endLine = tokenLine();
handleProductionOrFail(CLOSEPAREN, ")", "end", "while loop condition");
-
+
const Identifier* unused = 0;
startLoop();
TreeStatement statement = parseStatement(context, unused);
@@ -412,261 +683,435 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseWhileStatemen
}
template <typename LexerType>
-template <class TreeBuilder> TreeExpression Parser<LexerType>::parseVarDeclarationList(TreeBuilder& context, int& declarations, TreeDeconstructionPattern& lastPattern, TreeExpression& lastInitializer, JSTextPosition& identStart, JSTextPosition& initStart, JSTextPosition& initEnd)
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseVariableDeclarationList(TreeBuilder& context, int& declarations, TreeDestructuringPattern& lastPattern, TreeExpression& lastInitializer, JSTextPosition& identStart, JSTextPosition& initStart, JSTextPosition& initEnd, VarDeclarationListContext declarationListContext, DeclarationType declarationType, ExportType exportType, bool& forLoopConstDoesNotHaveInitializer)
{
- TreeExpression varDecls = 0;
+ ASSERT(declarationType == DeclarationType::LetDeclaration || declarationType == DeclarationType::VarDeclaration || declarationType == DeclarationType::ConstDeclaration);
+ TreeExpression head = 0;
+ TreeExpression tail = 0;
const Identifier* lastIdent;
+ JSToken lastIdentToken;
+ AssignmentContext assignmentContext = assignmentContextFromDeclarationType(declarationType);
do {
lastIdent = 0;
- lastPattern = 0;
+ lastPattern = TreeDestructuringPattern(0);
JSTokenLocation location(tokenLocation());
next();
TreeExpression node = 0;
declarations++;
bool hasInitializer = false;
- if (match(IDENT)) {
+ if (matchSpecIdentifier()) {
+ failIfTrue(match(LET) && (declarationType == DeclarationType::LetDeclaration || declarationType == DeclarationType::ConstDeclaration),
+ "Can't use 'let' as an identifier name for a LexicalDeclaration");
+ semanticFailIfTrue(isDisallowedIdentifierAwait(m_token), "Can't use 'await' as a ", declarationTypeToVariableKind(declarationType), " ", disallowedIdentifierAwaitReason());
JSTextPosition varStart = tokenStartPosition();
+ JSTokenLocation varStartLocation(tokenLocation());
identStart = varStart;
const Identifier* name = m_token.m_data.ident;
lastIdent = name;
+ lastIdentToken = m_token;
next();
hasInitializer = match(EQUAL);
- failIfFalseIfStrict(declareVariable(name), "Cannot declare a variable named ", name->impl(), " in strict mode");
- context.addVar(name, (hasInitializer || (!m_allowsIn && (match(INTOKEN) || isofToken()))) ? DeclarationStacks::HasInitializer : 0);
+ DeclarationResultMask declarationResult = declareVariable(name, declarationType);
+ if (declarationResult != DeclarationResult::Valid) {
+ failIfTrueIfStrict(declarationResult & DeclarationResult::InvalidStrictMode, "Cannot declare a variable named ", name->impl(), " in strict mode");
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration) {
+ if (declarationType == DeclarationType::LetDeclaration)
+ internalFailWithMessage(false, "Cannot declare a let variable twice: '", name->impl(), "'");
+ if (declarationType == DeclarationType::ConstDeclaration)
+ internalFailWithMessage(false, "Cannot declare a const variable twice: '", name->impl(), "'");
+ ASSERT(declarationType == DeclarationType::VarDeclaration);
+ internalFailWithMessage(false, "Cannot declare a var variable that shadows a let/const/class variable: '", name->impl(), "'");
+ }
+ }
+ if (exportType == ExportType::Exported) {
+ semanticFailIfFalse(exportName(*name), "Cannot export a duplicate name '", name->impl(), "'");
+ m_moduleScopeData->exportBinding(*name);
+ }
+
if (hasInitializer) {
JSTextPosition varDivot = tokenStartPosition() + 1;
initStart = tokenStartPosition();
next(TreeBuilder::DontBuildStrings); // consume '='
+ propagateError();
TreeExpression initializer = parseAssignmentExpression(context);
initEnd = lastTokenEndPosition();
lastInitializer = initializer;
failIfFalse(initializer, "Expected expression as the intializer for the variable '", name->impl(), "'");
- node = context.createAssignResolve(location, *name, initializer, varStart, varDivot, lastTokenEndPosition());
+ node = context.createAssignResolve(location, *name, initializer, varStart, varDivot, lastTokenEndPosition(), assignmentContext);
+ } else {
+ if (declarationListContext == ForLoopContext && declarationType == DeclarationType::ConstDeclaration)
+ forLoopConstDoesNotHaveInitializer = true;
+ failIfTrue(declarationListContext != ForLoopContext && declarationType == DeclarationType::ConstDeclaration, "const declared variable '", name->impl(), "'", " must have an initializer");
+ if (declarationType == DeclarationType::VarDeclaration)
+ node = context.createEmptyVarExpression(varStartLocation, *name);
+ else
+ node = context.createEmptyLetExpression(varStartLocation, *name);
}
} else {
lastIdent = 0;
- auto pattern = parseDeconstructionPattern(context, DeconstructToVariables);
- failIfFalse(pattern, "Cannot parse this deconstruction pattern");
+ auto pattern = parseDestructuringPattern(context, destructuringKindFromDeclarationType(declarationType), exportType, nullptr, nullptr, assignmentContext);
+ failIfFalse(pattern, "Cannot parse this destructuring pattern");
hasInitializer = match(EQUAL);
+ failIfTrue(declarationListContext == VarDeclarationContext && !hasInitializer, "Expected an initializer in destructuring variable declaration");
lastPattern = pattern;
if (hasInitializer) {
next(TreeBuilder::DontBuildStrings); // consume '='
- TreeExpression rhs = parseExpression(context);
- node = context.createDeconstructingAssignment(location, pattern, rhs);
+ TreeExpression rhs = parseAssignmentExpression(context);
+ propagateError();
+ ASSERT(rhs);
+ node = context.createDestructuringAssignment(location, pattern, rhs);
+ lastInitializer = rhs;
}
}
-
- if (hasInitializer) {
- if (!varDecls)
- varDecls = node;
- else
- varDecls = context.combineCommaNodes(location, varDecls, node);
+
+ if (node) {
+ if (!head)
+ head = node;
+ else if (!tail) {
+ head = context.createCommaExpr(location, head);
+ tail = context.appendToCommaExpr(location, head, head, node);
+ } else
+ tail = context.appendToCommaExpr(location, head, tail, node);
}
} while (match(COMMA));
if (lastIdent)
- lastPattern = createBindingPattern(context, DeconstructToVariables, *lastIdent, 0);
- return varDecls;
+ lastPattern = context.createBindingLocation(lastIdentToken.m_location, *lastIdent, lastIdentToken.m_startPosition, lastIdentToken.m_endPosition, assignmentContext);
+
+ return head;
+}
+
+template <typename LexerType>
+bool Parser<LexerType>::declareRestOrNormalParameter(const Identifier& name, const Identifier** duplicateIdentifier)
+{
+ DeclarationResultMask declarationResult = declareParameter(&name);
+ if ((declarationResult & DeclarationResult::InvalidStrictMode) && strictMode()) {
+ semanticFailIfTrue(isEvalOrArguments(&name), "Cannot destructure to a parameter name '", name.impl(), "' in strict mode");
+ if (m_parserState.lastFunctionName && name == *m_parserState.lastFunctionName)
+ semanticFail("Cannot declare a parameter named '", name.impl(), "' as it shadows the name of a strict mode function");
+ semanticFailureDueToKeyword("parameter name");
+ if (!m_lexer->isReparsingFunction() && hasDeclaredParameter(name))
+ semanticFail("Cannot declare a parameter named '", name.impl(), "' in strict mode as it has already been declared");
+ semanticFail("Cannot declare a parameter named '", name.impl(), "' in strict mode");
+ }
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration) {
+ // It's not always an error to define a duplicate parameter.
+ // It's only an error when there are default parameter values or destructuring parameters.
+ // We note this value now so we can check it later.
+ if (duplicateIdentifier)
+ *duplicateIdentifier = &name;
+ }
+
+ return true;
}
template <typename LexerType>
-template <class TreeBuilder> TreeDeconstructionPattern Parser<LexerType>::createBindingPattern(TreeBuilder& context, DeconstructionKind kind, const Identifier& name, int depth)
+template <class TreeBuilder> TreeDestructuringPattern Parser<LexerType>::createBindingPattern(TreeBuilder& context, DestructuringKind kind, ExportType exportType, const Identifier& name, JSToken token, AssignmentContext bindingContext, const Identifier** duplicateIdentifier)
{
- ASSERT(!name.isEmpty());
ASSERT(!name.isNull());
- ASSERT(name.impl()->isIdentifier());
- if (depth) {
- if (kind == DeconstructToVariables)
- failIfFalseIfStrict(declareVariable(&name), "Cannot deconstruct to a variable named '", name.impl(), "' in strict mode");
- if (kind == DeconstructToParameters) {
- auto bindingResult = declareBoundParameter(&name);
- if (bindingResult == Scope::StrictBindingFailed && strictMode()) {
- semanticFailIfTrue(m_vm->propertyNames->arguments == name || m_vm->propertyNames->eval == name, "Cannot deconstruct to a parameter name '", name.impl(), "' in strict mode");
- if (m_lastFunctionName && name == *m_lastFunctionName)
- semanticFail("Cannot deconstruct to '", name.impl(), "' as it shadows the name of a strict mode function");
- semanticFailureDueToKeyword("bound parameter name");
- if (hasDeclaredParameter(name))
- semanticFail("Cannot deconstruct to '", name.impl(), "' as it has already been declared");
- semanticFail("Cannot bind to a parameter named '", name.impl(), "' in strict mode");
- }
- if (bindingResult == Scope::BindingFailed) {
- semanticFailureDueToKeyword("bound parameter name");
- if (hasDeclaredParameter(name))
- semanticFail("Cannot deconstruct to '", name.impl(), "' as it has already been declared");
- semanticFail("Cannot deconstruct to a parameter named '", name.impl(), "'");
- }
- }
- if (kind != DeconstructToExpressions)
- context.addVar(&name, kind == DeconstructToParameters ? 0 : DeclarationStacks::HasInitializer);
- } else {
- if (kind == DeconstructToVariables) {
- failIfFalseIfStrict(declareVariable(&name), "Cannot declare a variable named '", name.impl(), "' in strict mode");
- context.addVar(&name, DeclarationStacks::HasInitializer);
- }
-
- if (kind == DeconstructToParameters) {
- bool declarationResult = declareParameter(&name);
- if (!declarationResult && strictMode()) {
- semanticFailIfTrue(m_vm->propertyNames->arguments == name || m_vm->propertyNames->eval == name, "Cannot deconstruct to a parameter name '", name.impl(), "' in strict mode");
- if (m_lastFunctionName && name == *m_lastFunctionName)
- semanticFail("Cannot declare a parameter named '", name.impl(), "' as it shadows the name of a strict mode function");
- semanticFailureDueToKeyword("parameter name");
- if (hasDeclaredParameter(name))
- semanticFail("Cannot declare a parameter named '", name.impl(), "' in strict mode as it has already been declared");
- semanticFail("Cannot declare a parameter named '", name.impl(), "' in strict mode");
- }
+ ASSERT(name.impl()->isAtomic() || name.impl()->isSymbol());
+
+ switch (kind) {
+ case DestructuringKind::DestructureToVariables: {
+ DeclarationResultMask declarationResult = declareVariable(&name);
+ failIfTrueIfStrict(declarationResult & DeclarationResult::InvalidStrictMode, "Cannot declare a variable named '", name.impl(), "' in strict mode");
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration)
+ internalFailWithMessage(false, "Cannot declare a var variable that shadows a let/const/class variable: '", name.impl(), "'");
+ break;
+ }
+
+ case DestructuringKind::DestructureToLet:
+ case DestructuringKind::DestructureToConst:
+ case DestructuringKind::DestructureToCatchParameters: {
+ DeclarationResultMask declarationResult = declareVariable(&name, kind == DestructuringKind::DestructureToConst ? DeclarationType::ConstDeclaration : DeclarationType::LetDeclaration);
+ if (declarationResult != DeclarationResult::Valid) {
+ failIfTrueIfStrict(declarationResult & DeclarationResult::InvalidStrictMode, "Cannot destructure to a variable named '", name.impl(), "' in strict mode");
+ failIfTrue(declarationResult & DeclarationResult::InvalidDuplicateDeclaration, "Cannot declare a lexical variable twice: '", name.impl(), "'");
}
+ break;
+ }
+
+ case DestructuringKind::DestructureToParameters: {
+ declareRestOrNormalParameter(name, duplicateIdentifier);
+ propagateError();
+ break;
}
- return context.createBindingLocation(m_token.m_location, name, m_token.m_startPosition, m_token.m_endPosition);
+
+ case DestructuringKind::DestructureToExpressions: {
+ break;
+ }
+ }
+
+ if (exportType == ExportType::Exported) {
+ semanticFailIfFalse(exportName(name), "Cannot export a duplicate name '", name.impl(), "'");
+ m_moduleScopeData->exportBinding(name);
+ }
+ return context.createBindingLocation(token.m_location, name, token.m_startPosition, token.m_endPosition, bindingContext);
}
template <typename LexerType>
-template <class TreeBuilder> TreeDeconstructionPattern Parser<LexerType>::tryParseDeconstructionPatternExpression(TreeBuilder& context)
+template <class TreeBuilder> NEVER_INLINE TreeDestructuringPattern Parser<LexerType>::createAssignmentElement(TreeBuilder& context, TreeExpression& assignmentTarget, const JSTextPosition& startPosition, const JSTextPosition& endPosition)
{
- return parseDeconstructionPattern(context, DeconstructToExpressions);
+ return context.createAssignmentElement(assignmentTarget, startPosition, endPosition);
}
template <typename LexerType>
-template <class TreeBuilder> TreeDeconstructionPattern Parser<LexerType>::parseDeconstructionPattern(TreeBuilder& context, DeconstructionKind kind, int depth)
+template <class TreeBuilder> TreeSourceElements Parser<LexerType>::parseArrowFunctionSingleExpressionBodySourceElements(TreeBuilder& context)
{
+ ASSERT(!match(OPENBRACE));
+
+ JSTokenLocation location(tokenLocation());
+ JSTextPosition start = tokenStartPosition();
+
failIfStackOverflow();
- int nonLHSCount = m_nonLHSCount;
- TreeDeconstructionPattern pattern;
+ TreeExpression expr = parseAssignmentExpression(context);
+ failIfFalse(expr, "Cannot parse the arrow function expression");
+
+ context.setEndOffset(expr, m_lastTokenEndPosition.offset);
+
+ JSTextPosition end = tokenEndPosition();
+
+ TreeSourceElements sourceElements = context.createSourceElements();
+ TreeStatement body = context.createReturnStatement(location, expr, start, end);
+ context.setEndOffset(body, m_lastTokenEndPosition.offset);
+ recordPauseLocation(context.breakpointLocation(body));
+ context.appendStatement(sourceElements, body);
+
+ return sourceElements;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeDestructuringPattern Parser<LexerType>::tryParseDestructuringPatternExpression(TreeBuilder& context, AssignmentContext bindingContext)
+{
+ return parseDestructuringPattern(context, DestructuringKind::DestructureToExpressions, ExportType::NotExported, nullptr, nullptr, bindingContext);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeDestructuringPattern Parser<LexerType>::parseBindingOrAssignmentElement(TreeBuilder& context, DestructuringKind kind, ExportType exportType, const Identifier** duplicateIdentifier, bool* hasDestructuringPattern, AssignmentContext bindingContext, int depth)
+{
+ if (kind == DestructuringKind::DestructureToExpressions)
+ return parseAssignmentElement(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth);
+ return parseDestructuringPattern(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeDestructuringPattern Parser<LexerType>::parseAssignmentElement(TreeBuilder& context, DestructuringKind kind, ExportType exportType, const Identifier** duplicateIdentifier, bool* hasDestructuringPattern, AssignmentContext bindingContext, int depth)
+{
+ TreeDestructuringPattern assignmentTarget = 0;
+
+ if (match(OPENBRACE) || match(OPENBRACKET)) {
+ SavePoint savePoint = createSavePoint();
+ assignmentTarget = parseDestructuringPattern(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth);
+ if (assignmentTarget && !match(DOT) && !match(OPENBRACKET) && !match(OPENPAREN) && !match(BACKQUOTE))
+ return assignmentTarget;
+ restoreSavePoint(savePoint);
+ }
+
+ JSTextPosition startPosition = tokenStartPosition();
+ auto element = parseMemberExpression(context);
+
+ semanticFailIfFalse(element && context.isAssignmentLocation(element), "Invalid destructuring assignment target");
+
+ if (strictMode() && m_parserState.lastIdentifier && context.isResolve(element)) {
+ bool isEvalOrArguments = m_vm->propertyNames->eval == *m_parserState.lastIdentifier || m_vm->propertyNames->arguments == *m_parserState.lastIdentifier;
+ failIfTrueIfStrict(isEvalOrArguments, "Cannot modify '", m_parserState.lastIdentifier->impl(), "' in strict mode");
+ }
+
+ return createAssignmentElement(context, element, startPosition, lastTokenEndPosition());
+}
+
+static const char* destructuringKindToVariableKindName(DestructuringKind kind)
+{
+ switch (kind) {
+ case DestructuringKind::DestructureToLet:
+ case DestructuringKind::DestructureToConst:
+ return "lexical variable name";
+ case DestructuringKind::DestructureToVariables:
+ return "variable name";
+ case DestructuringKind::DestructureToParameters:
+ return "parameter name";
+ case DestructuringKind::DestructureToCatchParameters:
+ return "catch parameter name";
+ case DestructuringKind::DestructureToExpressions:
+ return "expression name";
+ }
+ RELEASE_ASSERT_NOT_REACHED();
+ return "invalid";
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeDestructuringPattern Parser<LexerType>::parseDestructuringPattern(TreeBuilder& context, DestructuringKind kind, ExportType exportType, const Identifier** duplicateIdentifier, bool* hasDestructuringPattern, AssignmentContext bindingContext, int depth)
+{
+ failIfStackOverflow();
+ int nonLHSCount = m_parserState.nonLHSCount;
+ TreeDestructuringPattern pattern;
switch (m_token.m_type) {
case OPENBRACKET: {
+ JSTextPosition divotStart = tokenStartPosition();
auto arrayPattern = context.createArrayPattern(m_token.m_location);
next();
- if (kind == DeconstructToExpressions && match(CLOSEBRACKET))
- return 0;
- failIfTrue(match(CLOSEBRACKET), "There must be at least one bound property in an array deconstruction pattern");
+
+ if (hasDestructuringPattern)
+ *hasDestructuringPattern = true;
+
+ bool restElementWasFound = false;
+
do {
while (match(COMMA)) {
context.appendArrayPatternSkipEntry(arrayPattern, m_token.m_location);
next();
}
propagateError();
+
+ if (match(CLOSEBRACKET))
+ break;
+
+ if (UNLIKELY(match(DOTDOTDOT))) {
+ JSTokenLocation location = m_token.m_location;
+ next();
+ auto innerPattern = parseBindingOrAssignmentElement(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth + 1);
+ if (kind == DestructuringKind::DestructureToExpressions && !innerPattern)
+ return 0;
+ failIfFalse(innerPattern, "Cannot parse this destructuring pattern");
+ context.appendArrayPatternRestEntry(arrayPattern, location, innerPattern);
+ restElementWasFound = true;
+ break;
+ }
+
JSTokenLocation location = m_token.m_location;
- auto innerPattern = parseDeconstructionPattern(context, kind, depth + 1);
- if (kind == DeconstructToExpressions && !innerPattern)
+ auto innerPattern = parseBindingOrAssignmentElement(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth + 1);
+ if (kind == DestructuringKind::DestructureToExpressions && !innerPattern)
return 0;
- failIfFalse(innerPattern, "Cannot parse this deconstruction pattern");
- context.appendArrayPatternEntry(arrayPattern, location, innerPattern);
+ failIfFalse(innerPattern, "Cannot parse this destructuring pattern");
+ TreeExpression defaultValue = parseDefaultValueForDestructuringPattern(context);
+ propagateError();
+ context.appendArrayPatternEntry(arrayPattern, location, innerPattern, defaultValue);
} while (consume(COMMA));
-
- if (kind == DeconstructToExpressions && !match(CLOSEBRACKET))
- return 0;
- consumeOrFail(CLOSEBRACKET, "Expected either a closing ']' or a ',' following an element deconstruction pattern");
+ consumeOrFail(CLOSEBRACKET, restElementWasFound ? "Expected a closing ']' following a rest element destructuring pattern" : "Expected either a closing ']' or a ',' following an element destructuring pattern");
+ context.finishArrayPattern(arrayPattern, divotStart, divotStart, lastTokenEndPosition());
pattern = arrayPattern;
break;
}
case OPENBRACE: {
+ auto objectPattern = context.createObjectPattern(m_token.m_location);
next();
-
- if (kind == DeconstructToExpressions && match(CLOSEBRACE))
- return 0;
- failIfTrue(match(CLOSEBRACE), "There must be at least one bound property in an object deconstruction pattern");
- auto objectPattern = context.createObjectPattern(m_token.m_location);
- bool wasString = false;
+ if (hasDestructuringPattern)
+ *hasDestructuringPattern = true;
+
do {
- Identifier propertyName;
- TreeDeconstructionPattern innerPattern = 0;
+ bool wasString = false;
+
+ if (match(CLOSEBRACE))
+ break;
+
+ const Identifier* propertyName = nullptr;
+ TreeExpression propertyExpression = 0;
+ TreeDestructuringPattern innerPattern = 0;
JSTokenLocation location = m_token.m_location;
- if (match(IDENT)) {
- propertyName = *m_token.m_data.ident;
+ if (matchSpecIdentifier()) {
+ failIfTrue(match(LET) && (kind == DestructuringKind::DestructureToLet || kind == DestructuringKind::DestructureToConst), "Can't use 'let' as an identifier name for a LexicalDeclaration");
+ propertyName = m_token.m_data.ident;
+ JSToken identifierToken = m_token;
next();
if (consume(COLON))
- innerPattern = parseDeconstructionPattern(context, kind, depth + 1);
- else
- innerPattern = createBindingPattern(context, kind, propertyName, depth);
+ innerPattern = parseBindingOrAssignmentElement(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth + 1);
+ else {
+ if (kind == DestructuringKind::DestructureToExpressions) {
+ bool isEvalOrArguments = m_vm->propertyNames->eval == *propertyName || m_vm->propertyNames->arguments == *propertyName;
+ if (isEvalOrArguments && strictMode())
+ reclassifyExpressionError(ErrorIndicatesPattern, ErrorIndicatesNothing);
+ failIfTrueIfStrict(isEvalOrArguments, "Cannot modify '", propertyName->impl(), "' in strict mode");
+ }
+ semanticFailIfTrue(isDisallowedIdentifierAwait(identifierToken), "Can't use 'await' as a ", destructuringKindToVariableKindName(kind), " ", disallowedIdentifierAwaitReason());
+ innerPattern = createBindingPattern(context, kind, exportType, *propertyName, identifierToken, bindingContext, duplicateIdentifier);
+ }
} else {
JSTokenType tokenType = m_token.m_type;
switch (m_token.m_type) {
- case NUMBER:
- propertyName = Identifier::from(m_vm, m_token.m_data.doubleValue);
+ case DOUBLE:
+ case INTEGER:
+ propertyName = &m_parserArena.identifierArena().makeNumericIdentifier(const_cast<VM*>(m_vm), m_token.m_data.doubleValue);
break;
case STRING:
- propertyName = *m_token.m_data.ident;
+ propertyName = m_token.m_data.ident;
wasString = true;
break;
+ case OPENBRACKET:
+ next();
+ propertyExpression = parseAssignmentExpression(context);
+ failIfFalse(propertyExpression, "Cannot parse computed property name");
+ matchOrFail(CLOSEBRACKET, "Expected ']' to end end a computed property name");
+ break;
default:
if (m_token.m_type != RESERVED && m_token.m_type != RESERVED_IF_STRICT && !(m_token.m_type & KeywordTokenFlag)) {
- if (kind == DeconstructToExpressions)
+ if (kind == DestructuringKind::DestructureToExpressions)
return 0;
failWithMessage("Expected a property name");
}
- propertyName = *m_token.m_data.ident;
+ propertyName = m_token.m_data.ident;
break;
}
next();
if (!consume(COLON)) {
- if (kind == DeconstructToExpressions)
+ if (kind == DestructuringKind::DestructureToExpressions)
return 0;
- semanticFailIfTrue(tokenType == RESERVED, "Cannot use abbreviated deconstruction syntax for reserved name '", propertyName.impl(), "'");
- semanticFailIfTrue(tokenType == RESERVED_IF_STRICT, "Cannot use abbreviated deconstruction syntax for reserved name '", propertyName.impl(), "' in strict mode");
- semanticFailIfTrue(tokenType & KeywordTokenFlag, "Cannot use abbreviated deconstruction syntax for keyword '", propertyName.impl(), "'");
+ semanticFailIfTrue(tokenType == RESERVED, "Cannot use abbreviated destructuring syntax for reserved name '", propertyName->impl(), "'");
+ semanticFailIfTrue(tokenType == RESERVED_IF_STRICT, "Cannot use abbreviated destructuring syntax for reserved name '", propertyName->impl(), "' in strict mode");
+ semanticFailIfTrue(tokenType & KeywordTokenFlag, "Cannot use abbreviated destructuring syntax for keyword '", propertyName->impl(), "'");
- failWithMessage("Expected a ':' prior to named property deconstruction");
+ failWithMessage("Expected a ':' prior to a named destructuring property");
}
- innerPattern = parseDeconstructionPattern(context, kind, depth + 1);
+ innerPattern = parseBindingOrAssignmentElement(context, kind, exportType, duplicateIdentifier, hasDestructuringPattern, bindingContext, depth + 1);
}
- if (kind == DeconstructToExpressions && !innerPattern)
+ if (kind == DestructuringKind::DestructureToExpressions && !innerPattern)
return 0;
- failIfFalse(innerPattern, "Cannot parse this deconstruction pattern");
- context.appendObjectPatternEntry(objectPattern, location, wasString, propertyName, innerPattern);
+ failIfFalse(innerPattern, "Cannot parse this destructuring pattern");
+ TreeExpression defaultValue = parseDefaultValueForDestructuringPattern(context);
+ propagateError();
+ if (propertyExpression)
+ context.appendObjectPatternEntry(objectPattern, location, propertyExpression, innerPattern, defaultValue);
+ else {
+ ASSERT(propertyName);
+ context.appendObjectPatternEntry(objectPattern, location, wasString, *propertyName, innerPattern, defaultValue);
+ }
} while (consume(COMMA));
- if (kind == DeconstructToExpressions && !match(CLOSEBRACE))
+
+ if (kind == DestructuringKind::DestructureToExpressions && !match(CLOSEBRACE))
return 0;
- consumeOrFail(CLOSEBRACE, "Expected either a closing '}' or an ',' after a property deconstruction pattern");
+ consumeOrFail(CLOSEBRACE, "Expected either a closing '}' or an ',' after a property destructuring pattern");
pattern = objectPattern;
break;
}
default: {
- if (!match(IDENT)) {
- if (kind == DeconstructToExpressions)
+ if (!matchSpecIdentifier()) {
+ if (kind == DestructuringKind::DestructureToExpressions)
return 0;
- semanticFailureDueToKeyword("variable name");
+ semanticFailureDueToKeyword(destructuringKindToVariableKindName(kind));
failWithMessage("Expected a parameter pattern or a ')' in parameter list");
}
- pattern = createBindingPattern(context, kind, *m_token.m_data.ident, depth);
+ failIfTrue(match(LET) && (kind == DestructuringKind::DestructureToLet || kind == DestructuringKind::DestructureToConst), "Can't use 'let' as an identifier name for a LexicalDeclaration");
+ semanticFailIfTrue(isDisallowedIdentifierAwait(m_token), "Can't use 'await' as a ", destructuringKindToVariableKindName(kind), " ", disallowedIdentifierAwaitReason());
+ pattern = createBindingPattern(context, kind, exportType, *m_token.m_data.ident, m_token, bindingContext, duplicateIdentifier);
next();
break;
}
}
- m_nonLHSCount = nonLHSCount;
+ m_parserState.nonLHSCount = nonLHSCount;
return pattern;
}
template <typename LexerType>
-template <class TreeBuilder> TreeConstDeclList Parser<LexerType>::parseConstDeclarationList(TreeBuilder& context)
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseDefaultValueForDestructuringPattern(TreeBuilder& context)
{
- failIfTrue(strictMode(), "Const declarations are not supported in strict mode");
- TreeConstDeclList constDecls = 0;
- TreeConstDeclList tail = 0;
- do {
- JSTokenLocation location(tokenLocation());
- next();
- matchOrFail(IDENT, "Expected an identifier name in const declaration");
- const Identifier* name = m_token.m_data.ident;
- next();
- bool hasInitializer = match(EQUAL);
- declareVariable(name);
- context.addVar(name, DeclarationStacks::IsConstant | (hasInitializer ? DeclarationStacks::HasInitializer : 0));
-
- TreeExpression initializer = 0;
- if (hasInitializer) {
- next(TreeBuilder::DontBuildStrings); // consume '='
- initializer = parseAssignmentExpression(context);
- failIfFalse(!!initializer, "Unable to parse initializer");
- }
- tail = context.appendConstDecl(location, tail, name, initializer);
- if (!constDecls)
- constDecls = tail;
- } while (match(COMMA));
- return constDecls;
+ if (!match(EQUAL))
+ return 0;
+
+ next(TreeBuilder::DontBuildStrings); // consume '='
+ return parseAssignmentExpression(context);
}
template <typename LexerType>
@@ -677,33 +1122,71 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseForStatement(
int startLine = tokenLine();
next();
handleProductionOrFail(OPENPAREN, "(", "start", "for-loop header");
- int nonLHSCount = m_nonLHSCount;
+ int nonLHSCount = m_parserState.nonLHSCount;
int declarations = 0;
+ JSTokenLocation declLocation(tokenLocation());
JSTextPosition declsStart;
JSTextPosition declsEnd;
TreeExpression decls = 0;
- TreeDeconstructionPattern pattern = 0;
- if (match(VAR)) {
+ TreeDestructuringPattern pattern = 0;
+ bool isVarDeclaraton = match(VAR);
+ bool isLetDeclaration = match(LET);
+ bool isConstDeclaration = match(CONSTTOKEN);
+ bool forLoopConstDoesNotHaveInitializer = false;
+
+ VariableEnvironment dummySet;
+ VariableEnvironment* lexicalVariables = nullptr;
+ AutoCleanupLexicalScope lexicalScope;
+
+ auto gatherLexicalVariablesIfNecessary = [&] {
+ if (isLetDeclaration || isConstDeclaration) {
+ ScopeRef scope = lexicalScope.scope();
+ lexicalVariables = &scope->finalizeLexicalEnvironment();
+ } else
+ lexicalVariables = &dummySet;
+ };
+
+ auto popLexicalScopeIfNecessary = [&] {
+ if (isLetDeclaration || isConstDeclaration)
+ popScope(lexicalScope, TreeBuilder::NeedsFreeVariableInfo);
+ };
+
+ if (isVarDeclaraton || isLetDeclaration || isConstDeclaration) {
/*
- for (var IDENT in expression) statement
- for (var varDeclarationList; expressionOpt; expressionOpt)
+ for (var/let/const IDENT in/of expression) statement
+ for (var/let/const varDeclarationList; expressionOpt; expressionOpt)
*/
- TreeDeconstructionPattern forInTarget = 0;
+ if (isLetDeclaration || isConstDeclaration) {
+ ScopeRef newScope = pushScope();
+ newScope->setIsLexicalScope();
+ newScope->preventVarDeclarations();
+ lexicalScope.setIsValid(newScope, this);
+ }
+
+ TreeDestructuringPattern forInTarget = 0;
TreeExpression forInInitializer = 0;
m_allowsIn = false;
JSTextPosition initStart;
JSTextPosition initEnd;
- decls = parseVarDeclarationList(context, declarations, forInTarget, forInInitializer, declsStart, initStart, initEnd);
+ DeclarationType declarationType;
+ if (isVarDeclaraton)
+ declarationType = DeclarationType::VarDeclaration;
+ else if (isLetDeclaration)
+ declarationType = DeclarationType::LetDeclaration;
+ else if (isConstDeclaration)
+ declarationType = DeclarationType::ConstDeclaration;
+ else
+ RELEASE_ASSERT_NOT_REACHED();
+ decls = parseVariableDeclarationList(context, declarations, forInTarget, forInInitializer, declsStart, initStart, initEnd, ForLoopContext, declarationType, ExportType::NotExported, forLoopConstDoesNotHaveInitializer);
m_allowsIn = true;
propagateError();
// Remainder of a standard for loop is handled identically
if (match(SEMICOLON))
goto standardForLoop;
-
- failIfFalse(declarations == 1, "must declare variables after 'var'");
- failIfTrue(forInInitializer, "Cannot use initialiser syntax in a for-in loop");
-
+
+ failIfFalse(declarations == 1, "can only declare a single variable in an enumeration");
+
// Handle for-in with var declaration
JSTextPosition inLocation = tokenStartPosition();
bool isOfEnumeration = false;
@@ -712,9 +1195,16 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseForStatement(
isOfEnumeration = true;
next();
}
-
+ bool hasAnyAssignments = !!forInInitializer;
+ if (hasAnyAssignments) {
+ if (isOfEnumeration)
+ internalFailWithMessage(false, "Cannot assign to the loop variable inside a for-of loop header");
+ if (strictMode() || (isLetDeclaration || isConstDeclaration) || !context.isBindingNode(forInTarget))
+ internalFailWithMessage(false, "Cannot assign to the loop variable inside a for-in loop header");
+ }
TreeExpression expr = parseExpression(context);
failIfFalse(expr, "Expected expression to enumerate");
+ recordPauseLocation(context.breakpointLocation(expr));
JSTextPosition exprEnd = lastTokenEndPosition();
int endLine = tokenLine();
@@ -726,20 +1216,29 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseForStatement(
TreeStatement statement = parseStatement(context, unused);
endLoop();
failIfFalse(statement, "Expected statement as body of for-", isOfEnumeration ? "of" : "in", " statement");
+ gatherLexicalVariablesIfNecessary();
+ TreeStatement result;
if (isOfEnumeration)
- return context.createForOfLoop(location, forInTarget, expr, statement, declsStart, inLocation, exprEnd, startLine, endLine);
- return context.createForInLoop(location, forInTarget, expr, statement, declsStart, inLocation, exprEnd, startLine, endLine);
+ result = context.createForOfLoop(location, forInTarget, expr, statement, declLocation, declsStart, inLocation, exprEnd, startLine, endLine, *lexicalVariables);
+ else {
+ if (isVarDeclaraton && forInInitializer)
+ result = context.createForInLoop(location, decls, expr, statement, declLocation, declsStart, inLocation, exprEnd, startLine, endLine, *lexicalVariables);
+ else
+ result = context.createForInLoop(location, forInTarget, expr, statement, declLocation, declsStart, inLocation, exprEnd, startLine, endLine, *lexicalVariables);
+ }
+ popLexicalScopeIfNecessary();
+ return result;
}
if (!match(SEMICOLON)) {
if (match(OPENBRACE) || match(OPENBRACKET)) {
SavePoint savePoint = createSavePoint();
declsStart = tokenStartPosition();
- pattern = tryParseDeconstructionPatternExpression(context);
+ pattern = tryParseDestructuringPatternExpression(context, AssignmentContext::DeclarationStatement);
declsEnd = lastTokenEndPosition();
if (pattern && (match(INTOKEN) || (match(IDENT) && *m_token.m_data.ident == m_vm->propertyNames->of)))
goto enumerationLoop;
- pattern = 0;
+ pattern = TreeDestructuringPattern(0);
restoreSavePoint(savePoint);
}
m_allowsIn = false;
@@ -748,17 +1247,22 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseForStatement(
declsEnd = lastTokenEndPosition();
m_allowsIn = true;
failIfFalse(decls, "Cannot parse for loop declarations");
+ recordPauseLocation(context.breakpointLocation(decls));
}
if (match(SEMICOLON)) {
standardForLoop:
// Standard for loop
+ if (decls)
+ recordPauseLocation(context.breakpointLocation(decls));
next();
TreeExpression condition = 0;
+ failIfTrue(forLoopConstDoesNotHaveInitializer && isConstDeclaration, "const variables in for loops must have initializers");
if (!match(SEMICOLON)) {
condition = parseExpression(context);
failIfFalse(condition, "Cannot parse for loop condition expression");
+ recordPauseLocation(context.breakpointLocation(condition));
}
consumeOrFail(SEMICOLON, "Expected a ';' after the for loop condition expression");
@@ -766,6 +1270,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseForStatement(
if (!match(CLOSEPAREN)) {
increment = parseExpression(context);
failIfFalse(increment, "Cannot parse for loop iteration expression");
+ recordPauseLocation(context.breakpointLocation(increment));
}
int endLine = tokenLine();
handleProductionOrFail(CLOSEPAREN, ")", "end", "for-loop header");
@@ -774,12 +1279,15 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseForStatement(
TreeStatement statement = parseStatement(context, unused);
endLoop();
failIfFalse(statement, "Expected a statement as the body of a for loop");
- return context.createForLoop(location, decls, condition, increment, statement, startLine, endLine);
+ gatherLexicalVariablesIfNecessary();
+ TreeStatement result = context.createForLoop(location, decls, condition, increment, statement, startLine, endLine, *lexicalVariables);
+ popLexicalScopeIfNecessary();
+ return result;
}
- // For-in loop
+ // For-in and For-of loop
enumerationLoop:
- failIfFalse(nonLHSCount == m_nonLHSCount, "Expected a reference on the left hand side of an enumeration statement");
+ failIfFalse(nonLHSCount == m_parserState.nonLHSCount, "Expected a reference on the left hand side of an enumeration statement");
bool isOfEnumeration = false;
if (!consume(INTOKEN)) {
failIfFalse(match(IDENT) && *m_token.m_data.ident == m_vm->propertyNames->of, "Expected either 'in' or 'of' in enumeration syntax");
@@ -788,6 +1296,7 @@ enumerationLoop:
}
TreeExpression expr = parseExpression(context);
failIfFalse(expr, "Cannot parse subject for-", isOfEnumeration ? "of" : "in", " statement");
+ recordPauseLocation(context.breakpointLocation(expr));
JSTextPosition exprEnd = lastTokenEndPosition();
int endLine = tokenLine();
@@ -797,15 +1306,24 @@ enumerationLoop:
TreeStatement statement = parseStatement(context, unused);
endLoop();
failIfFalse(statement, "Expected a statement as the body of a for-", isOfEnumeration ? "of" : "in", "loop");
+ gatherLexicalVariablesIfNecessary();
+ TreeStatement result;
if (pattern) {
ASSERT(!decls);
if (isOfEnumeration)
- return context.createForOfLoop(location, pattern, expr, statement, declsStart, declsEnd, exprEnd, startLine, endLine);
- return context.createForInLoop(location, pattern, expr, statement, declsStart, declsEnd, exprEnd, startLine, endLine);
+ result = context.createForOfLoop(location, pattern, expr, statement, declLocation, declsStart, declsEnd, exprEnd, startLine, endLine, *lexicalVariables);
+ else
+ result = context.createForInLoop(location, pattern, expr, statement, declLocation, declsStart, declsEnd, exprEnd, startLine, endLine, *lexicalVariables);
+
+ popLexicalScopeIfNecessary();
+ return result;
}
if (isOfEnumeration)
- return context.createForOfLoop(location, decls, expr, statement, declsStart, declsEnd, exprEnd, startLine, endLine);
- return context.createForInLoop(location, decls, expr, statement, declsStart, declsEnd, exprEnd, startLine, endLine);
+ result = context.createForOfLoop(location, decls, expr, statement, declLocation, declsStart, declsEnd, exprEnd, startLine, endLine, *lexicalVariables);
+ else
+ result = context.createForInLoop(location, decls, expr, statement, declLocation, declsStart, declsEnd, exprEnd, startLine, endLine, *lexicalVariables);
+ popLexicalScopeIfNecessary();
+ return result;
}
template <typename LexerType>
@@ -819,9 +1337,9 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseBreakStatemen
if (autoSemiColon()) {
semanticFailIfFalse(breakIsValid(), "'break' is only valid inside a switch or loop statement");
- return context.createBreakStatement(location, start, end);
+ return context.createBreakStatement(location, &m_vm->propertyNames->nullIdentifier, start, end);
}
- matchOrFail(IDENT, "Expected an identifier as the target for a break statement");
+ failIfFalse(matchSpecIdentifier(), "Expected an identifier as the target for a break statement");
const Identifier* ident = m_token.m_data.ident;
semanticFailIfFalse(getLabel(ident), "Cannot use the undeclared label '", ident->impl(), "'");
end = tokenEndPosition();
@@ -841,13 +1359,13 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseContinueState
if (autoSemiColon()) {
semanticFailIfFalse(continueIsValid(), "'continue' is only valid inside a loop statement");
- return context.createContinueStatement(location, start, end);
+ return context.createContinueStatement(location, &m_vm->propertyNames->nullIdentifier, start, end);
}
- matchOrFail(IDENT, "Expected an identifier as the target for a continue statement");
+ failIfFalse(matchSpecIdentifier(), "Expected an identifier as the target for a continue statement");
const Identifier* ident = m_token.m_data.ident;
ScopeLabelInfo* label = getLabel(ident);
semanticFailIfFalse(label, "Cannot use the undeclared label '", ident->impl(), "'");
- semanticFailIfFalse(label->m_isLoop, "Cannot continue to the label '", ident->impl(), "' as it is not targeting a loop");
+ semanticFailIfFalse(label->isLoop, "Cannot continue to the label '", ident->impl(), "' as it is not targeting a loop");
end = tokenEndPosition();
next();
failIfFalse(autoSemiColon(), "Expected a ';' following a targeted continue statement");
@@ -913,6 +1431,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseWithStatement
int start = tokenStart();
TreeExpression expr = parseExpression(context);
failIfFalse(expr, "Cannot parse 'with' subject expression");
+ recordPauseLocation(context.breakpointLocation(expr));
JSTextPosition end = lastTokenEndPosition();
int endLine = tokenLine();
handleProductionOrFail(CLOSEPAREN, ")", "start", "subject of a 'with' statement");
@@ -933,10 +1452,14 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseSwitchStateme
handleProductionOrFail(OPENPAREN, "(", "start", "subject of a 'switch'");
TreeExpression expr = parseExpression(context);
failIfFalse(expr, "Cannot parse switch subject expression");
+ recordPauseLocation(context.breakpointLocation(expr));
int endLine = tokenLine();
handleProductionOrFail(CLOSEPAREN, ")", "end", "subject of a 'switch'");
handleProductionOrFail(OPENBRACE, "{", "start", "body of a 'switch'");
+ AutoPopScopeRef lexicalScope(this, pushScope());
+ lexicalScope->setIsLexicalScope();
+ lexicalScope->preventVarDeclarations();
startSwitch();
TreeClauseList firstClauses = parseSwitchClauses(context);
propagateError();
@@ -949,8 +1472,9 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseSwitchStateme
endSwitch();
handleProductionOrFail(CLOSEBRACE, "}", "end", "body of a 'switch'");
- return context.createSwitchStatement(location, expr, firstClauses, defaultClause, secondClauses, startLine, endLine);
-
+ TreeStatement result = context.createSwitchStatement(location, expr, firstClauses, defaultClause, secondClauses, startLine, endLine, lexicalScope->finalizeLexicalEnvironment(), lexicalScope->takeFunctionDeclarations());
+ popScope(lexicalScope, TreeBuilder::NeedsFreeVariableInfo);
+ return result;
}
template <typename LexerType>
@@ -958,6 +1482,7 @@ template <class TreeBuilder> TreeClauseList Parser<LexerType>::parseSwitchClause
{
if (!match(CASE))
return 0;
+ unsigned startOffset = tokenStart();
next();
TreeExpression condition = parseExpression(context);
failIfFalse(condition, "Cannot parse switch clause");
@@ -965,10 +1490,12 @@ template <class TreeBuilder> TreeClauseList Parser<LexerType>::parseSwitchClause
TreeSourceElements statements = parseSourceElements(context, DontCheckForStrictMode);
failIfFalse(statements, "Cannot parse the body of a switch clause");
TreeClause clause = context.createClause(condition, statements);
+ context.setStartOffset(clause, startOffset);
TreeClauseList clauseList = context.createClauseList(clause);
TreeClauseList tail = clauseList;
while (match(CASE)) {
+ startOffset = tokenStart();
next();
TreeExpression condition = parseExpression(context);
failIfFalse(condition, "Cannot parse switch case expression");
@@ -976,6 +1503,7 @@ template <class TreeBuilder> TreeClauseList Parser<LexerType>::parseSwitchClause
TreeSourceElements statements = parseSourceElements(context, DontCheckForStrictMode);
failIfFalse(statements, "Cannot parse the body of a switch clause");
clause = context.createClause(condition, statements);
+ context.setStartOffset(clause, startOffset);
tail = context.createClauseList(tail, clause);
}
return clauseList;
@@ -986,11 +1514,14 @@ template <class TreeBuilder> TreeClause Parser<LexerType>::parseSwitchDefaultCla
{
if (!match(DEFAULT))
return 0;
+ unsigned startOffset = tokenStart();
next();
consumeOrFail(COLON, "Expected a ':' after switch default clause");
TreeSourceElements statements = parseSourceElements(context, DontCheckForStrictMode);
failIfFalse(statements, "Cannot parse the body of a switch default clause");
- return context.createClause(0, statements);
+ TreeClause result = context.createClause(0, statements);
+ context.setStartOffset(result, startOffset);
+ return result;
}
template <typename LexerType>
@@ -999,7 +1530,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseTryStatement(
ASSERT(match(TRY));
JSTokenLocation location(tokenLocation());
TreeStatement tryBlock = 0;
- const Identifier* ident = &m_vm->propertyNames->nullIdentifier;
+ TreeDestructuringPattern catchPattern = 0;
TreeStatement catchBlock = 0;
TreeStatement finallyBlock = 0;
int firstLine = tokenLine();
@@ -1009,26 +1540,31 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseTryStatement(
tryBlock = parseBlockStatement(context);
failIfFalse(tryBlock, "Cannot parse the body of try block");
int lastLine = m_lastTokenEndPosition.line;
-
+ VariableEnvironment catchEnvironment;
if (match(CATCH)) {
- currentScope()->setNeedsFullActivation();
next();
handleProductionOrFail(OPENPAREN, "(", "start", "'catch' target");
- if (!match(IDENT)) {
- semanticFailureDueToKeyword("catch variable name");
- failWithMessage("Expected identifier name as catch target");
- }
- ident = m_token.m_data.ident;
- next();
AutoPopScopeRef catchScope(this, pushScope());
- failIfFalseIfStrict(declareVariable(ident), "Cannot declare a catch variable named '", ident->impl(), "' in strict mode");
- catchScope->preventNewDecls();
+ catchScope->setIsLexicalScope();
+ catchScope->preventVarDeclarations();
+ const Identifier* ident = nullptr;
+ if (matchSpecIdentifier()) {
+ ident = m_token.m_data.ident;
+ catchPattern = context.createBindingLocation(m_token.m_location, *ident, m_token.m_startPosition, m_token.m_endPosition, AssignmentContext::DeclarationStatement);
+ next();
+ failIfTrueIfStrict(catchScope->declareLexicalVariable(ident, false) & DeclarationResult::InvalidStrictMode, "Cannot declare a catch variable named '", ident->impl(), "' in strict mode");
+ } else {
+ catchPattern = parseDestructuringPattern(context, DestructuringKind::DestructureToCatchParameters, ExportType::NotExported);
+ failIfFalse(catchPattern, "Cannot parse this destructuring pattern");
+ }
handleProductionOrFail(CLOSEPAREN, ")", "end", "'catch' target");
matchOrFail(OPENBRACE, "Expected exception handler to be a block statement");
catchBlock = parseBlockStatement(context);
failIfFalse(catchBlock, "Unable to parse 'catch' block");
- failIfFalse(popScope(catchScope, TreeBuilder::NeedsFreeVariableInfo), "Parse error");
+ catchEnvironment = catchScope->finalizeLexicalEnvironment();
+ RELEASE_ASSERT(!ident || (catchEnvironment.size() == 1 && catchEnvironment.contains(ident->impl())));
+ popScope(catchScope, TreeBuilder::NeedsFreeVariableInfo);
}
if (match(FINALLY)) {
@@ -1038,7 +1574,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseTryStatement(
failIfFalse(finallyBlock, "Cannot parse finally body");
}
failIfFalse(catchBlock || finallyBlock, "Try statements must have at least a catch or finally block");
- return context.createTryStatement(location, tryBlock, ident, catchBlock, finallyBlock, firstLine, lastLine);
+ return context.createTryStatement(location, tryBlock, catchPattern, catchBlock, finallyBlock, firstLine, lastLine, catchEnvironment);
}
template <typename LexerType>
@@ -1059,18 +1595,45 @@ template <typename LexerType>
template <class TreeBuilder> TreeStatement Parser<LexerType>::parseBlockStatement(TreeBuilder& context)
{
ASSERT(match(OPENBRACE));
+
+ // We should treat the first block statement of the function (the body of the function) as the lexical
+ // scope of the function itself, and not the lexical scope of a 'block' statement within the function.
+ AutoCleanupLexicalScope lexicalScope;
+ bool shouldPushLexicalScope = m_statementDepth > 0;
+ if (shouldPushLexicalScope) {
+ ScopeRef newScope = pushScope();
+ newScope->setIsLexicalScope();
+ newScope->preventVarDeclarations();
+ lexicalScope.setIsValid(newScope, this);
+ }
JSTokenLocation location(tokenLocation());
+ int startOffset = m_token.m_data.offset;
int start = tokenLine();
+ VariableEnvironment emptyEnvironment;
+ DeclarationStacks::FunctionStack emptyFunctionStack;
next();
if (match(CLOSEBRACE)) {
+ int endOffset = m_token.m_data.offset;
next();
- return context.createBlockStatement(location, 0, start, m_lastTokenEndPosition.line);
+ TreeStatement result = context.createBlockStatement(location, 0, start, m_lastTokenEndPosition.line, shouldPushLexicalScope ? currentScope()->finalizeLexicalEnvironment() : emptyEnvironment, shouldPushLexicalScope ? currentScope()->takeFunctionDeclarations() : WTFMove(emptyFunctionStack));
+ context.setStartOffset(result, startOffset);
+ context.setEndOffset(result, endOffset);
+ if (shouldPushLexicalScope)
+ popScope(lexicalScope, TreeBuilder::NeedsFreeVariableInfo);
+ return result;
}
TreeSourceElements subtree = parseSourceElements(context, DontCheckForStrictMode);
failIfFalse(subtree, "Cannot parse the body of the block statement");
matchOrFail(CLOSEBRACE, "Expected a closing '}' at the end of a block statement");
+ int endOffset = m_token.m_data.offset;
next();
- return context.createBlockStatement(location, subtree, start, m_lastTokenEndPosition.line);
+ TreeStatement result = context.createBlockStatement(location, subtree, start, m_lastTokenEndPosition.line, shouldPushLexicalScope ? currentScope()->finalizeLexicalEnvironment() : emptyEnvironment, shouldPushLexicalScope ? currentScope()->takeFunctionDeclarations() : WTFMove(emptyFunctionStack));
+ context.setStartOffset(result, startOffset);
+ context.setEndOffset(result, endOffset);
+ if (shouldPushLexicalScope)
+ popScope(lexicalScope, TreeBuilder::NeedsFreeVariableInfo);
+
+ return result;
}
template <typename LexerType>
@@ -1078,245 +1641,1080 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseStatement(Tre
{
DepthManager statementDepth(&m_statementDepth);
m_statementDepth++;
- directive = 0;
int nonTrivialExpressionCount = 0;
failIfStackOverflow();
+ TreeStatement result = 0;
+ bool shouldSetEndOffset = true;
+ bool shouldSetPauseLocation = false;
+ bool parentAllowsFunctionDeclarationAsStatement = m_immediateParentAllowsFunctionDeclarationInStatement;
+ m_immediateParentAllowsFunctionDeclarationInStatement = false;
+
switch (m_token.m_type) {
case OPENBRACE:
- return parseBlockStatement(context);
+ result = parseBlockStatement(context);
+ shouldSetEndOffset = false;
+ break;
case VAR:
- return parseVarDeclaration(context);
- case CONSTTOKEN:
- return parseConstDeclaration(context);
- case FUNCTION:
- failIfFalseIfStrict(m_statementDepth == 1, "Nested functions cannot be declared in strict mode");
- return parseFunctionDeclaration(context);
+ result = parseVariableDeclaration(context, DeclarationType::VarDeclaration);
+ shouldSetPauseLocation = true;
+ break;
+ case FUNCTION: {
+ const bool isAsync = false;
+ result = parseFunctionDeclarationStatement(context, isAsync, parentAllowsFunctionDeclarationAsStatement);
+ break;
+ }
case SEMICOLON: {
JSTokenLocation location(tokenLocation());
next();
- return context.createEmptyStatement(location);
+ result = context.createEmptyStatement(location);
+ shouldSetPauseLocation = true;
+ break;
}
case IF:
- return parseIfStatement(context);
+ result = parseIfStatement(context);
+ break;
case DO:
- return parseDoWhileStatement(context);
+ result = parseDoWhileStatement(context);
+ break;
case WHILE:
- return parseWhileStatement(context);
+ result = parseWhileStatement(context);
+ break;
case FOR:
- return parseForStatement(context);
+ result = parseForStatement(context);
+ break;
case CONTINUE:
- return parseContinueStatement(context);
+ result = parseContinueStatement(context);
+ shouldSetPauseLocation = true;
+ break;
case BREAK:
- return parseBreakStatement(context);
+ result = parseBreakStatement(context);
+ shouldSetPauseLocation = true;
+ break;
case RETURN:
- return parseReturnStatement(context);
+ result = parseReturnStatement(context);
+ shouldSetPauseLocation = true;
+ break;
case WITH:
- return parseWithStatement(context);
+ result = parseWithStatement(context);
+ break;
case SWITCH:
- return parseSwitchStatement(context);
+ result = parseSwitchStatement(context);
+ break;
case THROW:
- return parseThrowStatement(context);
+ result = parseThrowStatement(context);
+ shouldSetPauseLocation = true;
+ break;
case TRY:
- return parseTryStatement(context);
+ result = parseTryStatement(context);
+ break;
case DEBUGGER:
- return parseDebuggerStatement(context);
+ result = parseDebuggerStatement(context);
+ shouldSetPauseLocation = true;
+ break;
case EOFTOK:
case CASE:
case CLOSEBRACE:
case DEFAULT:
// These tokens imply the end of a set of source elements
return 0;
+ case LET: {
+ if (!strictMode())
+ goto identcase;
+ goto defaultCase;
+ }
+ case ASYNC:
+ if (maybeParseAsyncFunctionDeclarationStatement(context, result, parentAllowsFunctionDeclarationAsStatement))
+ break;
+ FALLTHROUGH;
case IDENT:
- return parseExpressionOrLabelStatement(context);
+ case AWAIT:
+ case YIELD: {
+ identcase:
+ bool allowFunctionDeclarationAsStatement = false;
+ result = parseExpressionOrLabelStatement(context, allowFunctionDeclarationAsStatement);
+ shouldSetPauseLocation = !context.shouldSkipPauseLocation(result);
+ break;
+ }
case STRING:
directive = m_token.m_data.ident;
if (directiveLiteralLength)
*directiveLiteralLength = m_token.m_location.endOffset - m_token.m_location.startOffset;
- nonTrivialExpressionCount = m_nonTrivialExpressionCount;
+ nonTrivialExpressionCount = m_parserState.nonTrivialExpressionCount;
FALLTHROUGH;
default:
+ defaultCase:
TreeStatement exprStatement = parseExpressionStatement(context);
- if (directive && nonTrivialExpressionCount != m_nonTrivialExpressionCount)
- directive = 0;
- return exprStatement;
+ if (directive && nonTrivialExpressionCount != m_parserState.nonTrivialExpressionCount)
+ directive = nullptr;
+ result = exprStatement;
+ shouldSetPauseLocation = true;
+ break;
+ }
+
+ if (result) {
+ if (shouldSetEndOffset)
+ context.setEndOffset(result, m_lastTokenEndPosition.offset);
+ if (shouldSetPauseLocation)
+ recordPauseLocation(context.breakpointLocation(result));
}
+
+ return result;
}
template <typename LexerType>
-template <class TreeBuilder> TreeFormalParameterList Parser<LexerType>::parseFormalParameters(TreeBuilder& context)
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseFunctionDeclarationStatement(TreeBuilder& context, bool isAsync, bool parentAllowsFunctionDeclarationAsStatement)
{
- auto parameter = parseDeconstructionPattern(context, DeconstructToParameters);
- failIfFalse(parameter, "Cannot parse parameter pattern");
- TreeFormalParameterList list = context.createFormalParameterList(parameter);
- TreeFormalParameterList tail = list;
- while (consume(COMMA)) {
- parameter = parseDeconstructionPattern(context, DeconstructToParameters);
- failIfFalse(parameter, "Cannot parse parameter pattern");
- tail = context.createFormalParameterList(tail, parameter);
+ semanticFailIfTrue(strictMode(), "Function declarations are only allowed inside blocks or switch statements in strict mode");
+ failIfFalse(parentAllowsFunctionDeclarationAsStatement, "Function declarations are only allowed inside block statements or at the top level of a program");
+ if (!currentScope()->isFunction()) {
+ // We only implement annex B.3.3 if we're in function mode. Otherwise, we fall back
+ // to hoisting behavior.
+ // FIXME: https://bugs.webkit.org/show_bug.cgi?id=155813
+ DepthManager statementDepth(&m_statementDepth);
+ m_statementDepth = 1;
+ if (isAsync)
+ return parseAsyncFunctionDeclaration(context);
+ return parseFunctionDeclaration(context);
}
- return list;
+
+ // Any function declaration that isn't in a block is a syntax error unless it's
+ // in an if/else statement. If it's in an if/else statement, we will magically
+ // treat it as if the if/else statement is inside a block statement.
+ // to the very top like a "var". For example:
+ // function a() {
+ // if (cond) function foo() { }
+ // }
+ // will be rewritten as:
+ // function a() {
+ // if (cond) { function foo() { } }
+ // }
+ AutoPopScopeRef blockScope(this, pushScope());
+ blockScope->setIsLexicalScope();
+ blockScope->preventVarDeclarations();
+ JSTokenLocation location(tokenLocation());
+ int start = tokenLine();
+
+ TreeStatement function = 0;
+ if (!isAsync)
+ function = parseFunctionDeclaration(context);
+ else
+ function = parseAsyncFunctionDeclaration(context);
+ propagateError();
+ failIfFalse(function, "Expected valid function statement after 'function' keyword");
+ TreeSourceElements sourceElements = context.createSourceElements();
+ context.appendStatement(sourceElements, function);
+ TreeStatement result = context.createBlockStatement(location, sourceElements, start, m_lastTokenEndPosition.line, currentScope()->finalizeLexicalEnvironment(), currentScope()->takeFunctionDeclarations());
+ popScope(blockScope, TreeBuilder::NeedsFreeVariableInfo);
+ return result;
}
template <typename LexerType>
-template <class TreeBuilder> TreeFunctionBody Parser<LexerType>::parseFunctionBody(TreeBuilder& context)
+template <class TreeBuilder> bool Parser<LexerType>::maybeParseAsyncFunctionDeclarationStatement(TreeBuilder& context, TreeStatement& result, bool parentAllowsFunctionDeclarationAsStatement)
{
- JSTokenLocation startLocation(tokenLocation());
- unsigned startColumn = tokenColumn();
+ ASSERT(match(ASYNC));
+ SavePoint savePoint = createSavePoint();
next();
+ if (match(FUNCTION) && !m_lexer->prevTerminator()) {
+ const bool isAsync = true;
+ result = parseFunctionDeclarationStatement(context, isAsync, parentAllowsFunctionDeclarationAsStatement);
+ return true;
+ }
+ restoreSavePoint(savePoint);
+ return false;
+}
- if (match(CLOSEBRACE)) {
- unsigned endColumn = tokenColumn();
- return context.createFunctionBody(startLocation, tokenLocation(), startColumn, endColumn, strictMode());
+template <typename LexerType>
+template <class TreeBuilder> bool Parser<LexerType>::parseFormalParameters(TreeBuilder& context, TreeFormalParameterList list, bool isArrowFunction, unsigned& parameterCount)
+{
+#define failIfDuplicateIfViolation() \
+ if (duplicateParameter) {\
+ semanticFailIfTrue(hasDefaultParameterValues, "Duplicate parameter '", duplicateParameter->impl(), "' not allowed in function with default parameter values");\
+ semanticFailIfTrue(hasDestructuringPattern, "Duplicate parameter '", duplicateParameter->impl(), "' not allowed in function with destructuring parameters");\
+ semanticFailIfTrue(isRestParameter, "Duplicate parameter '", duplicateParameter->impl(), "' not allowed in function with a rest parameter");\
+ semanticFailIfTrue(isArrowFunction, "Duplicate parameter '", duplicateParameter->impl(), "' not allowed in an arrow function");\
+ }
+
+ bool hasDefaultParameterValues = false;
+ bool hasDestructuringPattern = false;
+ bool isRestParameter = false;
+ const Identifier* duplicateParameter = nullptr;
+ unsigned restParameterStart = 0;
+ do {
+ TreeDestructuringPattern parameter = 0;
+ TreeExpression defaultValue = 0;
+
+ if (UNLIKELY(match(CLOSEPAREN)))
+ break;
+
+ if (match(DOTDOTDOT)) {
+ next();
+ semanticFailIfTrue(!m_parserState.allowAwait && match(AWAIT), "Can't use 'await' as a parameter name in an async function");
+ TreeDestructuringPattern destructuringPattern = parseDestructuringPattern(context, DestructuringKind::DestructureToParameters, ExportType::NotExported, &duplicateParameter, &hasDestructuringPattern);
+ propagateError();
+ parameter = context.createRestParameter(destructuringPattern, restParameterStart);
+ failIfTrue(match(COMMA), "Rest parameter should be the last parameter in a function declaration"); // Let's have a good error message for this common case.
+ isRestParameter = true;
+ } else
+ parameter = parseDestructuringPattern(context, DestructuringKind::DestructureToParameters, ExportType::NotExported, &duplicateParameter, &hasDestructuringPattern);
+ failIfFalse(parameter, "Cannot parse parameter pattern");
+ if (!isRestParameter) {
+ defaultValue = parseDefaultValueForDestructuringPattern(context);
+ if (defaultValue)
+ hasDefaultParameterValues = true;
+ }
+ propagateError();
+ failIfDuplicateIfViolation();
+ if (isRestParameter || defaultValue || hasDestructuringPattern)
+ currentScope()->setHasNonSimpleParameterList();
+ context.appendParameter(list, parameter, defaultValue);
+ if (!isRestParameter) {
+ restParameterStart++;
+ if (!hasDefaultParameterValues)
+ parameterCount++;
+ }
+ } while (!isRestParameter && consume(COMMA));
+
+ return true;
+#undef failIfDuplicateIfViolation
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeFunctionBody Parser<LexerType>::parseFunctionBody(
+ TreeBuilder& context, SyntaxChecker& syntaxChecker, const JSTokenLocation& startLocation, int startColumn, int functionKeywordStart, int functionNameStart, int parametersStart,
+ ConstructorKind constructorKind, SuperBinding superBinding, FunctionBodyType bodyType, unsigned parameterCount, SourceParseMode parseMode)
+{
+ bool isArrowFunctionBodyExpression = bodyType == ArrowFunctionBodyExpression;
+ if (!isArrowFunctionBodyExpression) {
+ next();
+ if (match(CLOSEBRACE)) {
+ unsigned endColumn = tokenColumn();
+ SuperBinding functionSuperBinding = adjustSuperBindingForBaseConstructor(constructorKind, superBinding, currentScope());
+ return context.createFunctionMetadata(startLocation, tokenLocation(), startColumn, endColumn, functionKeywordStart, functionNameStart, parametersStart, strictMode(), constructorKind, functionSuperBinding, parameterCount, parseMode, isArrowFunctionBodyExpression);
+ }
}
+
DepthManager statementDepth(&m_statementDepth);
m_statementDepth = 0;
- typename TreeBuilder::FunctionBodyBuilder bodyBuilder(const_cast<VM*>(m_vm), m_lexer.get());
- failIfFalse(parseSourceElements(bodyBuilder, CheckForStrictMode), "Cannot parse body of this function");
+ if (bodyType == ArrowFunctionBodyExpression) {
+ if (m_debuggerParseData)
+ failIfFalse(parseArrowFunctionSingleExpressionBodySourceElements(context), "Cannot parse body of this arrow function");
+ else
+ failIfFalse(parseArrowFunctionSingleExpressionBodySourceElements(syntaxChecker), "Cannot parse body of this arrow function");
+ } else {
+ if (m_debuggerParseData)
+ failIfFalse(parseSourceElements(context, CheckForStrictMode), bodyType == StandardFunctionBodyBlock ? "Cannot parse body of this function" : "Cannot parse body of this arrow function");
+ else
+ failIfFalse(parseSourceElements(syntaxChecker, CheckForStrictMode), bodyType == StandardFunctionBodyBlock ? "Cannot parse body of this function" : "Cannot parse body of this arrow function");
+ }
unsigned endColumn = tokenColumn();
- return context.createFunctionBody(startLocation, tokenLocation(), startColumn, endColumn, strictMode());
+ SuperBinding functionSuperBinding = adjustSuperBindingForBaseConstructor(constructorKind, superBinding, currentScope());
+ return context.createFunctionMetadata(startLocation, tokenLocation(), startColumn, endColumn, functionKeywordStart, functionNameStart, parametersStart, strictMode(), constructorKind, functionSuperBinding, parameterCount, parseMode, isArrowFunctionBodyExpression);
}
-static const char* stringForFunctionMode(FunctionParseMode mode)
+static const char* stringForFunctionMode(SourceParseMode mode)
{
switch (mode) {
- case GetterMode:
+ case SourceParseMode::GetterMode:
return "getter";
- case SetterMode:
+ case SourceParseMode::SetterMode:
return "setter";
- case FunctionMode:
+ case SourceParseMode::NormalFunctionMode:
return "function";
+ case SourceParseMode::MethodMode:
+ return "method";
+ case SourceParseMode::GeneratorBodyMode:
+ return "generator";
+ case SourceParseMode::GeneratorWrapperFunctionMode:
+ return "generator function";
+ case SourceParseMode::ArrowFunctionMode:
+ return "arrow function";
+ case SourceParseMode::AsyncFunctionMode:
+ case SourceParseMode::AsyncFunctionBodyMode:
+ return "async function";
+ case SourceParseMode::AsyncMethodMode:
+ return "async method";
+ case SourceParseMode::AsyncArrowFunctionBodyMode:
+ case SourceParseMode::AsyncArrowFunctionMode:
+ return "async arrow function";
+ case SourceParseMode::ProgramMode:
+ case SourceParseMode::ModuleAnalyzeMode:
+ case SourceParseMode::ModuleEvaluateMode:
+ RELEASE_ASSERT_NOT_REACHED();
+ return "";
}
RELEASE_ASSERT_NOT_REACHED();
return nullptr;
}
-template <typename LexerType>
-template <class TreeBuilder> bool Parser<LexerType>::parseFunctionInfo(TreeBuilder& context, FunctionRequirements requirements, FunctionParseMode mode, bool nameIsInContainingScope, const Identifier*& name, TreeFormalParameterList& parameters, TreeFunctionBody& body, unsigned& openBraceOffset, unsigned& closeBraceOffset, int& bodyStartLine, unsigned& bodyStartColumn)
+template <typename LexerType> template <class TreeBuilder, class FunctionInfoType> typename TreeBuilder::FormalParameterList Parser<LexerType>::parseFunctionParameters(TreeBuilder& context, SourceParseMode mode, FunctionInfoType& functionInfo)
{
- AutoPopScopeRef functionScope(this, pushScope());
- functionScope->setIsFunction();
- int functionNameStart = m_token.m_location.startOffset;
- const Identifier* lastFunctionName = m_lastFunctionName;
- m_lastFunctionName = nullptr;
- if (match(IDENT)) {
- name = m_token.m_data.ident;
- m_lastFunctionName = name;
- next();
- if (!nameIsInContainingScope)
- failIfFalseIfStrict(functionScope->declareVariable(name), "'", name->impl(), "' is not a valid ", stringForFunctionMode(mode), " name in strict mode");
- } else if (requirements == FunctionNeedsName) {
- if (match(OPENPAREN) && mode == FunctionMode)
- semanticFail("Function statements must have a name");
- semanticFailureDueToKeyword(stringForFunctionMode(mode), " name");
- failDueToUnexpectedToken();
- return false;
+ RELEASE_ASSERT(!(SourceParseModeSet(SourceParseMode::ProgramMode, SourceParseMode::ModuleAnalyzeMode, SourceParseMode::ModuleEvaluateMode).contains(mode)));
+ TreeFormalParameterList parameterList = context.createFormalParameterList();
+ SetForScope<FunctionParsePhase> functionParsePhasePoisoner(m_parserState.functionParsePhase, FunctionParsePhase::Parameters);
+
+ if (UNLIKELY((SourceParseModeSet(SourceParseMode::ArrowFunctionMode, SourceParseMode::AsyncArrowFunctionMode).contains(mode)))) {
+ if (!matchSpecIdentifier() && !match(OPENPAREN)) {
+ semanticFailureDueToKeyword(stringForFunctionMode(mode), " name");
+ failWithMessage("Expected an arrow function input parameter");
+ } else {
+ if (match(OPENPAREN)) {
+ next();
+
+ if (match(CLOSEPAREN)) {
+ functionInfo.parameterCount = 0;
+ } else {
+ bool isArrowFunction = true;
+ failIfFalse(parseFormalParameters(context, parameterList, isArrowFunction, functionInfo.parameterCount), "Cannot parse parameters for this ", stringForFunctionMode(mode));
+ }
+
+ consumeOrFail(CLOSEPAREN, "Expected a ')' or a ',' after a parameter declaration");
+ } else {
+ functionInfo.parameterCount = 1;
+ auto parameter = parseDestructuringPattern(context, DestructuringKind::DestructureToParameters, ExportType::NotExported);
+ failIfFalse(parameter, "Cannot parse parameter pattern");
+ context.appendParameter(parameterList, parameter, 0);
+ }
+ }
+
+ return parameterList;
}
+
if (!consume(OPENPAREN)) {
semanticFailureDueToKeyword(stringForFunctionMode(mode), " name");
failWithMessage("Expected an opening '(' before a ", stringForFunctionMode(mode), "'s parameter list");
}
- if (!match(CLOSEPAREN)) {
- parameters = parseFormalParameters(context);
- failIfFalse(parameters, "Cannot parse parameters for this ", stringForFunctionMode(mode));
+
+ if (mode == SourceParseMode::GetterMode) {
+ consumeOrFail(CLOSEPAREN, "getter functions must have no parameters");
+ functionInfo.parameterCount = 0;
+ } else if (mode == SourceParseMode::SetterMode) {
+ failIfTrue(match(CLOSEPAREN), "setter functions must have one parameter");
+ const Identifier* duplicateParameter = nullptr;
+ bool hasDestructuringPattern = false;
+ auto parameter = parseDestructuringPattern(context, DestructuringKind::DestructureToParameters, ExportType::NotExported, &duplicateParameter, &hasDestructuringPattern);
+ failIfFalse(parameter, "setter functions must have one parameter");
+ auto defaultValue = parseDefaultValueForDestructuringPattern(context);
+ propagateError();
+ if (defaultValue || hasDestructuringPattern) {
+ semanticFailIfTrue(duplicateParameter, "Duplicate parameter '", duplicateParameter->impl(), "' not allowed in function with non-simple parameter list");
+ currentScope()->setHasNonSimpleParameterList();
+ }
+ context.appendParameter(parameterList, parameter, defaultValue);
+ functionInfo.parameterCount = defaultValue ? 0 : 1;
+ failIfTrue(match(COMMA), "setter functions must have one parameter");
+ consumeOrFail(CLOSEPAREN, "Expected a ')' after a parameter declaration");
+ } else {
+ if (match(CLOSEPAREN)) {
+ functionInfo.parameterCount = 0;
+ } else {
+ bool isArrowFunction = false;
+ failIfFalse(parseFormalParameters(context, parameterList, isArrowFunction, functionInfo.parameterCount), "Cannot parse parameters for this ", stringForFunctionMode(mode));
+ }
+ consumeOrFail(CLOSEPAREN, "Expected a ')' or a ',' after a parameter declaration");
}
- consumeOrFail(CLOSEPAREN, "Expected a ')' or a ',' after a parameter declaration");
- matchOrFail(OPENBRACE, "Expected an opening '{' at the start of a ", stringForFunctionMode(mode), " body");
-
- openBraceOffset = m_token.m_data.offset;
- bodyStartLine = tokenLine();
- bodyStartColumn = m_token.m_data.offset - m_token.m_data.lineStartOffset;
- JSTokenLocation startLocation(tokenLocation());
-
- // If we know about this function already, we can use the cached info and skip the parser to the end of the function.
- if (const SourceProviderCacheItem* cachedInfo = TreeBuilder::CanUseFunctionCache ? findCachedFunctionInfo(openBraceOffset) : 0) {
- // If we're in a strict context, the cached function info must say it was strict too.
- ASSERT(!strictMode() || cachedInfo->strictMode);
- JSTokenLocation endLocation;
-
- endLocation.line = cachedInfo->closeBraceLine;
- endLocation.startOffset = cachedInfo->closeBraceOffset;
- endLocation.lineStartOffset = cachedInfo->closeBraceLineStartOffset;
-
- bool endColumnIsOnStartLine = (endLocation.line == bodyStartLine);
- ASSERT(endLocation.startOffset >= endLocation.lineStartOffset);
- unsigned bodyEndColumn = endColumnIsOnStartLine ?
- endLocation.startOffset - m_token.m_data.lineStartOffset :
- endLocation.startOffset - endLocation.lineStartOffset;
-
- body = context.createFunctionBody(startLocation, endLocation, bodyStartColumn, bodyEndColumn, cachedInfo->strictMode);
+
+ return parameterList;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> typename TreeBuilder::FormalParameterList Parser<LexerType>::createGeneratorParameters(TreeBuilder& context, unsigned& parameterCount)
+{
+ auto parameters = context.createFormalParameterList();
+
+ JSTokenLocation location(tokenLocation());
+ JSTextPosition position = tokenStartPosition();
+
+ auto addParameter = [&](const Identifier& name) {
+ declareParameter(&name);
+ auto binding = context.createBindingLocation(location, name, position, position, AssignmentContext::DeclarationStatement);
+ context.appendParameter(parameters, binding, 0);
+ ++parameterCount;
+ };
+
+ // @generator
+ addParameter(m_vm->propertyNames->builtinNames().generatorPrivateName());
+ // @generatorState
+ addParameter(m_vm->propertyNames->builtinNames().generatorStatePrivateName());
+ // @generatorValue
+ addParameter(m_vm->propertyNames->builtinNames().generatorValuePrivateName());
+ // @generatorResumeMode
+ addParameter(m_vm->propertyNames->builtinNames().generatorResumeModePrivateName());
+ // @generatorFrame
+ addParameter(m_vm->propertyNames->builtinNames().generatorFramePrivateName());
+
+ return parameters;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> bool Parser<LexerType>::parseFunctionInfo(TreeBuilder& context, FunctionNameRequirements requirements, SourceParseMode mode, bool nameIsInContainingScope, ConstructorKind constructorKind, SuperBinding expectedSuperBinding, int functionKeywordStart, ParserFunctionInfo<TreeBuilder>& functionInfo, FunctionDefinitionType functionDefinitionType)
+{
+ RELEASE_ASSERT(isFunctionParseMode(mode));
+
+ ScopeRef parentScope = currentScope();
+
+ bool isDisallowedAwaitFunctionName = isDisallowedIdentifierAwait(m_token);
+ const char* isDisallowedAwaitFunctionNameReason = isDisallowedAwaitFunctionName ? disallowedIdentifierAwaitReason() : nullptr;
+
+ AutoPopScopeRef functionScope(this, pushScope());
+ functionScope->setSourceParseMode(mode);
+ functionScope->setExpectedSuperBinding(expectedSuperBinding);
+ functionScope->setConstructorKind(constructorKind);
+ SetForScope<FunctionParsePhase> functionParsePhasePoisoner(m_parserState.functionParsePhase, FunctionParsePhase::Body);
+ int functionNameStart = m_token.m_location.startOffset;
+ const Identifier* lastFunctionName = m_parserState.lastFunctionName;
+ m_parserState.lastFunctionName = nullptr;
+ int parametersStart = -1;
+ JSTokenLocation startLocation;
+ int startColumn = -1;
+ FunctionBodyType functionBodyType;
+
+ auto loadCachedFunction = [&] () -> bool {
+ if (UNLIKELY(!Options::useSourceProviderCache()))
+ return false;
+
+ if (UNLIKELY(m_debuggerParseData))
+ return false;
+
+ ASSERT(parametersStart != -1);
+ ASSERT(startColumn != -1);
+
+ // If we know about this function already, we can use the cached info and skip the parser to the end of the function.
+ if (const SourceProviderCacheItem* cachedInfo = TreeBuilder::CanUseFunctionCache ? findCachedFunctionInfo(parametersStart) : 0) {
+ // If we're in a strict context, the cached function info must say it was strict too.
+ ASSERT(!strictMode() || cachedInfo->strictMode);
+ JSTokenLocation endLocation;
+
+ ConstructorKind constructorKind = static_cast<ConstructorKind>(cachedInfo->constructorKind);
+ SuperBinding expectedSuperBinding = static_cast<SuperBinding>(cachedInfo->expectedSuperBinding);
+
+ endLocation.line = cachedInfo->lastTokenLine;
+ endLocation.startOffset = cachedInfo->lastTokenStartOffset;
+ endLocation.lineStartOffset = cachedInfo->lastTokenLineStartOffset;
+ ASSERT(endLocation.startOffset >= endLocation.lineStartOffset);
+
+ bool endColumnIsOnStartLine = endLocation.line == functionInfo.startLine;
+ unsigned currentLineStartOffset = m_lexer->currentLineStartOffset();
+ unsigned bodyEndColumn = endColumnIsOnStartLine ? endLocation.startOffset - currentLineStartOffset : endLocation.startOffset - endLocation.lineStartOffset;
+
+ ASSERT(endLocation.startOffset >= endLocation.lineStartOffset);
+
+ FunctionBodyType functionBodyType;
+ if (UNLIKELY(SourceParseModeSet(SourceParseMode::ArrowFunctionMode, SourceParseMode::AsyncArrowFunctionMode).contains(mode)))
+ functionBodyType = cachedInfo->isBodyArrowExpression ? ArrowFunctionBodyExpression : ArrowFunctionBodyBlock;
+ else
+ functionBodyType = StandardFunctionBodyBlock;
+
+ SuperBinding functionSuperBinding = adjustSuperBindingForBaseConstructor(constructorKind, expectedSuperBinding, cachedInfo->needsSuperBinding, cachedInfo->usesEval, cachedInfo->innerArrowFunctionFeatures);
+
+ functionInfo.body = context.createFunctionMetadata(
+ startLocation, endLocation, startColumn, bodyEndColumn,
+ functionKeywordStart, functionNameStart, parametersStart,
+ cachedInfo->strictMode, constructorKind, functionSuperBinding,
+ cachedInfo->parameterCount,
+ mode, functionBodyType == ArrowFunctionBodyExpression);
+ functionInfo.endOffset = cachedInfo->endFunctionOffset;
+ functionInfo.parameterCount = cachedInfo->parameterCount;
+
+ functionScope->restoreFromSourceProviderCache(cachedInfo);
+ popScope(functionScope, TreeBuilder::NeedsFreeVariableInfo);
+
+ m_token = cachedInfo->endFunctionToken();
+
+ if (endColumnIsOnStartLine)
+ m_token.m_location.lineStartOffset = currentLineStartOffset;
+
+ m_lexer->setOffset(m_token.m_location.endOffset, m_token.m_location.lineStartOffset);
+ m_lexer->setLineNumber(m_token.m_location.line);
+
+ switch (functionBodyType) {
+ case ArrowFunctionBodyExpression:
+ next();
+ context.setEndOffset(functionInfo.body, m_lexer->currentOffset());
+ break;
+ case ArrowFunctionBodyBlock:
+ case StandardFunctionBodyBlock:
+ context.setEndOffset(functionInfo.body, m_lexer->currentOffset());
+ next();
+ break;
+ }
+ functionInfo.endLine = m_lastTokenEndPosition.line;
+ return true;
+ }
+
+ return false;
+ };
+
+ SyntaxChecker syntaxChecker(const_cast<VM*>(m_vm), m_lexer.get());
+
+ if (UNLIKELY((SourceParseModeSet(SourceParseMode::ArrowFunctionMode, SourceParseMode::AsyncArrowFunctionMode).contains(mode)))) {
+ startLocation = tokenLocation();
+ functionInfo.startLine = tokenLine();
+ startColumn = tokenColumn();
+
+ parametersStart = m_token.m_location.startOffset;
+ functionInfo.startOffset = parametersStart;
+ functionInfo.parametersStartColumn = startColumn;
+
+ if (loadCachedFunction())
+ return true;
+
+ {
+ // Parse formal parameters with [+Yield] parameterization, in order to ban YieldExpressions
+ // in ArrowFormalParameters, per ES6 #sec-arrow-function-definitions-static-semantics-early-errors.
+ Scope::MaybeParseAsGeneratorForScope parseAsGenerator(functionScope, parentScope->isGenerator());
+ SetForScope<bool> overrideAllowAwait(m_parserState.allowAwait, !isAsyncFunctionParseMode(mode));
+ parseFunctionParameters(syntaxChecker, mode, functionInfo);
+ propagateError();
+ }
+
+ matchOrFail(ARROWFUNCTION, "Expected a '=>' after arrow function parameter declaration");
+
+ if (m_lexer->prevTerminator())
+ failDueToUnexpectedToken();
+
+ ASSERT(constructorKind == ConstructorKind::None);
+
+ // Check if arrow body start with {. If it true it mean that arrow function is Fat arrow function
+ // and we need use common approach to parse function body
+ next();
+ functionBodyType = match(OPENBRACE) ? ArrowFunctionBodyBlock : ArrowFunctionBodyExpression;
+ } else {
+ // http://ecma-international.org/ecma-262/6.0/#sec-function-definitions
+ // FunctionExpression :
+ // function BindingIdentifieropt ( FormalParameters ) { FunctionBody }
+ //
+ // FunctionDeclaration[Yield, Default] :
+ // function BindingIdentifier[?Yield] ( FormalParameters ) { FunctionBody }
+ // [+Default] function ( FormalParameters ) { FunctionBody }
+ //
+ // GeneratorDeclaration[Yield, Default] :
+ // function * BindingIdentifier[?Yield] ( FormalParameters[Yield] ) { GeneratorBody }
+ // [+Default] function * ( FormalParameters[Yield] ) { GeneratorBody }
+ //
+ // GeneratorExpression :
+ // function * BindingIdentifier[Yield]opt ( FormalParameters[Yield] ) { GeneratorBody }
+ //
+ // The name of FunctionExpression and AsyncFunctionExpression can accept "yield" even in the context of generator.
+ bool upperScopeIsGenerator = false;
+ if (!(functionDefinitionType == FunctionDefinitionType::Expression && SourceParseModeSet(SourceParseMode::NormalFunctionMode, SourceParseMode::AsyncFunctionMode).contains(mode)))
+ upperScopeIsGenerator = upperScope(1)->isGenerator();
+
+ if (requirements != FunctionNameRequirements::Unnamed) {
+ ASSERT_WITH_MESSAGE(!(requirements == FunctionNameRequirements::None && !functionInfo.name), "When specifying FunctionNameRequirements::None, we need to initialize functionInfo.name with the default value in the caller side.");
+ if (matchSpecIdentifier(upperScopeIsGenerator)) {
+ functionInfo.name = m_token.m_data.ident;
+ m_parserState.lastFunctionName = functionInfo.name;
+ if (UNLIKELY(isDisallowedAwaitFunctionName))
+ semanticFailIfTrue(functionDefinitionType == FunctionDefinitionType::Declaration || isAsyncFunctionWrapperParseMode(mode), "Cannot declare function named 'await' ", isDisallowedAwaitFunctionNameReason);
+ else if (isAsyncFunctionWrapperParseMode(mode) && match(AWAIT) && functionDefinitionType == FunctionDefinitionType::Expression)
+ semanticFail("Cannot declare async function named 'await'");
+ next();
+ if (!nameIsInContainingScope)
+ failIfTrueIfStrict(functionScope->declareCallee(functionInfo.name) & DeclarationResult::InvalidStrictMode, "'", functionInfo.name->impl(), "' is not a valid ", stringForFunctionMode(mode), " name in strict mode");
+ } else if (requirements == FunctionNameRequirements::Named) {
+ if (match(OPENPAREN)) {
+ semanticFailIfTrue(mode == SourceParseMode::NormalFunctionMode, "Function statements must have a name");
+ semanticFailIfTrue(mode == SourceParseMode::AsyncFunctionMode, "Async function statements must have a name");
+ }
+ semanticFailureDueToKeyword(stringForFunctionMode(mode), " name");
+ failDueToUnexpectedToken();
+ return false;
+ }
+ ASSERT(functionInfo.name);
+ }
+
+ startLocation = tokenLocation();
+ functionInfo.startLine = tokenLine();
+ startColumn = tokenColumn();
+ functionInfo.parametersStartColumn = startColumn;
+
+ parametersStart = m_token.m_location.startOffset;
+ functionInfo.startOffset = parametersStart;
+
+ if (loadCachedFunction())
+ return true;
+ {
+ SetForScope<bool> overrideAllowAwait(m_parserState.allowAwait, !isAsyncFunctionParseMode(mode));
+ parseFunctionParameters(syntaxChecker, mode, functionInfo);
+ propagateError();
+ }
- functionScope->restoreFromSourceProviderCache(cachedInfo);
- failIfFalse(popScope(functionScope, TreeBuilder::NeedsFreeVariableInfo), "Parser error");
+ matchOrFail(OPENBRACE, "Expected an opening '{' at the start of a ", stringForFunctionMode(mode), " body");
- closeBraceOffset = cachedInfo->closeBraceOffset;
+ // BytecodeGenerator emits code to throw TypeError when a class constructor is "call"ed.
+ // Set ConstructorKind to None for non-constructor methods of classes.
+
+ if (m_defaultConstructorKind != ConstructorKind::None) {
+ constructorKind = m_defaultConstructorKind;
+ expectedSuperBinding = m_defaultConstructorKind == ConstructorKind::Extends ? SuperBinding::Needed : SuperBinding::NotNeeded;
+ }
+
+ functionBodyType = StandardFunctionBodyBlock;
+ }
- context.setFunctionNameStart(body, functionNameStart);
- m_token = cachedInfo->closeBraceToken();
+ functionScope->setConstructorKind(constructorKind);
+ functionScope->setExpectedSuperBinding(expectedSuperBinding);
- m_lexer->setOffset(m_token.m_location.endOffset, m_token.m_location.lineStartOffset);
- m_lexer->setLineNumber(m_token.m_location.line);
-
- next();
- return true;
+ m_parserState.lastFunctionName = lastFunctionName;
+ ParserState oldState = internalSaveParserState();
+
+ // FIXME: https://bugs.webkit.org/show_bug.cgi?id=156962
+ // This loop collects the set of capture candidates that aren't
+ // part of the set of this function's declared parameters. We will
+ // figure out which parameters are captured for this function when
+ // we actually generate code for it. For now, we just propagate to
+ // our parent scopes which variables we might have closed over that
+ // belong to them. This is necessary for correctness when using
+ // the source provider cache because we can't close over a variable
+ // that we don't claim to close over. The source provider cache must
+ // know this information to properly cache this function.
+ // This might work itself out nicer if we declared a different
+ // Scope struct for the parameters (because they are indeed implemented
+ // as their own scope).
+ UniquedStringImplPtrSet nonLocalCapturesFromParameterExpressions;
+ functionScope->forEachUsedVariable([&] (UniquedStringImpl* impl) {
+ if (!functionScope->hasDeclaredParameter(impl)) {
+ nonLocalCapturesFromParameterExpressions.add(impl);
+ if (TreeBuilder::NeedsFreeVariableInfo)
+ parentScope->addClosedVariableCandidateUnconditionally(impl);
+ }
+ });
+
+ auto performParsingFunctionBody = [&] {
+ return parseFunctionBody(context, syntaxChecker, startLocation, startColumn, functionKeywordStart, functionNameStart, parametersStart, constructorKind, expectedSuperBinding, functionBodyType, functionInfo.parameterCount, mode);
+ };
+
+ if (isGeneratorOrAsyncFunctionWrapperParseMode(mode)) {
+ AutoPopScopeRef generatorBodyScope(this, pushScope());
+ SourceParseMode innerParseMode = SourceParseMode::GeneratorBodyMode;
+ if (isAsyncFunctionWrapperParseMode(mode)) {
+ innerParseMode = mode == SourceParseMode::AsyncArrowFunctionMode
+ ? SourceParseMode::AsyncArrowFunctionBodyMode
+ : SourceParseMode::AsyncFunctionBodyMode;
+ }
+ generatorBodyScope->setSourceParseMode(innerParseMode);
+ generatorBodyScope->setConstructorKind(ConstructorKind::None);
+ generatorBodyScope->setExpectedSuperBinding(expectedSuperBinding);
+
+ // Disallow 'use strict' directives in the implicit inner function if
+ // needed.
+ if (functionScope->hasNonSimpleParameterList())
+ generatorBodyScope->setHasNonSimpleParameterList();
+
+ functionInfo.body = performParsingFunctionBody();
+
+ // When a generator has a "use strict" directive, a generator function wrapping it should be strict mode.
+ if (generatorBodyScope->strictMode())
+ functionScope->setStrictMode();
+
+ popScope(generatorBodyScope, TreeBuilder::NeedsFreeVariableInfo);
+ } else
+ functionInfo.body = performParsingFunctionBody();
+
+ restoreParserState(oldState);
+ failIfFalse(functionInfo.body, "Cannot parse the body of this ", stringForFunctionMode(mode));
+ context.setEndOffset(functionInfo.body, m_lexer->currentOffset());
+ if (functionScope->strictMode() && requirements != FunctionNameRequirements::Unnamed) {
+ ASSERT(functionInfo.name);
+ RELEASE_ASSERT(SourceParseModeSet(SourceParseMode::NormalFunctionMode, SourceParseMode::MethodMode, SourceParseMode::ArrowFunctionMode, SourceParseMode::GeneratorBodyMode, SourceParseMode::GeneratorWrapperFunctionMode).contains(mode) || isAsyncFunctionWrapperParseMode(mode));
+ semanticFailIfTrue(m_vm->propertyNames->arguments == *functionInfo.name, "'", functionInfo.name->impl(), "' is not a valid function name in strict mode");
+ semanticFailIfTrue(m_vm->propertyNames->eval == *functionInfo.name, "'", functionInfo.name->impl(), "' is not a valid function name in strict mode");
}
- m_lastFunctionName = lastFunctionName;
- ParserState oldState = saveState();
- body = parseFunctionBody(context);
- restoreState(oldState);
- failIfFalse(body, "Cannot parse the body of this ", stringForFunctionMode(mode));
- if (functionScope->strictMode() && name) {
- RELEASE_ASSERT(mode == FunctionMode);
- semanticFailIfTrue(m_vm->propertyNames->arguments == *name, "'", name->impl(), "' is not a valid function name in strict mode");
- semanticFailIfTrue(m_vm->propertyNames->eval == *name, "'", name->impl(), "' is not a valid function name in strict mode");
- }
- closeBraceOffset = m_token.m_data.offset;
- unsigned closeBraceLine = m_token.m_data.line;
- unsigned closeBraceLineStartOffset = m_token.m_data.lineStartOffset;
+
+ JSTokenLocation location = JSTokenLocation(m_token.m_location);
+ functionInfo.endOffset = m_token.m_data.offset;
+ if (functionBodyType == ArrowFunctionBodyExpression) {
+ location = locationBeforeLastToken();
+ functionInfo.endOffset = location.endOffset;
+ } else {
+ recordFunctionEntryLocation(JSTextPosition(startLocation.line, startLocation.startOffset, startLocation.lineStartOffset));
+ recordFunctionLeaveLocation(JSTextPosition(location.line, location.startOffset, location.lineStartOffset));
+ }
+
// Cache the tokenizer state and the function scope the first time the function is parsed.
// Any future reparsing can then skip the function.
- static const int minimumFunctionLengthToCache = 16;
+ // For arrow function is 8 = x=>x + 4 symbols;
+ // For ordinary function is 16 = function(){} + 4 symbols
+ const int minimumSourceLengthToCache = functionBodyType == StandardFunctionBodyBlock ? 16 : 8;
std::unique_ptr<SourceProviderCacheItem> newInfo;
- int functionLength = closeBraceOffset - openBraceOffset;
- if (TreeBuilder::CanUseFunctionCache && m_functionCache && functionLength > minimumFunctionLengthToCache) {
+ int sourceLength = functionInfo.endOffset - functionInfo.startOffset;
+ if (TreeBuilder::CanUseFunctionCache && m_functionCache && sourceLength > minimumSourceLengthToCache) {
SourceProviderCacheItemCreationParameters parameters;
+ parameters.endFunctionOffset = functionInfo.endOffset;
parameters.functionNameStart = functionNameStart;
- parameters.closeBraceLine = closeBraceLine;
- parameters.closeBraceOffset = closeBraceOffset;
- parameters.closeBraceLineStartOffset = closeBraceLineStartOffset;
- functionScope->fillParametersForSourceProviderCache(parameters);
+ parameters.lastTokenLine = location.line;
+ parameters.lastTokenStartOffset = location.startOffset;
+ parameters.lastTokenEndOffset = location.endOffset;
+ parameters.lastTokenLineStartOffset = location.lineStartOffset;
+ parameters.parameterCount = functionInfo.parameterCount;
+ parameters.constructorKind = constructorKind;
+ parameters.expectedSuperBinding = expectedSuperBinding;
+ if (functionBodyType == ArrowFunctionBodyExpression) {
+ parameters.isBodyArrowExpression = true;
+ parameters.tokenType = m_token.m_type;
+ }
+ functionScope->fillParametersForSourceProviderCache(parameters, nonLocalCapturesFromParameterExpressions);
newInfo = SourceProviderCacheItem::create(parameters);
-
}
- context.setFunctionNameStart(body, functionNameStart);
- failIfFalse(popScope(functionScope, TreeBuilder::NeedsFreeVariableInfo), "Parser error");
- matchOrFail(CLOSEBRACE, "Expected a closing '}' after a ", stringForFunctionMode(mode), " body");
+ popScope(functionScope, TreeBuilder::NeedsFreeVariableInfo);
+ if (functionBodyType != ArrowFunctionBodyExpression) {
+ matchOrFail(CLOSEBRACE, "Expected a closing '}' after a ", stringForFunctionMode(mode), " body");
+ next();
+ }
+
if (newInfo)
- m_functionCache->add(openBraceOffset, std::move(newInfo));
+ m_functionCache->add(functionInfo.startOffset, WTFMove(newInfo));
- next();
+ functionInfo.endLine = m_lastTokenEndPosition.line;
return true;
}
+static NO_RETURN_DUE_TO_CRASH FunctionMetadataNode* getMetadata(ParserFunctionInfo<SyntaxChecker>&) { RELEASE_ASSERT_NOT_REACHED(); }
+static FunctionMetadataNode* getMetadata(ParserFunctionInfo<ASTBuilder>& info) { return info.body; }
+
template <typename LexerType>
-template <class TreeBuilder> TreeStatement Parser<LexerType>::parseFunctionDeclaration(TreeBuilder& context)
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseFunctionDeclaration(TreeBuilder& context, ExportType exportType, DeclarationDefaultContext declarationDefaultContext)
{
ASSERT(match(FUNCTION));
JSTokenLocation location(tokenLocation());
+ unsigned functionKeywordStart = tokenStart();
next();
- const Identifier* name = 0;
- TreeFormalParameterList parameters = 0;
- TreeFunctionBody body = 0;
- unsigned openBraceOffset = 0;
- unsigned closeBraceOffset = 0;
- int bodyStartLine = 0;
- unsigned bodyStartColumn = 0;
- failIfFalse((parseFunctionInfo(context, FunctionNeedsName, FunctionMode, true, name, parameters, body, openBraceOffset, closeBraceOffset, bodyStartLine, bodyStartColumn)), "Cannot parse this function");
- failIfFalse(name, "Function statements must have a name");
- failIfFalseIfStrict(declareVariable(name), "Cannot declare a function named '", name->impl(), "' in strict mode");
- return context.createFuncDeclStatement(location, name, body, parameters, openBraceOffset, closeBraceOffset, bodyStartLine, m_lastTokenEndPosition.line, bodyStartColumn);
+ SourceParseMode parseMode = SourceParseMode::NormalFunctionMode;
+ if (consume(TIMES))
+ parseMode = SourceParseMode::GeneratorWrapperFunctionMode;
+
+ ParserFunctionInfo<TreeBuilder> functionInfo;
+ FunctionNameRequirements requirements = FunctionNameRequirements::Named;
+ if (declarationDefaultContext == DeclarationDefaultContext::ExportDefault) {
+ // Under the "export default" context, function declaration does not require the function name.
+ //
+ // ExportDeclaration:
+ // ...
+ // export default HoistableDeclaration[~Yield, +Default]
+ // ...
+ //
+ // HoistableDeclaration[Yield, Default]:
+ // FunctionDeclaration[?Yield, ?Default]
+ // GeneratorDeclaration[?Yield, ?Default]
+ //
+ // FunctionDeclaration[Yield, Default]:
+ // ...
+ // [+Default] function ( FormalParameters[~Yield] ) { FunctionBody[~Yield] }
+ //
+ // GeneratorDeclaration[Yield, Default]:
+ // ...
+ // [+Default] function * ( FormalParameters[+Yield] ) { GeneratorBody }
+ //
+ // In this case, we use "*default*" as this function declaration's name.
+ requirements = FunctionNameRequirements::None;
+ functionInfo.name = &m_vm->propertyNames->builtinNames().starDefaultPrivateName();
+ }
+
+ failIfFalse((parseFunctionInfo(context, requirements, parseMode, true, ConstructorKind::None, SuperBinding::NotNeeded, functionKeywordStart, functionInfo, FunctionDefinitionType::Declaration)), "Cannot parse this function");
+ ASSERT(functionInfo.name);
+
+ std::pair<DeclarationResultMask, ScopeRef> functionDeclaration = declareFunction(functionInfo.name);
+ DeclarationResultMask declarationResult = functionDeclaration.first;
+ failIfTrueIfStrict(declarationResult & DeclarationResult::InvalidStrictMode, "Cannot declare a function named '", functionInfo.name->impl(), "' in strict mode");
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration)
+ internalFailWithMessage(false, "Cannot declare a function that shadows a let/const/class/function variable '", functionInfo.name->impl(), "' in strict mode");
+ if (exportType == ExportType::Exported) {
+ ASSERT_WITH_MESSAGE(declarationDefaultContext != DeclarationDefaultContext::ExportDefault, "Export default case will export the name and binding in the caller.");
+ semanticFailIfFalse(exportName(*functionInfo.name), "Cannot export a duplicate function name: '", functionInfo.name->impl(), "'");
+ m_moduleScopeData->exportBinding(*functionInfo.name);
+ }
+
+ TreeStatement result = context.createFuncDeclStatement(location, functionInfo);
+ if (TreeBuilder::CreatesAST)
+ functionDeclaration.second->appendFunction(getMetadata(functionInfo));
+ return result;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseAsyncFunctionDeclaration(TreeBuilder& context, ExportType exportType, DeclarationDefaultContext declarationDefaultContext)
+{
+ ASSERT(match(FUNCTION));
+ JSTokenLocation location(tokenLocation());
+ unsigned functionKeywordStart = tokenStart();
+ next();
+ ParserFunctionInfo<TreeBuilder> functionInfo;
+ SourceParseMode parseMode = SourceParseMode::AsyncFunctionMode;
+ FunctionNameRequirements requirements = FunctionNameRequirements::Named;
+ if (declarationDefaultContext == DeclarationDefaultContext::ExportDefault) {
+ // Under the "export default" context, function declaration does not require the function name.
+ //
+ // ExportDeclaration:
+ // ...
+ // export default HoistableDeclaration[~Yield, +Default]
+ // ...
+ //
+ // HoistableDeclaration[Yield, Default]:
+ // FunctionDeclaration[?Yield, ?Default]
+ // GeneratorDeclaration[?Yield, ?Default]
+ //
+ // FunctionDeclaration[Yield, Default]:
+ // ...
+ // [+Default] function ( FormalParameters[~Yield] ) { FunctionBody[~Yield] }
+ //
+ // GeneratorDeclaration[Yield, Default]:
+ // ...
+ // [+Default] function * ( FormalParameters[+Yield] ) { GeneratorBody }
+ //
+ // In this case, we use "*default*" as this function declaration's name.
+ requirements = FunctionNameRequirements::None;
+ functionInfo.name = &m_vm->propertyNames->builtinNames().starDefaultPrivateName();
+ }
+
+ failIfFalse((parseFunctionInfo(context, requirements, parseMode, true, ConstructorKind::None, SuperBinding::NotNeeded, functionKeywordStart, functionInfo, FunctionDefinitionType::Declaration)), "Cannot parse this async function");
+ failIfFalse(functionInfo.name, "Async function statements must have a name");
+
+ std::pair<DeclarationResultMask, ScopeRef> functionDeclaration = declareFunction(functionInfo.name);
+ DeclarationResultMask declarationResult = functionDeclaration.first;
+ failIfTrueIfStrict(declarationResult & DeclarationResult::InvalidStrictMode, "Cannot declare an async function named '", functionInfo.name->impl(), "' in strict mode");
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration)
+ internalFailWithMessage(false, "Cannot declare an async function that shadows a let/const/class/function variable '", functionInfo.name->impl(), "' in strict mode");
+ if (exportType == ExportType::Exported) {
+ semanticFailIfFalse(exportName(*functionInfo.name), "Cannot export a duplicate function name: '", functionInfo.name->impl(), "'");
+ m_moduleScopeData->exportBinding(*functionInfo.name);
+ }
+
+ TreeStatement result = context.createFuncDeclStatement(location, functionInfo);
+ if (TreeBuilder::CreatesAST)
+ functionDeclaration.second->appendFunction(getMetadata(functionInfo));
+ return result;
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseClassDeclaration(TreeBuilder& context, ExportType exportType, DeclarationDefaultContext declarationDefaultContext)
+{
+ ASSERT(match(CLASSTOKEN));
+ JSTokenLocation location(tokenLocation());
+ JSTextPosition classStart = tokenStartPosition();
+ unsigned classStartLine = tokenLine();
+
+ ParserClassInfo<TreeBuilder> info;
+ FunctionNameRequirements requirements = FunctionNameRequirements::Named;
+ if (declarationDefaultContext == DeclarationDefaultContext::ExportDefault) {
+ // Under the "export default" context, class declaration does not require the class name.
+ //
+ // ExportDeclaration:
+ // ...
+ // export default ClassDeclaration[~Yield, +Default]
+ // ...
+ //
+ // ClassDeclaration[Yield, Default]:
+ // ...
+ // [+Default] class ClassTail[?Yield]
+ //
+ // In this case, we use "*default*" as this class declaration's name.
+ requirements = FunctionNameRequirements::None;
+ info.className = &m_vm->propertyNames->builtinNames().starDefaultPrivateName();
+ }
+
+ TreeClassExpression classExpr = parseClass(context, requirements, info);
+ failIfFalse(classExpr, "Failed to parse class");
+ ASSERT(info.className);
+
+ DeclarationResultMask declarationResult = declareVariable(info.className, DeclarationType::LetDeclaration);
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration)
+ internalFailWithMessage(false, "Cannot declare a class twice: '", info.className->impl(), "'");
+ if (exportType == ExportType::Exported) {
+ ASSERT_WITH_MESSAGE(declarationDefaultContext != DeclarationDefaultContext::ExportDefault, "Export default case will export the name and binding in the caller.");
+ semanticFailIfFalse(exportName(*info.className), "Cannot export a duplicate class name: '", info.className->impl(), "'");
+ m_moduleScopeData->exportBinding(*info.className);
+ }
+
+ JSTextPosition classEnd = lastTokenEndPosition();
+ unsigned classEndLine = tokenLine();
+
+ return context.createClassDeclStatement(location, classExpr, classStart, classEnd, classStartLine, classEndLine);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeClassExpression Parser<LexerType>::parseClass(TreeBuilder& context, FunctionNameRequirements requirements, ParserClassInfo<TreeBuilder>& info)
+{
+ ASSERT(match(CLASSTOKEN));
+ JSTokenLocation location(tokenLocation());
+ info.startLine = location.line;
+ info.startColumn = tokenColumn();
+ info.startOffset = location.startOffset;
+ next();
+
+ AutoPopScopeRef classScope(this, pushScope());
+ classScope->setIsLexicalScope();
+ classScope->preventVarDeclarations();
+ classScope->setStrictMode();
+
+ ASSERT_WITH_MESSAGE(requirements != FunctionNameRequirements::Unnamed, "Currently, there is no caller that uses FunctionNameRequirements::Unnamed for class syntax.");
+ ASSERT_WITH_MESSAGE(!(requirements == FunctionNameRequirements::None && !info.className), "When specifying FunctionNameRequirements::None, we need to initialize info.className with the default value in the caller side.");
+ if (match(IDENT)) {
+ info.className = m_token.m_data.ident;
+ next();
+ failIfTrue(classScope->declareLexicalVariable(info.className, true) & DeclarationResult::InvalidStrictMode, "'", info.className->impl(), "' is not a valid class name");
+ } else if (requirements == FunctionNameRequirements::Named) {
+ if (match(OPENBRACE))
+ semanticFail("Class statements must have a name");
+ semanticFailureDueToKeyword("class name");
+ failDueToUnexpectedToken();
+ }
+ ASSERT(info.className);
+
+ TreeExpression parentClass = 0;
+ if (consume(EXTENDS)) {
+ parentClass = parseMemberExpression(context);
+ failIfFalse(parentClass, "Cannot parse the parent class name");
+ }
+ const ConstructorKind constructorKind = parentClass ? ConstructorKind::Extends : ConstructorKind::Base;
+
+ consumeOrFail(OPENBRACE, "Expected opening '{' at the start of a class body");
+
+ TreeExpression constructor = 0;
+ TreePropertyList staticMethods = 0;
+ TreePropertyList instanceMethods = 0;
+ TreePropertyList instanceMethodsTail = 0;
+ TreePropertyList staticMethodsTail = 0;
+ while (!match(CLOSEBRACE)) {
+ if (match(SEMICOLON)) {
+ next();
+ continue;
+ }
+
+ JSTokenLocation methodLocation(tokenLocation());
+ unsigned methodStart = tokenStart();
+
+ // For backwards compatibility, "static" is a non-reserved keyword in non-strict mode.
+ bool isStaticMethod = false;
+ if (match(RESERVED_IF_STRICT) && *m_token.m_data.ident == m_vm->propertyNames->staticKeyword) {
+ SavePoint savePoint = createSavePoint();
+ next();
+ if (match(OPENPAREN)) {
+ // Reparse "static()" as a method named "static".
+ restoreSavePoint(savePoint);
+ } else
+ isStaticMethod = true;
+ }
+
+ // FIXME: Figure out a way to share more code with parseProperty.
+ const CommonIdentifiers& propertyNames = *m_vm->propertyNames;
+ const Identifier* ident = &propertyNames.nullIdentifier;
+ TreeExpression computedPropertyName = 0;
+ bool isGetter = false;
+ bool isSetter = false;
+ bool isGenerator = false;
+ bool isAsync = false;
+ bool isAsyncMethod = false;
+ if (consume(TIMES))
+ isGenerator = true;
+
+parseMethod:
+ switch (m_token.m_type) {
+ namedKeyword:
+ case STRING:
+ ident = m_token.m_data.ident;
+ ASSERT(ident);
+ next();
+ break;
+ case ASYNC:
+ isAsync = !isGenerator && !isAsyncMethod;
+ FALLTHROUGH;
+ case IDENT:
+ case AWAIT:
+ ident = m_token.m_data.ident;
+ ASSERT(ident);
+ next();
+ if (!isGenerator && !isAsyncMethod && (matchIdentifierOrKeyword() || match(STRING) || match(DOUBLE) || match(INTEGER) || match(OPENBRACKET))) {
+ isGetter = *ident == propertyNames.get;
+ isSetter = *ident == propertyNames.set;
+
+ if (UNLIKELY(isAsync && !m_lexer->prevTerminator())) {
+ isAsyncMethod = true;
+ goto parseMethod;
+ }
+ }
+ break;
+ case DOUBLE:
+ case INTEGER:
+ ident = &m_parserArena.identifierArena().makeNumericIdentifier(const_cast<VM*>(m_vm), m_token.m_data.doubleValue);
+ ASSERT(ident);
+ next();
+ break;
+ case OPENBRACKET:
+ next();
+ computedPropertyName = parseAssignmentExpression(context);
+ failIfFalse(computedPropertyName, "Cannot parse computed property name");
+ handleProductionOrFail(CLOSEBRACKET, "]", "end", "computed property name");
+ break;
+ default:
+ if (m_token.m_type & KeywordTokenFlag)
+ goto namedKeyword;
+ failDueToUnexpectedToken();
+ }
+
+ TreeProperty property;
+ const bool alwaysStrictInsideClass = true;
+ if (isGetter || isSetter) {
+ bool isClassProperty = true;
+ property = parseGetterSetter(context, alwaysStrictInsideClass, isGetter ? PropertyNode::Getter : PropertyNode::Setter, methodStart,
+ ConstructorKind::None, isClassProperty);
+ failIfFalse(property, "Cannot parse this method");
+ } else {
+ ParserFunctionInfo<TreeBuilder> methodInfo;
+ bool isConstructor = !isStaticMethod && *ident == propertyNames.constructor;
+ SourceParseMode parseMode = SourceParseMode::MethodMode;
+ if (isAsyncMethod) {
+ isConstructor = false;
+ parseMode = SourceParseMode::AsyncMethodMode;
+ semanticFailIfTrue(*ident == m_vm->propertyNames->prototype, "Cannot declare an async method named 'prototype'");
+ semanticFailIfTrue(*ident == m_vm->propertyNames->constructor, "Cannot declare an async method named 'constructor'");
+ } else if (isGenerator) {
+ isConstructor = false;
+ parseMode = SourceParseMode::GeneratorWrapperFunctionMode;
+ semanticFailIfTrue(*ident == m_vm->propertyNames->prototype, "Cannot declare a generator named 'prototype'");
+ semanticFailIfTrue(*ident == m_vm->propertyNames->constructor, "Cannot declare a generator named 'constructor'");
+ }
+ methodInfo.name = isConstructor ? info.className : ident;
+ failIfFalse((parseFunctionInfo(context, FunctionNameRequirements::Unnamed, parseMode, false, isConstructor ? constructorKind : ConstructorKind::None, SuperBinding::Needed, methodStart, methodInfo, FunctionDefinitionType::Method)), "Cannot parse this method");
+
+ TreeExpression method = context.createMethodDefinition(methodLocation, methodInfo);
+ if (isConstructor) {
+ semanticFailIfTrue(constructor, "Cannot declare multiple constructors in a single class");
+ constructor = method;
+ continue;
+ }
+
+ // FIXME: Syntax error when super() is called
+ semanticFailIfTrue(isStaticMethod && methodInfo.name && *methodInfo.name == propertyNames.prototype,
+ "Cannot declare a static method named 'prototype'");
+
+ bool isClassProperty = true;
+ if (computedPropertyName) {
+ property = context.createProperty(computedPropertyName, method, static_cast<PropertyNode::Type>(PropertyNode::Constant | PropertyNode::Computed),
+ PropertyNode::Unknown, alwaysStrictInsideClass, SuperBinding::Needed, isClassProperty);
+ } else
+ property = context.createProperty(methodInfo.name, method, PropertyNode::Constant, PropertyNode::Unknown, alwaysStrictInsideClass, SuperBinding::Needed, isClassProperty);
+ }
+
+ TreePropertyList& tail = isStaticMethod ? staticMethodsTail : instanceMethodsTail;
+ if (tail)
+ tail = context.createPropertyList(methodLocation, property, tail);
+ else {
+ tail = context.createPropertyList(methodLocation, property);
+ if (isStaticMethod)
+ staticMethods = tail;
+ else
+ instanceMethods = tail;
+ }
+ }
+
+ info.endOffset = tokenLocation().endOffset - 1;
+ consumeOrFail(CLOSEBRACE, "Expected a closing '}' after a class body");
+
+ auto classExpression = context.createClassExpr(location, info, classScope->finalizeLexicalEnvironment(), constructor, parentClass, instanceMethods, staticMethods);
+ popScope(classScope, TreeBuilder::NeedsFreeVariableInfo);
+ return classExpression;
}
struct LabelInfo {
@@ -1333,7 +2731,7 @@ struct LabelInfo {
};
template <typename LexerType>
-template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionOrLabelStatement(TreeBuilder& context)
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionOrLabelStatement(TreeBuilder& context, bool allowFunctionDeclarationAsStatement)
{
/* Expression and Label statements are ambiguous at LL(1), so we have a
@@ -1354,6 +2752,10 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionOrL
failDueToUnexpectedToken();
return context.createExprStatement(location, expression, start, m_lastTokenEndPosition.line);
}
+
+ if (UNLIKELY(match(AWAIT)))
+ semanticFailIfTrue(isDisallowedIdentifierAwait(m_token), "Can't use 'await' as a label ", disallowedIdentifierAwaitReason());
+
const Identifier* ident = m_token.m_data.ident;
JSTextPosition end = tokenEndPosition();
next();
@@ -1366,7 +2768,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionOrL
failIfTrue(getLabel(ident), "Cannot find scope for the label '", ident->impl(), "'");
labels.append(LabelInfo(ident, start, end));
}
- } while (match(IDENT));
+ } while (matchSpecIdentifier());
bool isLoop = false;
switch (m_token.m_type) {
case FOR:
@@ -1379,14 +2781,16 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionOrL
break;
}
const Identifier* unused = 0;
+ ScopeRef labelScope = currentScope();
if (!m_syntaxAlreadyValidated) {
for (size_t i = 0; i < labels.size(); i++)
pushLabel(labels[i].m_ident, isLoop);
}
+ m_immediateParentAllowsFunctionDeclarationInStatement = allowFunctionDeclarationAsStatement;
TreeStatement statement = parseStatement(context, unused);
if (!m_syntaxAlreadyValidated) {
for (size_t i = 0; i < labels.size(); i++)
- popLabel();
+ popLabel(labelScope);
}
failIfFalse(statement, "Cannot parse statement");
for (size_t i = 0; i < labels.size(); i++) {
@@ -1399,6 +2803,19 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionOrL
template <typename LexerType>
template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExpressionStatement(TreeBuilder& context)
{
+ switch (m_token.m_type) {
+ // Consult: http://www.ecma-international.org/ecma-262/6.0/index.html#sec-expression-statement
+ // The ES6 spec mandates that we should fail from FUNCTION token here. We handle this case
+ // in parseStatement() which is the only caller of parseExpressionStatement().
+ // We actually allow FUNCTION in situations where it should not be allowed unless we're in strict mode.
+ case CLASSTOKEN:
+ failWithMessage("'class' declaration is not directly within a block statement");
+ break;
+ default:
+ // FIXME: when implementing 'let' we should fail when we see the token sequence "let [".
+ // https://bugs.webkit.org/show_bug.cgi?id=142944
+ break;
+ }
JSTextPosition start = tokenStartPosition();
JSTokenLocation location(tokenLocation());
TreeExpression expression = parseExpression(context);
@@ -1414,14 +2831,16 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseIfStatement(T
JSTokenLocation ifLocation(tokenLocation());
int start = tokenLine();
next();
- handleProductionOrFail(OPENPAREN, "(", "start", "'if' condition");
+ handleProductionOrFail2(OPENPAREN, "(", "start", "'if' condition");
TreeExpression condition = parseExpression(context);
failIfFalse(condition, "Expected a expression as the condition for an if statement");
+ recordPauseLocation(context.breakpointLocation(condition));
int end = tokenLine();
- handleProductionOrFail(CLOSEPAREN, ")", "end", "'if' condition");
+ handleProductionOrFail2(CLOSEPAREN, ")", "end", "'if' condition");
const Identifier* unused = 0;
+ m_immediateParentAllowsFunctionDeclarationInStatement = true;
TreeStatement trueBlock = parseStatement(context, unused);
failIfFalse(trueBlock, "Expected a statement as the body of an if block");
@@ -1438,6 +2857,7 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseIfStatement(T
next();
if (!match(IF)) {
const Identifier* unused = 0;
+ m_immediateParentAllowsFunctionDeclarationInStatement = true;
TreeStatement block = parseStatement(context, unused);
failIfFalse(block, "Expected a statement as the body of an else block");
statementStack.append(block);
@@ -1447,13 +2867,15 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseIfStatement(T
int innerStart = tokenLine();
next();
- handleProductionOrFail(OPENPAREN, "(", "start", "'if' condition");
+ handleProductionOrFail2(OPENPAREN, "(", "start", "'if' condition");
TreeExpression innerCondition = parseExpression(context);
failIfFalse(innerCondition, "Expected a expression as the condition for an if statement");
+ recordPauseLocation(context.breakpointLocation(innerCondition));
int innerEnd = tokenLine();
- handleProductionOrFail(CLOSEPAREN, ")", "end", "'if' condition");
+ handleProductionOrFail2(CLOSEPAREN, ")", "end", "'if' condition");
const Identifier* unused = 0;
+ m_immediateParentAllowsFunctionDeclarationInStatement = true;
TreeStatement innerTrueBlock = parseStatement(context, unused);
failIfFalse(innerTrueBlock, "Expected a statement as the body of an if block");
tokenLocationStack.append(tempLocation);
@@ -1471,7 +2893,9 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseIfStatement(T
posStack.removeLast();
JSTokenLocation elseLocation = tokenLocationStack.last();
tokenLocationStack.removeLast();
- statementStack.append(context.createIfStatement(elseLocation, condition, trueBlock, 0, pos.first, pos.second));
+ TreeStatement ifStatement = context.createIfStatement(elseLocation, condition, trueBlock, 0, pos.first, pos.second);
+ context.setEndOffset(ifStatement, context.endOffset(trueBlock));
+ statementStack.append(ifStatement);
}
while (!exprStack.isEmpty()) {
@@ -1485,57 +2909,533 @@ template <class TreeBuilder> TreeStatement Parser<LexerType>::parseIfStatement(T
posStack.removeLast();
JSTokenLocation elseLocation = tokenLocationStack.last();
tokenLocationStack.removeLast();
- statementStack.append(context.createIfStatement(elseLocation, condition, trueBlock, falseBlock, pos.first, pos.second));
+ TreeStatement ifStatement = context.createIfStatement(elseLocation, condition, trueBlock, falseBlock, pos.first, pos.second);
+ context.setEndOffset(ifStatement, context.endOffset(falseBlock));
+ statementStack.append(ifStatement);
}
return context.createIfStatement(ifLocation, condition, trueBlock, statementStack.last(), start, end);
}
template <typename LexerType>
+template <class TreeBuilder> typename TreeBuilder::ModuleName Parser<LexerType>::parseModuleName(TreeBuilder& context)
+{
+ // ModuleName (ModuleSpecifier in the spec) represents the module name imported by the script.
+ // http://www.ecma-international.org/ecma-262/6.0/#sec-imports
+ // http://www.ecma-international.org/ecma-262/6.0/#sec-exports
+ JSTokenLocation specifierLocation(tokenLocation());
+ failIfFalse(match(STRING), "Imported modules names must be string literals");
+ const Identifier* moduleName = m_token.m_data.ident;
+ next();
+ return context.createModuleName(specifierLocation, *moduleName);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> typename TreeBuilder::ImportSpecifier Parser<LexerType>::parseImportClauseItem(TreeBuilder& context, ImportSpecifierType specifierType)
+{
+ // Produced node is the item of the ImportClause.
+ // That is the ImportSpecifier, ImportedDefaultBinding or NameSpaceImport.
+ // http://www.ecma-international.org/ecma-262/6.0/#sec-imports
+ JSTokenLocation specifierLocation(tokenLocation());
+ JSToken localNameToken;
+ const Identifier* importedName = nullptr;
+ const Identifier* localName = nullptr;
+
+ switch (specifierType) {
+ case ImportSpecifierType::NamespaceImport: {
+ // NameSpaceImport :
+ // * as ImportedBinding
+ // e.g.
+ // * as namespace
+ ASSERT(match(TIMES));
+ importedName = &m_vm->propertyNames->timesIdentifier;
+ next();
+
+ failIfFalse(matchContextualKeyword(m_vm->propertyNames->as), "Expected 'as' before imported binding name");
+ next();
+
+ failIfFalse(matchSpecIdentifier(), "Expected a variable name for the import declaration");
+ localNameToken = m_token;
+ localName = m_token.m_data.ident;
+ next();
+ break;
+ }
+
+ case ImportSpecifierType::NamedImport: {
+ // ImportSpecifier :
+ // ImportedBinding
+ // IdentifierName as ImportedBinding
+ // e.g.
+ // A
+ // A as B
+ ASSERT(matchIdentifierOrKeyword());
+ localNameToken = m_token;
+ localName = m_token.m_data.ident;
+ importedName = localName;
+ next();
+
+ if (matchContextualKeyword(m_vm->propertyNames->as)) {
+ next();
+ failIfFalse(matchSpecIdentifier(), "Expected a variable name for the import declaration");
+ localNameToken = m_token;
+ localName = m_token.m_data.ident;
+ next();
+ }
+ break;
+ }
+
+ case ImportSpecifierType::DefaultImport: {
+ // ImportedDefaultBinding :
+ // ImportedBinding
+ ASSERT(matchSpecIdentifier());
+ localNameToken = m_token;
+ localName = m_token.m_data.ident;
+ importedName = &m_vm->propertyNames->defaultKeyword;
+ next();
+ break;
+ }
+ }
+
+ semanticFailIfTrue(localNameToken.m_type == AWAIT, "Cannot use 'await' as an imported binding name");
+ semanticFailIfTrue(localNameToken.m_type & KeywordTokenFlag, "Cannot use keyword as imported binding name");
+ DeclarationResultMask declarationResult = declareVariable(localName, DeclarationType::ConstDeclaration, (specifierType == ImportSpecifierType::NamespaceImport) ? DeclarationImportType::ImportedNamespace : DeclarationImportType::Imported);
+ if (declarationResult != DeclarationResult::Valid) {
+ failIfTrueIfStrict(declarationResult & DeclarationResult::InvalidStrictMode, "Cannot declare an imported binding named ", localName->impl(), " in strict mode");
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration)
+ internalFailWithMessage(false, "Cannot declare an imported binding name twice: '", localName->impl(), "'");
+ }
+
+ return context.createImportSpecifier(specifierLocation, *importedName, *localName);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseImportDeclaration(TreeBuilder& context)
+{
+ // http://www.ecma-international.org/ecma-262/6.0/#sec-imports
+ ASSERT(match(IMPORT));
+ JSTokenLocation importLocation(tokenLocation());
+ next();
+
+ auto specifierList = context.createImportSpecifierList();
+
+ if (match(STRING)) {
+ // import ModuleSpecifier ;
+ auto moduleName = parseModuleName(context);
+ failIfFalse(moduleName, "Cannot parse the module name");
+ failIfFalse(autoSemiColon(), "Expected a ';' following a targeted import declaration");
+ return context.createImportDeclaration(importLocation, specifierList, moduleName);
+ }
+
+ bool isFinishedParsingImport = false;
+ if (matchSpecIdentifier()) {
+ // ImportedDefaultBinding :
+ // ImportedBinding
+ auto specifier = parseImportClauseItem(context, ImportSpecifierType::DefaultImport);
+ failIfFalse(specifier, "Cannot parse the default import");
+ context.appendImportSpecifier(specifierList, specifier);
+ if (match(COMMA))
+ next();
+ else
+ isFinishedParsingImport = true;
+ }
+
+ if (!isFinishedParsingImport) {
+ if (match(TIMES)) {
+ // import NameSpaceImport FromClause ;
+ auto specifier = parseImportClauseItem(context, ImportSpecifierType::NamespaceImport);
+ failIfFalse(specifier, "Cannot parse the namespace import");
+ context.appendImportSpecifier(specifierList, specifier);
+ } else if (match(OPENBRACE)) {
+ // NamedImports :
+ // { }
+ // { ImportsList }
+ // { ImportsList , }
+ next();
+
+ while (!match(CLOSEBRACE)) {
+ failIfFalse(matchIdentifierOrKeyword(), "Expected an imported name for the import declaration");
+ auto specifier = parseImportClauseItem(context, ImportSpecifierType::NamedImport);
+ failIfFalse(specifier, "Cannot parse the named import");
+ context.appendImportSpecifier(specifierList, specifier);
+ if (!consume(COMMA))
+ break;
+ }
+ handleProductionOrFail2(CLOSEBRACE, "}", "end", "import list");
+ } else
+ failWithMessage("Expected namespace import or import list");
+ }
+
+ // FromClause :
+ // from ModuleSpecifier
+
+ failIfFalse(matchContextualKeyword(m_vm->propertyNames->from), "Expected 'from' before imported module name");
+ next();
+
+ auto moduleName = parseModuleName(context);
+ failIfFalse(moduleName, "Cannot parse the module name");
+ failIfFalse(autoSemiColon(), "Expected a ';' following a targeted import declaration");
+
+ return context.createImportDeclaration(importLocation, specifierList, moduleName);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> typename TreeBuilder::ExportSpecifier Parser<LexerType>::parseExportSpecifier(TreeBuilder& context, Vector<std::pair<const Identifier*, const Identifier*>>& maybeExportedLocalNames, bool& hasKeywordForLocalBindings)
+{
+ // ExportSpecifier :
+ // IdentifierName
+ // IdentifierName as IdentifierName
+ // http://www.ecma-international.org/ecma-262/6.0/#sec-exports
+ ASSERT(matchIdentifierOrKeyword());
+ JSTokenLocation specifierLocation(tokenLocation());
+ if (m_token.m_type & KeywordTokenFlag)
+ hasKeywordForLocalBindings = true;
+ const Identifier* localName = m_token.m_data.ident;
+ const Identifier* exportedName = localName;
+ next();
+
+ if (matchContextualKeyword(m_vm->propertyNames->as)) {
+ next();
+ failIfFalse(matchIdentifierOrKeyword(), "Expected an exported name for the export declaration");
+ exportedName = m_token.m_data.ident;
+ next();
+ }
+
+ semanticFailIfFalse(exportName(*exportedName), "Cannot export a duplicate name '", exportedName->impl(), "'");
+ maybeExportedLocalNames.append(std::make_pair(localName, exportedName));
+ return context.createExportSpecifier(specifierLocation, *localName, *exportedName);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeStatement Parser<LexerType>::parseExportDeclaration(TreeBuilder& context)
+{
+ // http://www.ecma-international.org/ecma-262/6.0/#sec-exports
+ ASSERT(match(EXPORT));
+ JSTokenLocation exportLocation(tokenLocation());
+ next();
+
+ switch (m_token.m_type) {
+ case TIMES: {
+ // export * FromClause ;
+ next();
+
+ failIfFalse(matchContextualKeyword(m_vm->propertyNames->from), "Expected 'from' before exported module name");
+ next();
+ auto moduleName = parseModuleName(context);
+ failIfFalse(moduleName, "Cannot parse the 'from' clause");
+ failIfFalse(autoSemiColon(), "Expected a ';' following a targeted export declaration");
+
+ return context.createExportAllDeclaration(exportLocation, moduleName);
+ }
+
+ case DEFAULT: {
+ // export default HoistableDeclaration[Default]
+ // export default ClassDeclaration[Default]
+ // export default [lookahead not-in {function, class}] AssignmentExpression[In] ;
+
+ next();
+
+ TreeStatement result = 0;
+ bool isFunctionOrClassDeclaration = false;
+ const Identifier* localName = nullptr;
+
+ bool startsWithFunction = match(FUNCTION);
+ if (startsWithFunction || match(CLASSTOKEN)) {
+ SavePoint savePoint = createSavePoint();
+ isFunctionOrClassDeclaration = true;
+ next();
+
+ // ES6 Generators
+ if (startsWithFunction && match(TIMES))
+ next();
+ if (match(IDENT))
+ localName = m_token.m_data.ident;
+ restoreSavePoint(savePoint);
+ } else if (match(ASYNC)) {
+ SavePoint savePoint = createSavePoint();
+ next();
+ if (match(FUNCTION) && !m_lexer->prevTerminator()) {
+ next();
+ if (match(IDENT))
+ localName = m_token.m_data.ident;
+ isFunctionOrClassDeclaration = true;
+ }
+ restoreSavePoint(savePoint);
+ }
+
+ if (!localName)
+ localName = &m_vm->propertyNames->builtinNames().starDefaultPrivateName();
+
+ if (isFunctionOrClassDeclaration) {
+ if (startsWithFunction) {
+ ASSERT(match(FUNCTION));
+ DepthManager statementDepth(&m_statementDepth);
+ m_statementDepth = 1;
+ result = parseFunctionDeclaration(context, ExportType::NotExported, DeclarationDefaultContext::ExportDefault);
+ } else if (match(CLASSTOKEN)) {
+ result = parseClassDeclaration(context, ExportType::NotExported, DeclarationDefaultContext::ExportDefault);
+ } else {
+ ASSERT(match(ASYNC));
+ next();
+ DepthManager statementDepth(&m_statementDepth);
+ m_statementDepth = 1;
+ result = parseAsyncFunctionDeclaration(context, ExportType::NotExported, DeclarationDefaultContext::ExportDefault);
+ }
+ } else {
+ // export default expr;
+ //
+ // It should be treated as the same to the following.
+ //
+ // const *default* = expr;
+ // export { *default* as default }
+ //
+ // In the above example, *default* is the invisible variable to the users.
+ // We use the private symbol to represent the name of this variable.
+ JSTokenLocation location(tokenLocation());
+ JSTextPosition start = tokenStartPosition();
+ TreeExpression expression = parseAssignmentExpression(context);
+ failIfFalse(expression, "Cannot parse expression");
+
+ DeclarationResultMask declarationResult = declareVariable(&m_vm->propertyNames->builtinNames().starDefaultPrivateName(), DeclarationType::ConstDeclaration);
+ if (declarationResult & DeclarationResult::InvalidDuplicateDeclaration)
+ internalFailWithMessage(false, "Only one 'default' export is allowed");
+
+ TreeExpression assignment = context.createAssignResolve(location, m_vm->propertyNames->builtinNames().starDefaultPrivateName(), expression, start, start, tokenEndPosition(), AssignmentContext::ConstDeclarationStatement);
+ result = context.createExprStatement(location, assignment, start, tokenEndPosition());
+ failIfFalse(autoSemiColon(), "Expected a ';' following a targeted export declaration");
+ }
+ failIfFalse(result, "Cannot parse the declaration");
+
+ semanticFailIfFalse(exportName(m_vm->propertyNames->defaultKeyword), "Only one 'default' export is allowed");
+ m_moduleScopeData->exportBinding(*localName, m_vm->propertyNames->defaultKeyword);
+ return context.createExportDefaultDeclaration(exportLocation, result, *localName);
+ }
+
+ case OPENBRACE: {
+ // export ExportClause FromClause ;
+ // export ExportClause ;
+ //
+ // ExportClause :
+ // { }
+ // { ExportsList }
+ // { ExportsList , }
+ //
+ // ExportsList :
+ // ExportSpecifier
+ // ExportsList , ExportSpecifier
+
+ next();
+
+ auto specifierList = context.createExportSpecifierList();
+ Vector<std::pair<const Identifier*, const Identifier*>> maybeExportedLocalNames;
+
+ bool hasKeywordForLocalBindings = false;
+ while (!match(CLOSEBRACE)) {
+ failIfFalse(matchIdentifierOrKeyword(), "Expected a variable name for the export declaration");
+ auto specifier = parseExportSpecifier(context, maybeExportedLocalNames, hasKeywordForLocalBindings);
+ failIfFalse(specifier, "Cannot parse the named export");
+ context.appendExportSpecifier(specifierList, specifier);
+ if (!consume(COMMA))
+ break;
+ }
+ handleProductionOrFail2(CLOSEBRACE, "}", "end", "export list");
+
+ typename TreeBuilder::ModuleName moduleName = 0;
+ if (matchContextualKeyword(m_vm->propertyNames->from)) {
+ next();
+ moduleName = parseModuleName(context);
+ failIfFalse(moduleName, "Cannot parse the 'from' clause");
+ }
+ failIfFalse(autoSemiColon(), "Expected a ';' following a targeted export declaration");
+
+ if (!moduleName) {
+ semanticFailIfTrue(hasKeywordForLocalBindings, "Cannot use keyword as exported variable name");
+ // Since this export declaration does not have module specifier part, it exports the local bindings.
+ // While the export declaration with module specifier does not have any effect on the current module's scope,
+ // the export named declaration without module specifier references the the local binding names.
+ // For example,
+ // export { A, B, C as D } from "mod"
+ // does not have effect on the current module's scope. But,
+ // export { A, B, C as D }
+ // will reference the current module's bindings.
+ for (const auto& pair : maybeExportedLocalNames) {
+ const Identifier* localName = pair.first;
+ const Identifier* exportedName = pair.second;
+ m_moduleScopeData->exportBinding(*localName, *exportedName);
+ }
+ }
+
+ return context.createExportNamedDeclaration(exportLocation, specifierList, moduleName);
+ }
+
+ default: {
+ // export VariableStatement
+ // export Declaration
+ TreeStatement result = 0;
+ switch (m_token.m_type) {
+ case VAR:
+ result = parseVariableDeclaration(context, DeclarationType::VarDeclaration, ExportType::Exported);
+ break;
+
+ case CONSTTOKEN:
+ result = parseVariableDeclaration(context, DeclarationType::ConstDeclaration, ExportType::Exported);
+ break;
+
+ case LET:
+ result = parseVariableDeclaration(context, DeclarationType::LetDeclaration, ExportType::Exported);
+ break;
+
+ case FUNCTION: {
+ DepthManager statementDepth(&m_statementDepth);
+ m_statementDepth = 1;
+ result = parseFunctionDeclaration(context, ExportType::Exported);
+ break;
+ }
+
+ case CLASSTOKEN:
+ result = parseClassDeclaration(context, ExportType::Exported);
+ break;
+
+ case ASYNC:
+ next();
+ semanticFailIfFalse(match(FUNCTION) && !m_lexer->prevTerminator(), "Expected 'function' keyword following 'async' keyword with no preceding line terminator");
+ result = parseAsyncFunctionDeclaration(context, ExportType::Exported);
+ break;
+
+ default:
+ failWithMessage("Expected either a declaration or a variable statement");
+ break;
+ }
+
+ failIfFalse(result, "Cannot parse the declaration");
+ return context.createExportLocalDeclaration(exportLocation, result);
+ }
+ }
+
+ RELEASE_ASSERT_NOT_REACHED();
+ return 0;
+}
+
+template <typename LexerType>
template <class TreeBuilder> TreeExpression Parser<LexerType>::parseExpression(TreeBuilder& context)
{
failIfStackOverflow();
JSTokenLocation location(tokenLocation());
TreeExpression node = parseAssignmentExpression(context);
failIfFalse(node, "Cannot parse expression");
+ context.setEndOffset(node, m_lastTokenEndPosition.offset);
if (!match(COMMA))
return node;
next();
- m_nonTrivialExpressionCount++;
- m_nonLHSCount++;
+ m_parserState.nonTrivialExpressionCount++;
+ m_parserState.nonLHSCount++;
TreeExpression right = parseAssignmentExpression(context);
failIfFalse(right, "Cannot parse expression in a comma expression");
- typename TreeBuilder::Comma commaNode = context.createCommaExpr(location, node, right);
+ context.setEndOffset(right, m_lastTokenEndPosition.offset);
+ typename TreeBuilder::Comma head = context.createCommaExpr(location, node);
+ typename TreeBuilder::Comma tail = context.appendToCommaExpr(location, head, head, right);
while (match(COMMA)) {
next(TreeBuilder::DontBuildStrings);
right = parseAssignmentExpression(context);
failIfFalse(right, "Cannot parse expression in a comma expression");
- context.appendToComma(commaNode, right);
+ context.setEndOffset(right, m_lastTokenEndPosition.offset);
+ tail = context.appendToCommaExpr(location, head, tail, right);
}
- return commaNode;
+ context.setEndOffset(head, m_lastTokenEndPosition.offset);
+ return head;
+}
+
+template <typename LexerType>
+template <typename TreeBuilder> TreeExpression Parser<LexerType>::parseAssignmentExpressionOrPropagateErrorClass(TreeBuilder& context)
+{
+ ExpressionErrorClassifier classifier(this);
+ auto assignment = parseAssignmentExpression(context, classifier);
+ if (!assignment)
+ classifier.propagateExpressionErrorClass();
+ return assignment;
}
template <typename LexerType>
template <typename TreeBuilder> TreeExpression Parser<LexerType>::parseAssignmentExpression(TreeBuilder& context)
{
+ ExpressionErrorClassifier classifier(this);
+ return parseAssignmentExpression(context, classifier);
+}
+
+template <typename LexerType>
+template <typename TreeBuilder> TreeExpression Parser<LexerType>::parseAssignmentExpression(TreeBuilder& context, ExpressionErrorClassifier& classifier)
+{
+ ASSERT(!hasError());
+
failIfStackOverflow();
+
+ if (match(YIELD) && !isYIELDMaskedAsIDENT(currentScope()->isGenerator()))
+ return parseYieldExpression(context);
+
JSTextPosition start = tokenStartPosition();
JSTokenLocation location(tokenLocation());
- int initialAssignmentCount = m_assignmentCount;
- int initialNonLHSCount = m_nonLHSCount;
- if (match(OPENBRACE) || match(OPENBRACKET)) {
- SavePoint savePoint = createSavePoint();
- auto pattern = tryParseDeconstructionPatternExpression(context);
- if (pattern && consume(EQUAL)) {
- auto rhs = parseAssignmentExpression(context);
- if (rhs)
- return context.createDeconstructingAssignment(location, pattern, rhs);
+ int initialAssignmentCount = m_parserState.assignmentCount;
+ int initialNonLHSCount = m_parserState.nonLHSCount;
+ bool maybeAssignmentPattern = match(OPENBRACE) || match(OPENBRACKET);
+ bool wasOpenParen = match(OPENPAREN);
+ // Do not use matchSpecIdentifier() here since it is slower than isIdentifierOrKeyword.
+ // Whether spec identifier is will be validated by isArrowFunctionParameters().
+ bool wasIdentifierOrKeyword = isIdentifierOrKeyword(m_token);
+ bool maybeValidArrowFunctionStart = wasOpenParen || wasIdentifierOrKeyword;
+ SavePoint savePoint = createSavePoint();
+ size_t usedVariablesSize = 0;
+
+ if (wasOpenParen) {
+ usedVariablesSize = currentScope()->currentUsedVariablesSize();
+ currentScope()->pushUsedVariableSet();
+ }
+
+ TreeExpression lhs = parseConditionalExpression(context);
+
+ if (maybeValidArrowFunctionStart && !match(EOFTOK)) {
+ bool isArrowFunctionToken = match(ARROWFUNCTION);
+ if (!lhs || isArrowFunctionToken) {
+ SavePointWithError errorRestorationSavePoint = createSavePointForError();
+ restoreSavePoint(savePoint);
+ bool isAsyncArrow = false;
+ if (UNLIKELY(classifier.indicatesPossibleAsyncArrowFunction())) {
+ ASSERT(match(ASYNC));
+ next();
+ isAsyncArrow = !m_lexer->prevTerminator();
+ }
+ if (isArrowFunctionParameters()) {
+ if (wasOpenParen)
+ currentScope()->revertToPreviousUsedVariables(usedVariablesSize);
+ return parseArrowFunctionExpression(context, isAsyncArrow);
+ }
+ if (isArrowFunctionToken)
+ propagateError();
+ restoreSavePointWithError(errorRestorationSavePoint);
+ if (isArrowFunctionToken)
+ failDueToUnexpectedToken();
}
+ }
+
+ if (!lhs && (!maybeAssignmentPattern || !classifier.indicatesPossiblePattern()))
+ propagateError();
+
+ if (maybeAssignmentPattern && (!lhs || (context.isObjectOrArrayLiteral(lhs) && match(EQUAL)))) {
+ SavePointWithError expressionErrorLocation = createSavePointForError();
restoreSavePoint(savePoint);
+ auto pattern = tryParseDestructuringPatternExpression(context, AssignmentContext::AssignmentExpression);
+ if (classifier.indicatesPossiblePattern() && (!pattern || !match(EQUAL))) {
+ restoreSavePointWithError(expressionErrorLocation);
+ return 0;
+ }
+ failIfFalse(pattern, "Cannot parse assignment pattern");
+ consumeOrFail(EQUAL, "Expected '=' following assignment pattern");
+ auto rhs = parseAssignmentExpression(context);
+ if (!rhs)
+ propagateError();
+ return context.createDestructuringAssignment(location, pattern, rhs);
}
- TreeExpression lhs = parseConditionalExpression(context);
+
failIfFalse(lhs, "Cannot parse expression");
- if (initialNonLHSCount != m_nonLHSCount) {
+ if (initialNonLHSCount != m_parserState.nonLHSCount) {
if (m_token.m_type >= EQUAL && m_token.m_type <= OREQUAL)
semanticFail("Left hand side of operator '", getToken(), "' must be a reference");
@@ -1559,24 +3459,26 @@ template <typename TreeBuilder> TreeExpression Parser<LexerType>::parseAssignmen
case XOREQUAL: op = OpXOrEq; break;
case OREQUAL: op = OpOrEq; break;
case MODEQUAL: op = OpModEq; break;
+ case POWEQUAL: op = OpPowEq; break;
default:
goto end;
}
- m_nonTrivialExpressionCount++;
+ m_parserState.nonTrivialExpressionCount++;
hadAssignment = true;
- context.assignmentStackAppend(assignmentStack, lhs, start, tokenStartPosition(), m_assignmentCount, op);
+ if (UNLIKELY(context.isNewTarget(lhs)))
+ internalFailWithMessage(false, "new.target can't be the left hand side of an assignment expression");
+ context.assignmentStackAppend(assignmentStack, lhs, start, tokenStartPosition(), m_parserState.assignmentCount, op);
start = tokenStartPosition();
- m_assignmentCount++;
+ m_parserState.assignmentCount++;
next(TreeBuilder::DontBuildStrings);
- if (strictMode() && m_lastIdentifier && context.isResolve(lhs)) {
- failIfTrueIfStrict(m_vm->propertyNames->eval == *m_lastIdentifier, "Cannot modify 'eval' in strict mode");
- failIfTrueIfStrict(m_vm->propertyNames->arguments == *m_lastIdentifier, "Cannot modify 'arguments' in strict mode");
- declareWrite(m_lastIdentifier);
- m_lastIdentifier = 0;
+ if (strictMode() && m_parserState.lastIdentifier && context.isResolve(lhs)) {
+ failIfTrueIfStrict(m_vm->propertyNames->eval == *m_parserState.lastIdentifier, "Cannot modify 'eval' in strict mode");
+ failIfTrueIfStrict(m_vm->propertyNames->arguments == *m_parserState.lastIdentifier, "Cannot modify 'arguments' in strict mode");
+ m_parserState.lastIdentifier = 0;
}
lhs = parseAssignmentExpression(context);
failIfFalse(lhs, "Cannot parse the right hand side of an assignment expression");
- if (initialNonLHSCount != m_nonLHSCount) {
+ if (initialNonLHSCount != m_parserState.nonLHSCount) {
if (m_token.m_type >= EQUAL && m_token.m_type <= OREQUAL)
semanticFail("Left hand side of operator '", getToken(), "' must be a reference");
break;
@@ -1584,18 +3486,66 @@ template <typename TreeBuilder> TreeExpression Parser<LexerType>::parseAssignmen
}
end:
if (hadAssignment)
- m_nonLHSCount++;
+ m_parserState.nonLHSCount++;
if (!TreeBuilder::CreatesAST)
return lhs;
while (assignmentStack)
- lhs = context.createAssignment(location, assignmentStack, lhs, initialAssignmentCount, m_assignmentCount, lastTokenEndPosition());
+ lhs = context.createAssignment(location, assignmentStack, lhs, initialAssignmentCount, m_parserState.assignmentCount, lastTokenEndPosition());
return lhs;
}
template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseYieldExpression(TreeBuilder& context)
+{
+ // YieldExpression[In] :
+ // yield
+ // yield [no LineTerminator here] AssignmentExpression[?In, Yield]
+ // yield [no LineTerminator here] * AssignmentExpression[?In, Yield]
+
+ // http://ecma-international.org/ecma-262/6.0/#sec-generator-function-definitions
+ failIfFalse(currentScope()->isGenerator() && !currentScope()->isArrowFunctionBoundary(), "Cannot use yield expression out of generator");
+
+ // http://ecma-international.org/ecma-262/6.0/#sec-generator-function-definitions-static-semantics-early-errors
+ failIfTrue(m_parserState.functionParsePhase == FunctionParsePhase::Parameters, "Cannot use yield expression within parameters");
+
+ JSTokenLocation location(tokenLocation());
+ JSTextPosition divotStart = tokenStartPosition();
+ ASSERT(match(YIELD));
+ SavePoint savePoint = createSavePoint();
+ next();
+ if (m_lexer->prevTerminator())
+ return context.createYield(location);
+
+ bool delegate = consume(TIMES);
+ JSTextPosition argumentStart = tokenStartPosition();
+ TreeExpression argument = parseAssignmentExpression(context);
+ if (!argument) {
+ restoreSavePoint(savePoint);
+ next();
+ return context.createYield(location);
+ }
+ return context.createYield(location, argument, delegate, divotStart, argumentStart, lastTokenEndPosition());
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseAwaitExpression(TreeBuilder& context)
+{
+ ASSERT(match(AWAIT));
+ ASSERT(currentScope()->isAsyncFunction());
+ failIfTrue(m_parserState.functionParsePhase == FunctionParsePhase::Parameters, "Cannot use await expression within parameters");
+ JSTokenLocation location(tokenLocation());
+ JSTextPosition divotStart = tokenStartPosition();
+ next();
+ JSTextPosition argumentStart = tokenStartPosition();
+ TreeExpression argument = parseUnaryExpression(context);
+ failIfFalse(argument, "Failed to parse await expression");
+ return context.createAwait(location, argument, divotStart, argumentStart, lastTokenEndPosition());
+}
+
+template <typename LexerType>
template <class TreeBuilder> TreeExpression Parser<LexerType>::parseConditionalExpression(TreeBuilder& context)
{
JSTokenLocation location(tokenLocation());
@@ -1603,21 +3553,25 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseConditionalE
failIfFalse(cond, "Cannot parse expression");
if (!match(QUESTION))
return cond;
- m_nonTrivialExpressionCount++;
- m_nonLHSCount++;
+ m_parserState.nonTrivialExpressionCount++;
+ m_parserState.nonLHSCount++;
next(TreeBuilder::DontBuildStrings);
TreeExpression lhs = parseAssignmentExpression(context);
failIfFalse(lhs, "Cannot parse left hand side of ternary operator");
+ context.setEndOffset(lhs, m_lastTokenEndPosition.offset);
consumeOrFailWithFlags(COLON, TreeBuilder::DontBuildStrings, "Expected ':' in ternary operator");
TreeExpression rhs = parseAssignmentExpression(context);
failIfFalse(rhs, "Cannot parse right hand side of ternary operator");
+ context.setEndOffset(rhs, m_lastTokenEndPosition.offset);
return context.createConditionalExpr(location, cond, lhs, rhs);
}
-ALWAYS_INLINE static bool isUnaryOp(JSTokenType token)
+ALWAYS_INLINE static bool isUnaryOpExcludingUpdateOp(JSTokenType token)
{
- return token & UnaryOpTokenFlag;
+ if (isUpdateOp(token))
+ return false;
+ return isUnaryOp(token);
}
template <typename LexerType>
@@ -1631,27 +3585,51 @@ int Parser<LexerType>::isBinaryOperator(JSTokenType token)
template <typename LexerType>
template <class TreeBuilder> TreeExpression Parser<LexerType>::parseBinaryExpression(TreeBuilder& context)
{
-
int operandStackDepth = 0;
int operatorStackDepth = 0;
typename TreeBuilder::BinaryExprContext binaryExprContext(context);
JSTokenLocation location(tokenLocation());
while (true) {
JSTextPosition exprStart = tokenStartPosition();
- int initialAssignments = m_assignmentCount;
+ int initialAssignments = m_parserState.assignmentCount;
+ JSTokenType leadingTokenTypeForUnaryExpression = m_token.m_type;
TreeExpression current = parseUnaryExpression(context);
failIfFalse(current, "Cannot parse expression");
- context.appendBinaryExpressionInfo(operandStackDepth, current, exprStart, lastTokenEndPosition(), lastTokenEndPosition(), initialAssignments != m_assignmentCount);
+ context.appendBinaryExpressionInfo(operandStackDepth, current, exprStart, lastTokenEndPosition(), lastTokenEndPosition(), initialAssignments != m_parserState.assignmentCount);
+
+ // 12.6 https://tc39.github.io/ecma262/#sec-exp-operator
+ // ExponentiationExpresion is described as follows.
+ //
+ // ExponentiationExpression[Yield]:
+ // UnaryExpression[?Yield]
+ // UpdateExpression[?Yield] ** ExponentiationExpression[?Yield]
+ //
+ // As we can see, the left hand side of the ExponentiationExpression is UpdateExpression, not UnaryExpression.
+ // So placing UnaryExpression not included in UpdateExpression here is a syntax error.
+ // This is intentional. For example, if UnaryExpression is allowed, we can have the code like `-x**y`.
+ // But this is confusing: `-(x**y)` OR `(-x)**y`, which interpretation is correct?
+ // To avoid this problem, ECMA262 makes unparenthesized exponentiation expression as operand of unary operators an early error.
+ // More rationale: https://mail.mozilla.org/pipermail/es-discuss/2015-September/044232.html
+ //
+ // Here, we guarantee that the left hand side of this expression is not unary expression by checking the leading operator of the parseUnaryExpression.
+ // This check just works. Let's consider the example,
+ // y <> -x ** z
+ // ^
+ // Check this.
+ // If the binary operator <> has higher precedence than one of "**", this check does not work.
+ // But it's OK for ** because the operator "**" has the highest operator precedence in the binary operators.
+ failIfTrue(match(POW) && isUnaryOpExcludingUpdateOp(leadingTokenTypeForUnaryExpression), "Amiguous unary expression in the left hand side of the exponentiation expression; parenthesis must be used to disambiguate the expression");
+
int precedence = isBinaryOperator(m_token.m_type);
if (!precedence)
break;
- m_nonTrivialExpressionCount++;
- m_nonLHSCount++;
+ m_parserState.nonTrivialExpressionCount++;
+ m_parserState.nonLHSCount++;
int operatorToken = m_token.m_type;
next(TreeBuilder::DontBuildStrings);
- while (operatorStackDepth && context.operatorStackHasHigherPrecedence(operatorStackDepth, precedence)) {
+ while (operatorStackDepth && context.operatorStackShouldReduce(precedence)) {
ASSERT(operandStackDepth > 1);
typename TreeBuilder::BinaryOperand rhs = context.getFromOperandStack(-1);
@@ -1678,94 +3656,230 @@ template <typename LexerType>
template <class TreeBuilder> TreeProperty Parser<LexerType>::parseProperty(TreeBuilder& context, bool complete)
{
bool wasIdent = false;
+ bool isAsync = false;
+ bool isGenerator = false;
+ bool isClassProperty = false;
+ bool isAsyncMethod = false;
+ if (consume(TIMES))
+ isGenerator = true;
+
+parseProperty:
switch (m_token.m_type) {
- namedProperty:
+ case ASYNC:
+ isAsync = !isGenerator && !isAsyncMethod;
+ FALLTHROUGH;
case IDENT:
+ case AWAIT:
wasIdent = true;
FALLTHROUGH;
case STRING: {
+namedProperty:
const Identifier* ident = m_token.m_data.ident;
- if (complete || (wasIdent && (*ident == m_vm->propertyNames->get || *ident == m_vm->propertyNames->set)))
+ unsigned getterOrSetterStartOffset = tokenStart();
+
+ if (complete || (wasIdent && !isGenerator && (*ident == m_vm->propertyNames->get || *ident == m_vm->propertyNames->set)) || isAsync)
nextExpectIdentifier(LexerFlagsIgnoreReservedWords);
else
nextExpectIdentifier(LexerFlagsIgnoreReservedWords | TreeBuilder::DontBuildKeywords);
-
- if (match(COLON)) {
+
+ if (!isGenerator && !isAsyncMethod && match(COLON)) {
next();
- TreeExpression node = parseAssignmentExpression(context);
+ TreeExpression node = parseAssignmentExpressionOrPropagateErrorClass(context);
failIfFalse(node, "Cannot parse expression for property declaration");
- return context.createProperty(ident, node, PropertyNode::Constant, complete);
+ context.setEndOffset(node, m_lexer->currentOffset());
+ return context.createProperty(ident, node, PropertyNode::Constant, PropertyNode::Unknown, complete, SuperBinding::NotNeeded, isClassProperty);
+ }
+
+ if (match(OPENPAREN)) {
+ auto method = parsePropertyMethod(context, ident, isGenerator, isAsyncMethod);
+ propagateError();
+ return context.createProperty(ident, method, PropertyNode::Constant, PropertyNode::KnownDirect, complete, SuperBinding::Needed, isClassProperty);
}
+ failIfTrue(isGenerator || isAsyncMethod, "Expected a parenthesis for argument list");
+
failIfFalse(wasIdent, "Expected an identifier as property name");
- const Identifier* accessorName = 0;
- TreeFormalParameterList parameters = 0;
- TreeFunctionBody body = 0;
- unsigned openBraceOffset = 0;
- unsigned closeBraceOffset = 0;
- int bodyStartLine = 0;
- unsigned bodyStartColumn = 0;
+
+ if (match(COMMA) || match(CLOSEBRACE)) {
+ JSTextPosition start = tokenStartPosition();
+ JSTokenLocation location(tokenLocation());
+ currentScope()->useVariable(ident, m_vm->propertyNames->eval == *ident);
+ if (currentScope()->isArrowFunction())
+ currentScope()->setInnerArrowFunctionUsesEval();
+ TreeExpression node = context.createResolve(location, *ident, start, lastTokenEndPosition());
+ return context.createProperty(ident, node, static_cast<PropertyNode::Type>(PropertyNode::Constant | PropertyNode::Shorthand), PropertyNode::KnownDirect, complete, SuperBinding::NotNeeded, isClassProperty);
+ }
+
+ if (match(EQUAL)) // CoverInitializedName is exclusive to BindingPattern and AssignmentPattern
+ classifyExpressionError(ErrorIndicatesPattern);
+
PropertyNode::Type type;
if (*ident == m_vm->propertyNames->get)
type = PropertyNode::Getter;
else if (*ident == m_vm->propertyNames->set)
type = PropertyNode::Setter;
- else
+ else if (UNLIKELY(isAsync && !isAsyncMethod)) {
+ isAsyncMethod = true;
+ failIfTrue(m_lexer->prevTerminator(), "Expected a property name following keyword 'async'");
+ goto parseProperty;
+ } else
failWithMessage("Expected a ':' following the property name '", ident->impl(), "'");
- const Identifier* stringPropertyName = 0;
- double numericPropertyName = 0;
- if (m_token.m_type == IDENT || m_token.m_type == STRING)
- stringPropertyName = m_token.m_data.ident;
- else if (m_token.m_type == NUMBER)
- numericPropertyName = m_token.m_data.doubleValue;
- else
- failDueToUnexpectedToken();
- JSTokenLocation location(tokenLocation());
- next();
- if (type == PropertyNode::Getter) {
- failIfFalse(match(OPENPAREN), "Expected a parameter list for getter definition");
- failIfFalse((parseFunctionInfo(context, FunctionNoRequirements, GetterMode, false, accessorName, parameters, body, openBraceOffset, closeBraceOffset, bodyStartLine, bodyStartColumn)), "Cannot parse getter definition");
- } else {
- failIfFalse(match(OPENPAREN), "Expected a parameter list for setter definition");
- failIfFalse((parseFunctionInfo(context, FunctionNoRequirements, SetterMode, false, accessorName, parameters, body, openBraceOffset, closeBraceOffset, bodyStartLine, bodyStartColumn)), "Cannot parse setter definition");
- }
- if (stringPropertyName)
- return context.createGetterOrSetterProperty(location, type, complete, stringPropertyName, parameters, body, openBraceOffset, closeBraceOffset, bodyStartLine, m_lastTokenEndPosition.line, bodyStartColumn);
- return context.createGetterOrSetterProperty(const_cast<VM*>(m_vm), location, type, complete, numericPropertyName, parameters, body, openBraceOffset, closeBraceOffset, bodyStartLine, m_lastTokenEndPosition.line, bodyStartColumn);
+ return parseGetterSetter(context, complete, type, getterOrSetterStartOffset, ConstructorKind::None, isClassProperty);
}
- case NUMBER: {
+ case DOUBLE:
+ case INTEGER: {
double propertyName = m_token.m_data.doubleValue;
next();
+
+ if (match(OPENPAREN)) {
+ const Identifier& ident = m_parserArena.identifierArena().makeNumericIdentifier(const_cast<VM*>(m_vm), propertyName);
+ auto method = parsePropertyMethod(context, &ident, isGenerator, isAsyncMethod);
+ propagateError();
+ return context.createProperty(&ident, method, PropertyNode::Constant, PropertyNode::Unknown, complete, SuperBinding::Needed, isClassProperty);
+ }
+ failIfTrue(isGenerator || isAsyncMethod, "Expected a parenthesis for argument list");
+
consumeOrFail(COLON, "Expected ':' after property name");
TreeExpression node = parseAssignmentExpression(context);
failIfFalse(node, "Cannot parse expression for property declaration");
- return context.createProperty(const_cast<VM*>(m_vm), propertyName, node, PropertyNode::Constant, complete);
+ context.setEndOffset(node, m_lexer->currentOffset());
+ return context.createProperty(const_cast<VM*>(m_vm), m_parserArena, propertyName, node, PropertyNode::Constant, PropertyNode::Unknown, complete, SuperBinding::NotNeeded, isClassProperty);
}
case OPENBRACKET: {
next();
- auto propertyName = parseExpression(context);
+ auto propertyName = parseAssignmentExpression(context);
failIfFalse(propertyName, "Cannot parse computed property name");
-
handleProductionOrFail(CLOSEBRACKET, "]", "end", "computed property name");
+
+ if (match(OPENPAREN)) {
+ auto method = parsePropertyMethod(context, &m_vm->propertyNames->nullIdentifier, isGenerator, isAsyncMethod);
+ propagateError();
+ return context.createProperty(propertyName, method, static_cast<PropertyNode::Type>(PropertyNode::Constant | PropertyNode::Computed), PropertyNode::KnownDirect, complete, SuperBinding::Needed, isClassProperty);
+ }
+ failIfTrue(isGenerator || isAsyncMethod, "Expected a parenthesis for argument list");
+
consumeOrFail(COLON, "Expected ':' after property name");
TreeExpression node = parseAssignmentExpression(context);
failIfFalse(node, "Cannot parse expression for property declaration");
- return context.createProperty(const_cast<VM*>(m_vm), propertyName, node, PropertyNode::Constant, complete);
+ context.setEndOffset(node, m_lexer->currentOffset());
+ return context.createProperty(propertyName, node, static_cast<PropertyNode::Type>(PropertyNode::Constant | PropertyNode::Computed), PropertyNode::Unknown, complete, SuperBinding::NotNeeded, isClassProperty);
}
default:
failIfFalse(m_token.m_type & KeywordTokenFlag, "Expected a property name");
+ wasIdent = true; // Treat keyword token as an identifier
goto namedProperty;
}
}
template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parsePropertyMethod(TreeBuilder& context, const Identifier* methodName, bool isGenerator, bool isAsyncMethod)
+{
+ JSTokenLocation methodLocation(tokenLocation());
+ unsigned methodStart = tokenStart();
+ ParserFunctionInfo<TreeBuilder> methodInfo;
+ methodInfo.name = methodName;
+ SourceParseMode parseMode = isGenerator ? SourceParseMode::GeneratorWrapperFunctionMode : isAsyncMethod ? SourceParseMode::AsyncMethodMode : SourceParseMode::MethodMode;
+ failIfFalse((parseFunctionInfo(context, FunctionNameRequirements::Unnamed, parseMode, false, ConstructorKind::None, SuperBinding::Needed, methodStart, methodInfo, FunctionDefinitionType::Method)), "Cannot parse this method");
+ return context.createMethodDefinition(methodLocation, methodInfo);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeProperty Parser<LexerType>::parseGetterSetter(TreeBuilder& context, bool strict, PropertyNode::Type type, unsigned getterOrSetterStartOffset,
+ ConstructorKind constructorKind, bool isClassProperty)
+{
+ const Identifier* stringPropertyName = 0;
+ double numericPropertyName = 0;
+ TreeExpression computedPropertyName = 0;
+
+ JSTokenLocation location(tokenLocation());
+
+ if (matchSpecIdentifier() || match(STRING) || m_token.m_type & KeywordTokenFlag) {
+ stringPropertyName = m_token.m_data.ident;
+ semanticFailIfTrue(isClassProperty && *stringPropertyName == m_vm->propertyNames->prototype,
+ "Cannot declare a static method named 'prototype'");
+ semanticFailIfTrue(isClassProperty && *stringPropertyName == m_vm->propertyNames->constructor,
+ "Cannot declare a getter or setter named 'constructor'");
+ next();
+ } else if (match(DOUBLE) || match(INTEGER)) {
+ numericPropertyName = m_token.m_data.doubleValue;
+ next();
+ } else if (match(OPENBRACKET)) {
+ next();
+ computedPropertyName = parseAssignmentExpression(context);
+ failIfFalse(computedPropertyName, "Cannot parse computed property name");
+ handleProductionOrFail(CLOSEBRACKET, "]", "end", "computed property name");
+ } else
+ failDueToUnexpectedToken();
+
+ ParserFunctionInfo<TreeBuilder> info;
+ if (type & PropertyNode::Getter) {
+ failIfFalse(match(OPENPAREN), "Expected a parameter list for getter definition");
+ failIfFalse((parseFunctionInfo(context, FunctionNameRequirements::Unnamed, SourceParseMode::GetterMode, false, constructorKind, SuperBinding::Needed, getterOrSetterStartOffset, info, FunctionDefinitionType::Method)), "Cannot parse getter definition");
+ } else {
+ failIfFalse(match(OPENPAREN), "Expected a parameter list for setter definition");
+ failIfFalse((parseFunctionInfo(context, FunctionNameRequirements::Unnamed, SourceParseMode::SetterMode, false, constructorKind, SuperBinding::Needed, getterOrSetterStartOffset, info, FunctionDefinitionType::Method)), "Cannot parse setter definition");
+ }
+
+ if (stringPropertyName)
+ return context.createGetterOrSetterProperty(location, type, strict, stringPropertyName, info, isClassProperty);
+
+ if (computedPropertyName)
+ return context.createGetterOrSetterProperty(location, static_cast<PropertyNode::Type>(type | PropertyNode::Computed), strict, computedPropertyName, info, isClassProperty);
+
+ return context.createGetterOrSetterProperty(const_cast<VM*>(m_vm), m_parserArena, location, type, strict, numericPropertyName, info, isClassProperty);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> bool Parser<LexerType>::shouldCheckPropertyForUnderscoreProtoDuplicate(TreeBuilder& context, const TreeProperty& property)
+{
+ if (m_syntaxAlreadyValidated)
+ return false;
+
+ if (!context.getName(property))
+ return false;
+
+ // A Constant property that is not a Computed or Shorthand Constant property.
+ return context.getType(property) == PropertyNode::Constant;
+}
+
+template <typename LexerType>
+void Parser<LexerType>::recordPauseLocation(const JSTextPosition& position)
+{
+ if (LIKELY(!m_debuggerParseData))
+ return;
+
+ if (position.line < 0)
+ return;
+
+ m_debuggerParseData->pausePositions.appendPause(position);
+}
+
+template <typename LexerType>
+void Parser<LexerType>::recordFunctionEntryLocation(const JSTextPosition& position)
+{
+ if (LIKELY(!m_debuggerParseData))
+ return;
+
+ m_debuggerParseData->pausePositions.appendEntry(position);
+}
+
+template <typename LexerType>
+void Parser<LexerType>::recordFunctionLeaveLocation(const JSTextPosition& position)
+{
+ if (LIKELY(!m_debuggerParseData))
+ return;
+
+ m_debuggerParseData->pausePositions.appendLeave(position);
+}
+
+template <typename LexerType>
template <class TreeBuilder> TreeExpression Parser<LexerType>::parseObjectLiteral(TreeBuilder& context)
{
- auto savePoint = createSavePoint();
+ SavePoint savePoint = createSavePoint();
consumeOrFailWithFlags(OPENBRACE, TreeBuilder::DontBuildStrings, "Expected opening '{' at the start of an object literal");
- JSTokenLocation location(tokenLocation());
- int oldNonLHSCount = m_nonLHSCount;
-
+ int oldNonLHSCount = m_parserState.nonLHSCount;
+
+ JSTokenLocation location(tokenLocation());
if (match(CLOSEBRACE)) {
next();
return context.createObjectLiteral(location);
@@ -1773,31 +3887,42 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseObjectLitera
TreeProperty property = parseProperty(context, false);
failIfFalse(property, "Cannot parse object literal property");
- if (!m_syntaxAlreadyValidated && context.getType(property) != PropertyNode::Constant) {
+
+ if (!m_syntaxAlreadyValidated && context.getType(property) & (PropertyNode::Getter | PropertyNode::Setter)) {
restoreSavePoint(savePoint);
return parseStrictObjectLiteral(context);
}
+
+ bool seenUnderscoreProto = false;
+ if (shouldCheckPropertyForUnderscoreProtoDuplicate(context, property))
+ seenUnderscoreProto = *context.getName(property) == m_vm->propertyNames->underscoreProto;
+
TreePropertyList propertyList = context.createPropertyList(location, property);
TreePropertyList tail = propertyList;
while (match(COMMA)) {
next(TreeBuilder::DontBuildStrings);
- // allow extra comma, see http://bugs.webkit.org/show_bug.cgi?id=5939
if (match(CLOSEBRACE))
break;
JSTokenLocation propertyLocation(tokenLocation());
property = parseProperty(context, false);
failIfFalse(property, "Cannot parse object literal property");
- if (!m_syntaxAlreadyValidated && context.getType(property) != PropertyNode::Constant) {
+ if (!m_syntaxAlreadyValidated && context.getType(property) & (PropertyNode::Getter | PropertyNode::Setter)) {
restoreSavePoint(savePoint);
return parseStrictObjectLiteral(context);
}
+ if (shouldCheckPropertyForUnderscoreProtoDuplicate(context, property)) {
+ if (*context.getName(property) == m_vm->propertyNames->underscoreProto) {
+ semanticFailIfTrue(seenUnderscoreProto, "Attempted to redefine __proto__ property");
+ seenUnderscoreProto = true;
+ }
+ }
tail = context.createPropertyList(propertyLocation, property, tail);
}
location = tokenLocation();
- handleProductionOrFail(CLOSEBRACE, "}", "end", "object literal");
+ handleProductionOrFail2(CLOSEBRACE, "}", "end", "object literal");
- m_nonLHSCount = oldNonLHSCount;
+ m_parserState.nonLHSCount = oldNonLHSCount;
return context.createObjectLiteral(location, propertyList);
}
@@ -1807,7 +3932,7 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseStrictObject
{
consumeOrFail(OPENBRACE, "Expected opening '{' at the start of an object literal");
- int oldNonLHSCount = m_nonLHSCount;
+ int oldNonLHSCount = m_parserState.nonLHSCount;
JSTokenLocation location(tokenLocation());
if (match(CLOSEBRACE)) {
@@ -1817,39 +3942,33 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseStrictObject
TreeProperty property = parseProperty(context, true);
failIfFalse(property, "Cannot parse object literal property");
-
- typedef HashMap<RefPtr<StringImpl>, unsigned, IdentifierRepHash> ObjectValidationMap;
- ObjectValidationMap objectValidator;
- // Add the first property
- if (!m_syntaxAlreadyValidated && context.getName(property))
- objectValidator.add(context.getName(property)->impl(), context.getType(property));
-
+
+ bool seenUnderscoreProto = false;
+ if (shouldCheckPropertyForUnderscoreProtoDuplicate(context, property))
+ seenUnderscoreProto = *context.getName(property) == m_vm->propertyNames->underscoreProto;
+
TreePropertyList propertyList = context.createPropertyList(location, property);
TreePropertyList tail = propertyList;
while (match(COMMA)) {
next();
- // allow extra comma, see http://bugs.webkit.org/show_bug.cgi?id=5939
if (match(CLOSEBRACE))
break;
JSTokenLocation propertyLocation(tokenLocation());
property = parseProperty(context, true);
failIfFalse(property, "Cannot parse object literal property");
- if (!m_syntaxAlreadyValidated && context.getName(property)) {
- ObjectValidationMap::AddResult propertyEntry = objectValidator.add(context.getName(property)->impl(), context.getType(property));
- if (!propertyEntry.isNewEntry) {
- semanticFailIfTrue(propertyEntry.iterator->value == PropertyNode::Constant, "Attempted to redefine property '", propertyEntry.iterator->key.get(), "'");
- semanticFailIfTrue(context.getType(property) == PropertyNode::Constant, "Attempted to redefine property '", propertyEntry.iterator->key.get(), "'");
- semanticFailIfTrue(context.getType(property) & propertyEntry.iterator->value, "Attempted to redefine property '", propertyEntry.iterator->key.get(), "'");
- propertyEntry.iterator->value |= context.getType(property);
+ if (shouldCheckPropertyForUnderscoreProtoDuplicate(context, property)) {
+ if (*context.getName(property) == m_vm->propertyNames->underscoreProto) {
+ semanticFailIfTrue(seenUnderscoreProto, "Attempted to redefine __proto__ property");
+ seenUnderscoreProto = true;
}
}
tail = context.createPropertyList(propertyLocation, property, tail);
}
location = tokenLocation();
- handleProductionOrFail(CLOSEBRACE, "}", "end", "object literal");
+ handleProductionOrFail2(CLOSEBRACE, "}", "end", "object literal");
- m_nonLHSCount = oldNonLHSCount;
+ m_parserState.nonLHSCount = oldNonLHSCount;
return context.createObjectLiteral(location, propertyList);
}
@@ -1859,7 +3978,7 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseArrayLiteral
{
consumeOrFailWithFlags(OPENBRACKET, TreeBuilder::DontBuildStrings, "Expected an opening '[' at the beginning of an array literal");
- int oldNonLHSCount = m_nonLHSCount;
+ int oldNonLHSCount = m_parserState.nonLHSCount;
int elisions = 0;
while (match(COMMA)) {
@@ -1878,11 +3997,11 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseArrayLiteral
auto start = m_token.m_startPosition;
auto divot = m_token.m_endPosition;
next();
- auto spreadExpr = parseAssignmentExpression(context);
+ auto spreadExpr = parseAssignmentExpressionOrPropagateErrorClass(context);
failIfFalse(spreadExpr, "Cannot parse subject of a spread operation");
elem = context.createSpreadExpression(spreadLocation, spreadExpr, start, divot, m_lastTokenEndPosition);
} else
- elem = parseAssignmentExpression(context);
+ elem = parseAssignmentExpressionOrPropagateErrorClass(context);
failIfFalse(elem, "Cannot parse array literal element");
typename TreeBuilder::ElementList elementList = context.createElementList(elisions, elem);
typename TreeBuilder::ElementList tail = elementList;
@@ -1906,13 +4025,13 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseArrayLiteral
auto start = m_token.m_startPosition;
auto divot = m_token.m_endPosition;
next();
- TreeExpression elem = parseAssignmentExpression(context);
+ TreeExpression elem = parseAssignmentExpressionOrPropagateErrorClass(context);
failIfFalse(elem, "Cannot parse subject of a spread operation");
auto spread = context.createSpreadExpression(spreadLocation, elem, start, divot, m_lastTokenEndPosition);
tail = context.createElementList(tail, elisions, spread);
continue;
}
- TreeExpression elem = parseAssignmentExpression(context);
+ TreeExpression elem = parseAssignmentExpressionOrPropagateErrorClass(context);
failIfFalse(elem, "Cannot parse array literal element");
tail = context.createElementList(tail, elisions, elem);
}
@@ -1923,16 +4042,128 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseArrayLiteral
semanticFail("The '...' operator should come before a target expression");
}
- m_nonLHSCount = oldNonLHSCount;
+ m_parserState.nonLHSCount = oldNonLHSCount;
return context.createArray(location, elementList);
}
template <typename LexerType>
+template <class TreeBuilder> TreeClassExpression Parser<LexerType>::parseClassExpression(TreeBuilder& context)
+{
+ ASSERT(match(CLASSTOKEN));
+ ParserClassInfo<TreeBuilder> info;
+ info.className = &m_vm->propertyNames->nullIdentifier;
+ return parseClass(context, FunctionNameRequirements::None, info);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseFunctionExpression(TreeBuilder& context)
+{
+ ASSERT(match(FUNCTION));
+ JSTokenLocation location(tokenLocation());
+ unsigned functionKeywordStart = tokenStart();
+ next();
+ ParserFunctionInfo<TreeBuilder> functionInfo;
+ functionInfo.name = &m_vm->propertyNames->nullIdentifier;
+ SourceParseMode parseMode = SourceParseMode::NormalFunctionMode;
+ if (consume(TIMES))
+ parseMode = SourceParseMode::GeneratorWrapperFunctionMode;
+ failIfFalse((parseFunctionInfo(context, FunctionNameRequirements::None, parseMode, false, ConstructorKind::None, SuperBinding::NotNeeded, functionKeywordStart, functionInfo, FunctionDefinitionType::Expression)), "Cannot parse function expression");
+ return context.createFunctionExpr(location, functionInfo);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseAsyncFunctionExpression(TreeBuilder& context)
+{
+ ASSERT(match(FUNCTION));
+ JSTokenLocation location(tokenLocation());
+ unsigned functionKeywordStart = tokenStart();
+ next();
+ ParserFunctionInfo<TreeBuilder> functionInfo;
+ functionInfo.name = &m_vm->propertyNames->nullIdentifier;
+ SourceParseMode parseMode = SourceParseMode::AsyncFunctionMode;
+ failIfFalse(parseFunctionInfo(context, FunctionNameRequirements::None, parseMode, false, ConstructorKind::None, SuperBinding::NotNeeded, functionKeywordStart, functionInfo, FunctionDefinitionType::Expression), "Cannot parse async function expression");
+ return context.createFunctionExpr(location, functionInfo);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> typename TreeBuilder::TemplateString Parser<LexerType>::parseTemplateString(TreeBuilder& context, bool isTemplateHead, typename LexerType::RawStringsBuildMode rawStringsBuildMode, bool& elementIsTail)
+{
+ if (isTemplateHead)
+ ASSERT(match(BACKQUOTE));
+ else
+ matchOrFail(CLOSEBRACE, "Expected a closing '}' following an expression in template literal");
+
+ // Re-scan the token to recognize it as Template Element.
+ m_token.m_type = m_lexer->scanTemplateString(&m_token, rawStringsBuildMode);
+ matchOrFail(TEMPLATE, "Expected an template element");
+ const Identifier* cooked = m_token.m_data.cooked;
+ const Identifier* raw = m_token.m_data.raw;
+ elementIsTail = m_token.m_data.isTail;
+ JSTokenLocation location(tokenLocation());
+ next();
+ return context.createTemplateString(location, cooked, raw);
+}
+
+template <typename LexerType>
+template <class TreeBuilder> typename TreeBuilder::TemplateLiteral Parser<LexerType>::parseTemplateLiteral(TreeBuilder& context, typename LexerType::RawStringsBuildMode rawStringsBuildMode)
+{
+ ASSERT(match(BACKQUOTE));
+ JSTokenLocation location(tokenLocation());
+ bool elementIsTail = false;
+
+ auto headTemplateString = parseTemplateString(context, true, rawStringsBuildMode, elementIsTail);
+ failIfFalse(headTemplateString, "Cannot parse head template element");
+
+ typename TreeBuilder::TemplateStringList templateStringList = context.createTemplateStringList(headTemplateString);
+ typename TreeBuilder::TemplateStringList templateStringTail = templateStringList;
+
+ if (elementIsTail)
+ return context.createTemplateLiteral(location, templateStringList);
+
+ failIfTrue(match(CLOSEBRACE), "Template literal expression cannot be empty");
+ TreeExpression expression = parseExpression(context);
+ failIfFalse(expression, "Cannot parse expression in template literal");
+
+ typename TreeBuilder::TemplateExpressionList templateExpressionList = context.createTemplateExpressionList(expression);
+ typename TreeBuilder::TemplateExpressionList templateExpressionTail = templateExpressionList;
+
+ auto templateString = parseTemplateString(context, false, rawStringsBuildMode, elementIsTail);
+ failIfFalse(templateString, "Cannot parse template element");
+ templateStringTail = context.createTemplateStringList(templateStringTail, templateString);
+
+ while (!elementIsTail) {
+ failIfTrue(match(CLOSEBRACE), "Template literal expression cannot be empty");
+ TreeExpression expression = parseExpression(context);
+ failIfFalse(expression, "Cannot parse expression in template literal");
+
+ templateExpressionTail = context.createTemplateExpressionList(templateExpressionTail, expression);
+
+ auto templateString = parseTemplateString(context, false, rawStringsBuildMode, elementIsTail);
+ failIfFalse(templateString, "Cannot parse template element");
+ templateStringTail = context.createTemplateStringList(templateStringTail, templateString);
+ }
+
+ return context.createTemplateLiteral(location, templateStringList, templateExpressionList);
+}
+
+template <class LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::createResolveAndUseVariable(TreeBuilder& context, const Identifier* ident, bool isEval, const JSTextPosition& start, const JSTokenLocation& location)
+{
+ currentScope()->useVariable(ident, isEval);
+ m_parserState.lastIdentifier = ident;
+ return context.createResolve(location, *ident, start, lastTokenEndPosition());
+}
+
+template <typename LexerType>
template <class TreeBuilder> TreeExpression Parser<LexerType>::parsePrimaryExpression(TreeBuilder& context)
{
failIfStackOverflow();
switch (m_token.m_type) {
+ case FUNCTION:
+ return parseFunctionExpression(context);
+ case CLASSTOKEN:
+ return parseClassExpression(context);
case OPENBRACE:
if (strictMode())
return parseStrictObjectLiteral(context);
@@ -1941,25 +4172,50 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parsePrimaryExpre
return parseArrayLiteral(context);
case OPENPAREN: {
next();
- int oldNonLHSCount = m_nonLHSCount;
+ int oldNonLHSCount = m_parserState.nonLHSCount;
TreeExpression result = parseExpression(context);
- m_nonLHSCount = oldNonLHSCount;
+ m_parserState.nonLHSCount = oldNonLHSCount;
handleProductionOrFail(CLOSEPAREN, ")", "end", "compound expression");
return result;
}
case THISTOKEN: {
JSTokenLocation location(tokenLocation());
next();
- return context.thisExpr(location);
+ if (currentScope()->isArrowFunction())
+ currentScope()->setInnerArrowFunctionUsesThis();
+ return context.createThisExpr(location);
+ }
+ case AWAIT:
+ if (m_parserState.functionParsePhase == FunctionParsePhase::Parameters)
+ failIfFalse(m_parserState.allowAwait, "Cannot use await expression within parameters");
+ goto identifierExpression;
+ case ASYNC: {
+ JSTextPosition start = tokenStartPosition();
+ const Identifier* ident = m_token.m_data.ident;
+ JSTokenLocation location(tokenLocation());
+ next();
+ if (match(FUNCTION) && !m_lexer->prevTerminator())
+ return parseAsyncFunctionExpression(context);
+
+ // Avoid using variable if it is an arrow function parameter
+ if (UNLIKELY(match(ARROWFUNCTION)))
+ return 0;
+
+ const bool isEval = false;
+ return createResolveAndUseVariable(context, ident, isEval, start, location);
}
case IDENT: {
+ identifierExpression:
JSTextPosition start = tokenStartPosition();
const Identifier* ident = m_token.m_data.ident;
JSTokenLocation location(tokenLocation());
next();
- currentScope()->useVariable(ident, m_vm->propertyNames->eval == *ident);
- m_lastIdentifier = ident;
- return context.createResolve(location, ident, start);
+
+ // Avoid using variable if it is an arrow function parameter
+ if (UNLIKELY(match(ARROWFUNCTION)))
+ return 0;
+
+ return createResolveAndUseVariable(context, ident, *ident == m_vm->propertyNames->eval, start, location);
}
case STRING: {
const Identifier* ident = m_token.m_data.ident;
@@ -1967,11 +4223,17 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parsePrimaryExpre
next();
return context.createString(location, ident);
}
- case NUMBER: {
+ case DOUBLE: {
double d = m_token.m_data.doubleValue;
JSTokenLocation location(tokenLocation());
next();
- return context.createNumberExpr(location, d);
+ return context.createDoubleExpr(location, d);
+ }
+ case INTEGER: {
+ double d = m_token.m_data.doubleValue;
+ JSTokenLocation location(tokenLocation());
+ next();
+ return context.createIntegerExpr(location, d);
}
case NULLTOKEN: {
JSTokenLocation location(tokenLocation());
@@ -1991,30 +4253,41 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parsePrimaryExpre
case DIVEQUAL:
case DIVIDE: {
/* regexp */
- const Identifier* pattern;
- const Identifier* flags;
if (match(DIVEQUAL))
- failIfFalse(m_lexer->scanRegExp(pattern, flags, '='), "Invalid regular expression");
+ m_token.m_type = m_lexer->scanRegExp(&m_token, '=');
else
- failIfFalse(m_lexer->scanRegExp(pattern, flags), "Invalid regular expression");
-
+ m_token.m_type = m_lexer->scanRegExp(&m_token);
+ matchOrFail(REGEXP, "Invalid regular expression");
+
+ const Identifier* pattern = m_token.m_data.pattern;
+ const Identifier* flags = m_token.m_data.flags;
JSTextPosition start = tokenStartPosition();
JSTokenLocation location(tokenLocation());
next();
TreeExpression re = context.createRegExp(location, *pattern, *flags, start);
if (!re) {
- const char* yarrErrorMsg = Yarr::checkSyntax(pattern->string());
+ const char* yarrErrorMsg = Yarr::checkSyntax(pattern->string(), flags->string());
regexFail(yarrErrorMsg);
}
return re;
}
+ case BACKQUOTE:
+ return parseTemplateLiteral(context, LexerType::RawStringsBuildMode::DontBuildRawStrings);
+ case YIELD:
+ if (!strictMode() && !currentScope()->isGenerator())
+ goto identifierExpression;
+ failDueToUnexpectedToken();
+ case LET:
+ if (!strictMode())
+ goto identifierExpression;
+ FALLTHROUGH;
default:
failDueToUnexpectedToken();
}
}
template <typename LexerType>
-template <class TreeBuilder> TreeArguments Parser<LexerType>::parseArguments(TreeBuilder& context, SpreadMode mode)
+template <class TreeBuilder> TreeArguments Parser<LexerType>::parseArguments(TreeBuilder& context)
{
consumeOrFailWithFlags(OPENPAREN, TreeBuilder::DontBuildStrings, "Expected opening '(' at start of argument list");
JSTokenLocation location(tokenLocation());
@@ -2022,122 +4295,257 @@ template <class TreeBuilder> TreeArguments Parser<LexerType>::parseArguments(Tre
next(TreeBuilder::DontBuildStrings);
return context.createArguments();
}
- if (match(DOTDOTDOT) && mode == AllowSpread) {
- JSTokenLocation spreadLocation(tokenLocation());
- auto start = m_token.m_startPosition;
- auto divot = m_token.m_endPosition;
- next();
- auto spreadExpr = parseAssignmentExpression(context);
- auto end = m_lastTokenEndPosition;
- if (!spreadExpr)
- failWithMessage("Cannot parse spread expression");
- if (!consume(CLOSEPAREN)) {
- if (match(COMMA))
- semanticFail("Spread operator may only be applied to the last argument passed to a function");
- handleProductionOrFail(CLOSEPAREN, ")", "end", "argument list");
- }
- auto spread = context.createSpreadExpression(spreadLocation, spreadExpr, start, divot, end);
- TreeArgumentsList argList = context.createArgumentsList(location, spread);
- return context.createArguments(argList);
- }
- TreeExpression firstArg = parseAssignmentExpression(context);
+ auto argumentsStart = m_token.m_startPosition;
+ auto argumentsDivot = m_token.m_endPosition;
+
+ ArgumentType argType = ArgumentType::Normal;
+ TreeExpression firstArg = parseArgument(context, argType);
failIfFalse(firstArg, "Cannot parse function argument");
-
+ semanticFailIfTrue(match(DOTDOTDOT), "The '...' operator should come before the target expression");
+
+ bool hasSpread = false;
+ if (argType == ArgumentType::Spread)
+ hasSpread = true;
TreeArgumentsList argList = context.createArgumentsList(location, firstArg);
TreeArgumentsList tail = argList;
+
while (match(COMMA)) {
JSTokenLocation argumentLocation(tokenLocation());
next(TreeBuilder::DontBuildStrings);
- TreeExpression arg = parseAssignmentExpression(context);
- failIfFalse(arg, "Cannot parse function argument");
+
+ if (UNLIKELY(match(CLOSEPAREN)))
+ break;
+
+ TreeExpression arg = parseArgument(context, argType);
+ propagateError();
+ semanticFailIfTrue(match(DOTDOTDOT), "The '...' operator should come before the target expression");
+
+ if (argType == ArgumentType::Spread)
+ hasSpread = true;
+
tail = context.createArgumentsList(argumentLocation, tail, arg);
}
- semanticFailIfTrue(match(DOTDOTDOT), "The '...' operator should come before the target expression");
- handleProductionOrFail(CLOSEPAREN, ")", "end", "argument list");
+
+ handleProductionOrFail2(CLOSEPAREN, ")", "end", "argument list");
+ if (hasSpread) {
+ TreeExpression spreadArray = context.createSpreadExpression(location, context.createArray(location, context.createElementList(argList)), argumentsStart, argumentsDivot, m_lastTokenEndPosition);
+ return context.createArguments(context.createArgumentsList(location, spreadArray));
+ }
+
return context.createArguments(argList);
}
template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseArgument(TreeBuilder& context, ArgumentType& type)
+{
+ if (UNLIKELY(match(DOTDOTDOT))) {
+ JSTokenLocation spreadLocation(tokenLocation());
+ auto start = m_token.m_startPosition;
+ auto divot = m_token.m_endPosition;
+ next();
+ TreeExpression spreadExpr = parseAssignmentExpression(context);
+ propagateError();
+ auto end = m_lastTokenEndPosition;
+ type = ArgumentType::Spread;
+ return context.createSpreadExpression(spreadLocation, spreadExpr, start, divot, end);
+ }
+
+ type = ArgumentType::Normal;
+ return parseAssignmentExpression(context);
+}
+
+template <typename LexerType>
template <class TreeBuilder> TreeExpression Parser<LexerType>::parseMemberExpression(TreeBuilder& context)
{
TreeExpression base = 0;
JSTextPosition expressionStart = tokenStartPosition();
int newCount = 0;
+ JSTokenLocation startLocation = tokenLocation();
JSTokenLocation location;
while (match(NEW)) {
next();
newCount++;
}
-
- if (match(FUNCTION)) {
- const Identifier* name = &m_vm->propertyNames->nullIdentifier;
- TreeFormalParameterList parameters = 0;
- TreeFunctionBody body = 0;
- unsigned openBraceOffset = 0;
- unsigned closeBraceOffset = 0;
- int bodyStartLine = 0;
- unsigned bodyStartColumn = 0;
- location = tokenLocation();
+
+ bool baseIsSuper = match(SUPER);
+ bool baseIsImport = match(IMPORT);
+ semanticFailIfTrue((baseIsSuper || baseIsImport) && newCount, "Cannot use new with ", getToken());
+
+ bool baseIsNewTarget = false;
+ if (newCount && match(DOT)) {
next();
- failIfFalse((parseFunctionInfo(context, FunctionNoRequirements, FunctionMode, false, name, parameters, body, openBraceOffset, closeBraceOffset, bodyStartLine, bodyStartColumn)), "Cannot parse function expression");
- base = context.createFunctionExpr(location, name, body, parameters, openBraceOffset, closeBraceOffset, bodyStartLine, m_lastTokenEndPosition.line, bodyStartColumn);
- } else
+ if (match(IDENT)) {
+ const Identifier* ident = m_token.m_data.ident;
+ if (m_vm->propertyNames->target == *ident) {
+ semanticFailIfFalse(currentScope()->isFunction() || closestParentOrdinaryFunctionNonLexicalScope()->evalContextType() == EvalContextType::FunctionEvalContext, "new.target is only valid inside functions");
+ baseIsNewTarget = true;
+ if (currentScope()->isArrowFunction())
+ currentScope()->setInnerArrowFunctionUsesNewTarget();
+ base = context.createNewTargetExpr(location);
+ newCount--;
+ next();
+ } else
+ failWithMessage("\"new.\" can only followed with target");
+ } else
+ failDueToUnexpectedToken();
+ }
+
+ bool baseIsAsyncKeyword = false;
+
+ if (baseIsSuper) {
+ ScopeRef closestOrdinaryFunctionScope = closestParentOrdinaryFunctionNonLexicalScope();
+ semanticFailIfFalse(currentScope()->isFunction() || (closestOrdinaryFunctionScope->isEvalContext() && closestOrdinaryFunctionScope->expectedSuperBinding() == SuperBinding::Needed), "super is not valid in this context");
+ base = context.createSuperExpr(location);
+ next();
+ ScopeRef functionScope = currentFunctionScope();
+ if (!functionScope->setNeedsSuperBinding()) {
+ // It unnecessary to check of using super during reparsing one more time. Also it can lead to syntax error
+ // in case of arrow function because during reparsing we don't know whether we currently parse the arrow function
+ // inside of the constructor or method.
+ if (!m_lexer->isReparsingFunction()) {
+ SuperBinding functionSuperBinding = !functionScope->isArrowFunction() && !closestOrdinaryFunctionScope->isEvalContext()
+ ? functionScope->expectedSuperBinding()
+ : closestOrdinaryFunctionScope->expectedSuperBinding();
+ semanticFailIfTrue(functionSuperBinding == SuperBinding::NotNeeded, "super is not valid in this context");
+ }
+ }
+ } else if (baseIsImport) {
+ next();
+ JSTextPosition expressionEnd = lastTokenEndPosition();
+ consumeOrFail(OPENPAREN, "import call expects exactly one argument");
+ TreeExpression expr = parseAssignmentExpression(context);
+ failIfFalse(expr, "Cannot parse expression");
+ consumeOrFail(CLOSEPAREN, "import call expects exactly one argument");
+ base = context.createImportExpr(location, expr, expressionStart, expressionEnd, lastTokenEndPosition());
+ } else if (!baseIsNewTarget) {
+ const bool isAsync = match(ASYNC);
+
base = parsePrimaryExpression(context);
-
+ failIfFalse(base, "Cannot parse base expression");
+ if (UNLIKELY(isAsync && context.isResolve(base) && !m_lexer->prevTerminator())) {
+ if (matchSpecIdentifier()) {
+ // AsyncArrowFunction
+ forceClassifyExpressionError(ErrorIndicatesAsyncArrowFunction);
+ failDueToUnexpectedToken();
+ }
+ baseIsAsyncKeyword = true;
+ }
+ }
+
failIfFalse(base, "Cannot parse base expression");
while (true) {
location = tokenLocation();
switch (m_token.m_type) {
case OPENBRACKET: {
- m_nonTrivialExpressionCount++;
+ m_parserState.nonTrivialExpressionCount++;
JSTextPosition expressionEnd = lastTokenEndPosition();
next();
- int nonLHSCount = m_nonLHSCount;
- int initialAssignments = m_assignmentCount;
+ int nonLHSCount = m_parserState.nonLHSCount;
+ int initialAssignments = m_parserState.assignmentCount;
TreeExpression property = parseExpression(context);
failIfFalse(property, "Cannot parse subscript expression");
- base = context.createBracketAccess(location, base, property, initialAssignments != m_assignmentCount, expressionStart, expressionEnd, tokenEndPosition());
+ base = context.createBracketAccess(startLocation, base, property, initialAssignments != m_parserState.assignmentCount, expressionStart, expressionEnd, tokenEndPosition());
+
+ if (UNLIKELY(baseIsSuper && currentScope()->isArrowFunction()))
+ currentFunctionScope()->setInnerArrowFunctionUsesSuperProperty();
+
handleProductionOrFail(CLOSEBRACKET, "]", "end", "subscript expression");
- m_nonLHSCount = nonLHSCount;
+ m_parserState.nonLHSCount = nonLHSCount;
break;
}
case OPENPAREN: {
- m_nonTrivialExpressionCount++;
- int nonLHSCount = m_nonLHSCount;
+ m_parserState.nonTrivialExpressionCount++;
+ int nonLHSCount = m_parserState.nonLHSCount;
if (newCount) {
newCount--;
JSTextPosition expressionEnd = lastTokenEndPosition();
- TreeArguments arguments = parseArguments(context, DontAllowSpread);
+ TreeArguments arguments = parseArguments(context);
failIfFalse(arguments, "Cannot parse call arguments");
base = context.createNewExpr(location, base, arguments, expressionStart, expressionEnd, lastTokenEndPosition());
} else {
+ size_t usedVariablesSize = currentScope()->currentUsedVariablesSize();
JSTextPosition expressionEnd = lastTokenEndPosition();
- TreeArguments arguments = parseArguments(context, AllowSpread);
+ TreeArguments arguments = parseArguments(context);
+
+ if (baseIsAsyncKeyword && (!arguments || match(ARROWFUNCTION))) {
+ currentScope()->revertToPreviousUsedVariables(usedVariablesSize);
+ forceClassifyExpressionError(ErrorIndicatesAsyncArrowFunction);
+ failDueToUnexpectedToken();
+ }
+
failIfFalse(arguments, "Cannot parse call arguments");
- base = context.makeFunctionCallNode(location, base, arguments, expressionStart, expressionEnd, lastTokenEndPosition());
+ if (baseIsSuper) {
+ ScopeRef functionScope = currentFunctionScope();
+ if (!functionScope->setHasDirectSuper()) {
+ // It unnecessary to check of using super during reparsing one more time. Also it can lead to syntax error
+ // in case of arrow function because during reparsing we don't know whether we currently parse the arrow function
+ // inside of the constructor or method.
+ if (!m_lexer->isReparsingFunction()) {
+ ScopeRef closestOrdinaryFunctionScope = closestParentOrdinaryFunctionNonLexicalScope();
+ ConstructorKind functionConstructorKind = !functionScope->isArrowFunction() && !closestOrdinaryFunctionScope->isEvalContext()
+ ? functionScope->constructorKind()
+ : closestOrdinaryFunctionScope->constructorKind();
+ semanticFailIfTrue(functionConstructorKind == ConstructorKind::None, "super is not valid in this context");
+ semanticFailIfTrue(functionConstructorKind != ConstructorKind::Extends, "super is not valid in this context");
+ }
+ }
+ if (currentScope()->isArrowFunction())
+ functionScope->setInnerArrowFunctionUsesSuperCall();
+ }
+ base = context.makeFunctionCallNode(startLocation, base, arguments, expressionStart, expressionEnd, lastTokenEndPosition());
}
- m_nonLHSCount = nonLHSCount;
+ m_parserState.nonLHSCount = nonLHSCount;
break;
}
case DOT: {
- m_nonTrivialExpressionCount++;
+ m_parserState.nonTrivialExpressionCount++;
JSTextPosition expressionEnd = lastTokenEndPosition();
nextExpectIdentifier(LexerFlagsIgnoreReservedWords | TreeBuilder::DontBuildKeywords);
matchOrFail(IDENT, "Expected a property name after '.'");
- base = context.createDotAccess(location, base, m_token.m_data.ident, expressionStart, expressionEnd, tokenEndPosition());
+ base = context.createDotAccess(startLocation, base, m_token.m_data.ident, expressionStart, expressionEnd, tokenEndPosition());
+ if (UNLIKELY(baseIsSuper && currentScope()->isArrowFunction()))
+ currentFunctionScope()->setInnerArrowFunctionUsesSuperProperty();
next();
break;
}
+ case BACKQUOTE: {
+ semanticFailIfTrue(baseIsSuper, "Cannot use super as tag for tagged templates");
+ JSTextPosition expressionEnd = lastTokenEndPosition();
+ int nonLHSCount = m_parserState.nonLHSCount;
+ typename TreeBuilder::TemplateLiteral templateLiteral = parseTemplateLiteral(context, LexerType::RawStringsBuildMode::BuildRawStrings);
+ failIfFalse(templateLiteral, "Cannot parse template literal");
+ base = context.createTaggedTemplate(startLocation, base, templateLiteral, expressionStart, expressionEnd, lastTokenEndPosition());
+ m_parserState.nonLHSCount = nonLHSCount;
+ break;
+ }
default:
goto endMemberExpression;
}
+ baseIsSuper = false;
}
endMemberExpression:
+ semanticFailIfTrue(baseIsSuper, "super is not valid in this context");
while (newCount--)
base = context.createNewExpr(location, base, expressionStart, lastTokenEndPosition());
return base;
}
+template <typename LexerType>
+template <class TreeBuilder> TreeExpression Parser<LexerType>::parseArrowFunctionExpression(TreeBuilder& context, bool isAsync)
+{
+ JSTokenLocation location;
+
+ unsigned functionKeywordStart = tokenStart();
+ location = tokenLocation();
+ ParserFunctionInfo<TreeBuilder> info;
+ info.name = &m_vm->propertyNames->nullIdentifier;
+
+ SourceParseMode parseMode = isAsync ? SourceParseMode::AsyncArrowFunctionMode : SourceParseMode::ArrowFunctionMode;
+ failIfFalse((parseFunctionInfo(context, FunctionNameRequirements::Unnamed, parseMode, true, ConstructorKind::None, SuperBinding::NotNeeded, functionKeywordStart, info, FunctionDefinitionType::Expression)), "Cannot parse arrow function expression");
+
+ return context.createArrowFunctionExpr(location, info);
+}
+
static const char* operatorString(bool prefix, unsigned tok)
{
switch (tok) {
@@ -2177,6 +4585,12 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseUnaryExpress
bool modifiesExpr = false;
bool requiresLExpr = false;
unsigned lastOperator = 0;
+
+ if (UNLIKELY(match(AWAIT) && currentFunctionScope()->isAsyncFunctionBoundary()))
+ return parseAwaitExpression(context);
+
+ JSTokenLocation location(tokenLocation());
+
while (isUnaryOp(m_token.m_type)) {
if (strictMode()) {
switch (m_token.m_type) {
@@ -2198,43 +4612,48 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseUnaryExpress
}
}
lastOperator = m_token.m_type;
- m_nonLHSCount++;
+ m_parserState.nonLHSCount++;
context.appendUnaryToken(tokenStackDepth, m_token.m_type, tokenStartPosition());
next();
- m_nonTrivialExpressionCount++;
+ m_parserState.nonTrivialExpressionCount++;
}
JSTextPosition subExprStart = tokenStartPosition();
ASSERT(subExprStart.offset >= subExprStart.lineStartOffset);
- JSTokenLocation location(tokenLocation());
TreeExpression expr = parseMemberExpression(context);
if (!expr) {
if (lastOperator)
failWithMessage("Cannot parse subexpression of ", operatorString(true, lastOperator), "operator");
failWithMessage("Cannot parse member expression");
}
+ if (UNLIKELY(lastOperator && context.isNewTarget(expr)))
+ internalFailWithMessage(false, "new.target can't come after a prefix operator");
bool isEvalOrArguments = false;
if (strictMode() && !m_syntaxAlreadyValidated) {
if (context.isResolve(expr))
- isEvalOrArguments = *m_lastIdentifier == m_vm->propertyNames->eval || *m_lastIdentifier == m_vm->propertyNames->arguments;
+ isEvalOrArguments = *m_parserState.lastIdentifier == m_vm->propertyNames->eval || *m_parserState.lastIdentifier == m_vm->propertyNames->arguments;
}
- failIfTrueIfStrict(isEvalOrArguments && modifiesExpr, "Cannot modify '", m_lastIdentifier->impl(), "' in strict mode");
+ failIfTrueIfStrict(isEvalOrArguments && modifiesExpr, "Cannot modify '", m_parserState.lastIdentifier->impl(), "' in strict mode");
switch (m_token.m_type) {
case PLUSPLUS:
- m_nonTrivialExpressionCount++;
- m_nonLHSCount++;
+ if (UNLIKELY(context.isNewTarget(expr)))
+ internalFailWithMessage(false, "new.target can't come before a postfix operator");
+ m_parserState.nonTrivialExpressionCount++;
+ m_parserState.nonLHSCount++;
expr = context.makePostfixNode(location, expr, OpPlusPlus, subExprStart, lastTokenEndPosition(), tokenEndPosition());
- m_assignmentCount++;
- failIfTrueIfStrict(isEvalOrArguments, "Cannot modify '", m_lastIdentifier->impl(), "' in strict mode");
+ m_parserState.assignmentCount++;
+ failIfTrueIfStrict(isEvalOrArguments, "Cannot modify '", m_parserState.lastIdentifier->impl(), "' in strict mode");
semanticFailIfTrue(requiresLExpr, "The ", operatorString(false, lastOperator), " operator requires a reference expression");
lastOperator = PLUSPLUS;
next();
break;
case MINUSMINUS:
- m_nonTrivialExpressionCount++;
- m_nonLHSCount++;
+ if (UNLIKELY(context.isNewTarget(expr)))
+ internalFailWithMessage(false, "new.target can't come before a postfix operator");
+ m_parserState.nonTrivialExpressionCount++;
+ m_parserState.nonLHSCount++;
expr = context.makePostfixNode(location, expr, OpMinusMinus, subExprStart, lastTokenEndPosition(), tokenEndPosition());
- m_assignmentCount++;
- failIfTrueIfStrict(isEvalOrArguments, "'", m_lastIdentifier->impl(), "' cannot be modified in strict mode");
+ m_parserState.assignmentCount++;
+ failIfTrueIfStrict(isEvalOrArguments, "'", m_parserState.lastIdentifier->impl(), "' cannot be modified in strict mode");
semanticFailIfTrue(requiresLExpr, "The ", operatorString(false, lastOperator), " operator requires a reference expression");
lastOperator = PLUSPLUS;
next();
@@ -2248,8 +4667,6 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseUnaryExpress
if (!TreeBuilder::CreatesAST && (m_syntaxAlreadyValidated || !strictMode()))
return expr;
- location = tokenLocation();
- location.line = m_lexer->lastLineNumber();
while (tokenStackDepth) {
switch (context.unaryTokenStackLastType(tokenStackDepth)) {
case EXCLAMATION:
@@ -2267,12 +4684,12 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseUnaryExpress
case PLUSPLUS:
case AUTOPLUSPLUS:
expr = context.makePrefixNode(location, expr, OpPlusPlus, context.unaryTokenStackLastStart(tokenStackDepth), subExprStart + 1, end);
- m_assignmentCount++;
+ m_parserState.assignmentCount++;
break;
case MINUSMINUS:
case AUTOMINUSMINUS:
expr = context.makePrefixNode(location, expr, OpMinusMinus, context.unaryTokenStackLastStart(tokenStackDepth), subExprStart + 1, end);
- m_assignmentCount++;
+ m_parserState.assignmentCount++;
break;
case TYPEOF:
expr = context.makeTypeOfNode(location, expr);
@@ -2281,7 +4698,7 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseUnaryExpress
expr = context.createVoid(location, expr);
break;
case DELETETOKEN:
- failIfTrueIfStrict(context.isResolve(expr), "Cannot delete unqualified property '", m_lastIdentifier->impl(), "' in strict mode");
+ failIfTrueIfStrict(context.isResolve(expr), "Cannot delete unqualified property '", m_parserState.lastIdentifier->impl(), "' in strict mode");
expr = context.makeDeleteNode(location, expr, context.unaryTokenStackLastStart(tokenStackDepth), end, end);
break;
default:
@@ -2294,7 +4711,6 @@ template <class TreeBuilder> TreeExpression Parser<LexerType>::parseUnaryExpress
return expr;
}
-
template <typename LexerType> void Parser<LexerType>::printUnexpectedTokenText(WTF::PrintStream& out)
{
switch (m_token.m_type) {
@@ -2323,7 +4739,7 @@ template <typename LexerType> void Parser<LexerType>::printUnexpectedTokenText(W
case INVALID_NUMERIC_LITERAL_ERRORTOK:
out.print("Invalid numeric literal: '", getToken(), "'");
return;
- case INVALID_OCTAL_NUMBER_ERRORTOK:
+ case UNTERMINATED_OCTAL_NUMBER_ERRORTOK:
out.print("Invalid use of octal: '", getToken(), "'");
return;
case INVALID_STRING_LITERAL_ERRORTOK:
@@ -2335,7 +4751,8 @@ template <typename LexerType> void Parser<LexerType>::printUnexpectedTokenText(W
case STRING:
out.print("Unexpected string literal ", getToken());
return;
- case NUMBER:
+ case INTEGER:
+ case DOUBLE:
out.print("Unexpected number '", getToken(), "'");
return;
@@ -2346,7 +4763,13 @@ template <typename LexerType> void Parser<LexerType>::printUnexpectedTokenText(W
case RESERVED:
out.print("Unexpected use of reserved word '", getToken(), "'");
return;
+
+ case INVALID_PRIVATE_NAME_ERRORTOK:
+ out.print("Invalid private name '", getToken(), "'");
+ return;
+ case ASYNC:
+ case AWAIT:
case IDENT:
out.print("Unexpected identifier '", getToken(), "'");
return;