From 9bc09564f41682987aa5d00c7955a02205fcb54f Mon Sep 17 00:00:00 2001 From: HyukWoo Park Date: Thu, 14 Dec 2023 19:57:29 +0900 Subject: [PATCH] Fix parsing bugs in UnaryExpression and Exponentiation operation Signed-off-by: HyukWoo Park --- src/parser/esprima_cpp/esprima.cpp | 115 ++++++++++++----------------- test/vendortest | 2 +- 2 files changed, 48 insertions(+), 69 deletions(-) diff --git a/src/parser/esprima_cpp/esprima.cpp b/src/parser/esprima_cpp/esprima.cpp index d217e1acb..61195f5eb 100644 --- a/src/parser/esprima_cpp/esprima.cpp +++ b/src/parser/esprima_cpp/esprima.cpp @@ -2829,46 +2829,35 @@ class Parser { template ASTNode parseUnaryExpression(ASTBuilder& builder) { + ASTNode exprNode = nullptr; if (this->lookahead.type == Token::PunctuatorToken) { auto punctuatorsKind = this->lookahead.valuePunctuatorKind; - ASTNode exprNode = nullptr; - if (punctuatorsKind == Plus) { + if (punctuatorsKind == Plus || punctuatorsKind == Minus || punctuatorsKind == Wave || punctuatorsKind == ExclamationMark) { this->nextToken(); MetaNode node = this->startNode(&this->lookahead); ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - exprNode = this->finalize(node, builder.createUnaryExpressionPlusNode(subExpr)); this->context->isAssignmentTarget = false; this->context->isBindingElement = false; - return exprNode; - } else if (punctuatorsKind == Minus) { - this->nextToken(); - if (this->lookahead.type == Token::NumericLiteralToken) { - return parseNumericLiteralNode(builder, true); - } else { - MetaNode node = this->startNode(&this->lookahead); - ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - exprNode = this->finalize(node, builder.createUnaryExpressionMinusNode(subExpr)); - this->context->isAssignmentTarget = false; - this->context->isBindingElement = false; + + switch (punctuatorsKind) { + case Plus: { + return this->finalize(node, builder.createUnaryExpressionPlusNode(subExpr)); + } + case Minus: { + return this->finalize(node, builder.createUnaryExpressionMinusNode(subExpr)); + } + case Wave: { + return this->finalize(node, builder.createUnaryExpressionBitwiseNotNode(subExpr)); + } + case ExclamationMark: { + return this->finalize(node, builder.createUnaryExpressionLogicalNotNode(subExpr)); + } + default: { + RELEASE_ASSERT_NOT_REACHED(); return exprNode; } - } else if (punctuatorsKind == Wave) { - this->nextToken(); - MetaNode node = this->startNode(&this->lookahead); - ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - exprNode = this->finalize(node, builder.createUnaryExpressionBitwiseNotNode(subExpr)); - this->context->isAssignmentTarget = false; - this->context->isBindingElement = false; - return exprNode; - } else if (punctuatorsKind == ExclamationMark) { - this->nextToken(); - MetaNode node = this->startNode(&this->lookahead); - ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - exprNode = this->finalize(node, builder.createUnaryExpressionLogicalNotNode(subExpr)); - this->context->isAssignmentTarget = false; - this->context->isBindingElement = false; - return exprNode; + } } else if (punctuatorsKind == Hash) { ALLOC_TOKEN(hashToken); this->nextToken(hashToken); @@ -2897,47 +2886,39 @@ class Parser { this->throwUnexpectedToken(*hashToken); } } else if (this->lookahead.type == Token::KeywordToken) { - ASTNode exprNode = nullptr; - - if (this->lookahead.valueKeywordKind == DeleteKeyword) { - this->nextToken(); - MetaNode node = this->startNode(&this->lookahead); - ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - - if (this->context->strict && subExpr->isIdentifier()) { - this->throwError(Messages::StrictDelete); - } + auto valueKeywordKind = this->lookahead.valueKeywordKind; - if (subExpr->type() == ASTNodeType::MemberExpression) { - if (subExpr->asMemberExpression()->isReferencePrivateField()) { - this->throwError(Messages::CannnotUsePrivateFieldHere); - } - } - - exprNode = this->finalize(node, builder.createUnaryExpressionDeleteNode(subExpr)); - this->context->isAssignmentTarget = false; - this->context->isBindingElement = false; - - return exprNode; - } else if (this->lookahead.valueKeywordKind == VoidKeyword) { + if (valueKeywordKind == DeleteKeyword || valueKeywordKind == VoidKeyword || valueKeywordKind == TypeofKeyword) { this->nextToken(); MetaNode node = this->startNode(&this->lookahead); ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - exprNode = this->finalize(node, builder.createUnaryExpressionVoidNode(subExpr)); this->context->isAssignmentTarget = false; this->context->isBindingElement = false; - return exprNode; - } else if (this->lookahead.valueKeywordKind == TypeofKeyword) { - this->nextToken(); - - MetaNode node = this->startNode(&this->lookahead); - ASTNode subExpr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - exprNode = this->finalize(node, builder.createUnaryExpressionTypeOfNode(subExpr)); - this->context->isAssignmentTarget = false; - this->context->isBindingElement = false; + switch (valueKeywordKind) { + case DeleteKeyword: { + if (this->context->strict && subExpr->isIdentifier()) { + this->throwError(Messages::StrictDelete); + } - return exprNode; + if (subExpr->type() == ASTNodeType::MemberExpression) { + if (subExpr->asMemberExpression()->isReferencePrivateField()) { + this->throwError(Messages::CannnotUsePrivateFieldHere); + } + } + return this->finalize(node, builder.createUnaryExpressionDeleteNode(subExpr)); + } + case VoidKeyword: { + return this->finalize(node, builder.createUnaryExpressionVoidNode(subExpr)); + } + case TypeofKeyword: { + return this->finalize(node, builder.createUnaryExpressionTypeOfNode(subExpr)); + } + default: { + RELEASE_ASSERT_NOT_REACHED(); + return exprNode; + } + } } } else if (this->matchContextualKeyword(AwaitKeyword)) { bool parseAwait = false; @@ -2961,8 +2942,7 @@ class Parser { } } if (parseAwait) { - ASTNode exprNode = this->parseAwaitExpression(builder); - return exprNode; + return this->parseAwaitExpression(builder); } } @@ -2974,11 +2954,10 @@ class Parser { { ALLOC_TOKEN(startToken); *startToken = this->lookahead; - bool seenMinus = this->match(Minus); ASTNode expr = this->inheritCoverGrammar(builder, &Parser::parseUnaryExpression); - if (!expr->isUnaryOperation() && this->match(Exponentiation)) { - if (UNLIKELY(seenMinus)) { + if (this->match(Exponentiation)) { + if (expr->isUnaryOperation() && (startToken->type != Token::PunctuatorToken || startToken->valuePunctuatorKind != PunctuatorKind::LeftParenthesis)) { this->throwError("Unary operator used immediately before exponentiation expression. Parenthesis must be used to disambiguate operator precedence"); } this->nextToken(); diff --git a/test/vendortest b/test/vendortest index 84643acf3..45a8c9b00 160000 --- a/test/vendortest +++ b/test/vendortest @@ -1 +1 @@ -Subproject commit 84643acf340624f0004fce798c675af8b837083b +Subproject commit 45a8c9b007201fbac0dc8c647535c58a3506c774