Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(291)

Side by Side Diff: src/parsing/parser-base.h

Issue 2323763002: [parser] Refactor of Parse*Statement*, part 4 (Closed)
Patch Set: The real patch Created 4 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #ifndef V8_PARSING_PARSER_BASE_H 5 #ifndef V8_PARSING_PARSER_BASE_H
6 #define V8_PARSING_PARSER_BASE_H 6 #define V8_PARSING_PARSER_BASE_H
7 7
8 #include "src/ast/ast.h" 8 #include "src/ast/ast.h"
9 #include "src/ast/scopes.h" 9 #include "src/ast/scopes.h"
10 #include "src/bailout-reason.h" 10 #include "src/bailout-reason.h"
(...skipping 656 matching lines...) Expand 10 before | Expand all | Expand 10 after
667 DeclarationScope* result = 667 DeclarationScope* result =
668 new (zone()) DeclarationScope(zone(), scope(), FUNCTION_SCOPE, kind); 668 new (zone()) DeclarationScope(zone(), scope(), FUNCTION_SCOPE, kind);
669 // TODO(verwaest): Move into the DeclarationScope constructor. 669 // TODO(verwaest): Move into the DeclarationScope constructor.
670 if (!IsArrowFunction(kind)) { 670 if (!IsArrowFunction(kind)) {
671 result->DeclareThis(ast_value_factory()); 671 result->DeclareThis(ast_value_factory());
672 result->DeclareDefaultFunctionVariables(ast_value_factory()); 672 result->DeclareDefaultFunctionVariables(ast_value_factory());
673 } 673 }
674 return result; 674 return result;
675 } 675 }
676 676
677 V8_INLINE DeclarationScope* GetDeclarationScope() const {
678 return scope()->GetDeclarationScope();
679 }
680 V8_INLINE DeclarationScope* GetClosureScope() const {
681 return scope()->GetClosureScope();
682 }
683
677 Scanner* scanner() const { return scanner_; } 684 Scanner* scanner() const { return scanner_; }
678 AstValueFactory* ast_value_factory() const { return ast_value_factory_; } 685 AstValueFactory* ast_value_factory() const { return ast_value_factory_; }
679 int position() const { return scanner_->location().beg_pos; } 686 int position() const { return scanner_->location().beg_pos; }
680 int peek_position() const { return scanner_->peek_location().beg_pos; } 687 int peek_position() const { return scanner_->peek_location().beg_pos; }
681 bool stack_overflow() const { return stack_overflow_; } 688 bool stack_overflow() const { return stack_overflow_; }
682 void set_stack_overflow() { stack_overflow_ = true; } 689 void set_stack_overflow() { stack_overflow_ = true; }
683 Mode mode() const { return mode_; } 690 Mode mode() const { return mode_; }
684 691
685 INLINE(Token::Value peek()) { 692 INLINE(Token::Value peek()) {
686 if (stack_overflow_) return Token::ILLEGAL; 693 if (stack_overflow_) return Token::ILLEGAL;
(...skipping 187 matching lines...) Expand 10 before | Expand all | Expand 10 after
874 LanguageMode old = scope()->language_mode(); 881 LanguageMode old = scope()->language_mode();
875 impl()->SetLanguageMode(scope(), old > mode ? old : mode); 882 impl()->SetLanguageMode(scope(), old > mode ? old : mode);
876 } 883 }
877 bool is_generator() const { return function_state_->is_generator(); } 884 bool is_generator() const { return function_state_->is_generator(); }
878 bool is_async_function() const { 885 bool is_async_function() const {
879 return function_state_->is_async_function(); 886 return function_state_->is_async_function();
880 } 887 }
881 bool is_resumable() const { return function_state_->is_resumable(); } 888 bool is_resumable() const { return function_state_->is_resumable(); }
882 889
883 // Report syntax errors. 890 // Report syntax errors.
884 void ReportMessage(MessageTemplate::Template message, const char* arg = NULL, 891 void ReportMessage(MessageTemplate::Template message) {
892 Scanner::Location source_location = scanner()->location();
893 impl()->ReportMessageAt(source_location, message,
894 static_cast<const char*>(nullptr), kSyntaxError);
895 }
896
897 template <typename T>
898 void ReportMessage(MessageTemplate::Template message, T arg,
885 ParseErrorType error_type = kSyntaxError) { 899 ParseErrorType error_type = kSyntaxError) {
886 Scanner::Location source_location = scanner()->location(); 900 Scanner::Location source_location = scanner()->location();
887 impl()->ReportMessageAt(source_location, message, arg, error_type); 901 impl()->ReportMessageAt(source_location, message, arg, error_type);
888 }
889
890 void ReportMessage(MessageTemplate::Template message, const AstRawString* arg,
891 ParseErrorType error_type = kSyntaxError) {
892 Scanner::Location source_location = scanner()->location();
893 impl()->ReportMessageAt(source_location, message, arg, error_type);
894 } 902 }
895 903
896 void ReportMessageAt(Scanner::Location location, 904 void ReportMessageAt(Scanner::Location location,
897 MessageTemplate::Template message, 905 MessageTemplate::Template message,
898 ParseErrorType error_type) { 906 ParseErrorType error_type) {
899 impl()->ReportMessageAt(location, message, 907 impl()->ReportMessageAt(location, message,
900 static_cast<const char*>(nullptr), error_type); 908 static_cast<const char*>(nullptr), error_type);
901 } 909 }
902 910
903 void GetUnexpectedTokenMessage( 911 void GetUnexpectedTokenMessage(
(...skipping 276 matching lines...) Expand 10 before | Expand all | Expand 10 after
1180 1188
1181 StatementT ParseVariableStatement(VariableDeclarationContext var_context, 1189 StatementT ParseVariableStatement(VariableDeclarationContext var_context,
1182 ZoneList<const AstRawString*>* names, 1190 ZoneList<const AstRawString*>* names,
1183 bool* ok); 1191 bool* ok);
1184 1192
1185 // Magical syntax support. 1193 // Magical syntax support.
1186 ExpressionT ParseV8Intrinsic(bool* ok); 1194 ExpressionT ParseV8Intrinsic(bool* ok);
1187 1195
1188 StatementT ParseDebuggerStatement(bool* ok); 1196 StatementT ParseDebuggerStatement(bool* ok);
1189 1197
1198 StatementT ParseExpressionOrLabelledStatement(
1199 ZoneList<const AstRawString*>* labels,
1200 AllowLabelledFunctionStatement allow_function, bool* ok);
1201 StatementT ParseIfStatement(ZoneList<const AstRawString*>* labels, bool* ok);
1202 StatementT ParseContinueStatement(bool* ok);
1203 StatementT ParseBreakStatement(ZoneList<const AstRawString*>* labels,
1204 bool* ok);
1205 StatementT ParseReturnStatement(bool* ok);
1206 StatementT ParseWithStatement(ZoneList<const AstRawString*>* labels,
1207 bool* ok);
1208
1190 bool IsNextLetKeyword(); 1209 bool IsNextLetKeyword();
1191 bool IsTrivialExpression(); 1210 bool IsTrivialExpression();
1192 1211
1193 // Checks if the expression is a valid reference expression (e.g., on the 1212 // Checks if the expression is a valid reference expression (e.g., on the
1194 // left-hand side of assignments). Although ruled out by ECMA as early errors, 1213 // left-hand side of assignments). Although ruled out by ECMA as early errors,
1195 // we allow calls for web compatibility and rewrite them to a runtime throw. 1214 // we allow calls for web compatibility and rewrite them to a runtime throw.
1196 ExpressionT CheckAndRewriteReferenceExpression( 1215 ExpressionT CheckAndRewriteReferenceExpression(
1197 ExpressionT expression, int beg_pos, int end_pos, 1216 ExpressionT expression, int beg_pos, int end_pos,
1198 MessageTemplate::Template message, bool* ok); 1217 MessageTemplate::Template message, bool* ok);
1199 ExpressionT CheckAndRewriteReferenceExpression( 1218 ExpressionT CheckAndRewriteReferenceExpression(
(...skipping 3111 matching lines...) Expand 10 before | Expand all | Expand 10 after
4311 // break point is present. 4330 // break point is present.
4312 // DebuggerStatement :: 4331 // DebuggerStatement ::
4313 // 'debugger' ';' 4332 // 'debugger' ';'
4314 4333
4315 int pos = peek_position(); 4334 int pos = peek_position();
4316 Expect(Token::DEBUGGER, CHECK_OK); 4335 Expect(Token::DEBUGGER, CHECK_OK);
4317 ExpectSemicolon(CHECK_OK); 4336 ExpectSemicolon(CHECK_OK);
4318 return factory()->NewDebuggerStatement(pos); 4337 return factory()->NewDebuggerStatement(pos);
4319 } 4338 }
4320 4339
4340 template <typename Impl>
4341 typename ParserBase<Impl>::StatementT
4342 ParserBase<Impl>::ParseExpressionOrLabelledStatement(
4343 ZoneList<const AstRawString*>* labels,
4344 AllowLabelledFunctionStatement allow_function, bool* ok) {
4345 // ExpressionStatement | LabelledStatement ::
4346 // Expression ';'
4347 // Identifier ':' Statement
4348 //
4349 // ExpressionStatement[Yield] :
4350 // [lookahead ∉ {{, function, class, let [}] Expression[In, ?Yield] ;
4351
4352 int pos = peek_position();
4353
4354 switch (peek()) {
4355 case Token::FUNCTION:
4356 case Token::LBRACE:
4357 UNREACHABLE(); // Always handled by the callers.
4358 case Token::CLASS:
4359 ReportUnexpectedToken(Next());
4360 *ok = false;
4361 return impl()->NullStatement();
4362 default:
4363 break;
4364 }
4365
4366 bool starts_with_identifier = peek_any_identifier();
4367 ExpressionT expr = ParseExpression(true, CHECK_OK);
4368 if (peek() == Token::COLON && starts_with_identifier &&
4369 impl()->IsIdentifier(expr)) {
4370 // The whole expression was a single identifier, and not, e.g.,
4371 // something starting with an identifier or a parenthesized identifier.
4372 labels = impl()->DeclareLabel(labels, expr, CHECK_OK);
adamk 2016/09/08 18:33:19 Can you add an AsIdentifierExpression() method to
nickie 2016/09/09 09:42:26 Done.
4373 Consume(Token::COLON);
4374 // ES#sec-labelled-function-declarations Labelled Function Declarations
4375 if (peek() == Token::FUNCTION && is_sloppy(language_mode())) {
4376 if (allow_function == kAllowLabelledFunctionStatement) {
4377 return impl()->ParseFunctionDeclaration(ok);
4378 } else {
4379 return ParseScopedStatement(labels, true, ok);
4380 }
4381 }
4382 return ParseStatement(labels, kDisallowLabelledFunctionStatement, ok);
4383 }
4384
4385 // If we have an extension, we allow a native function declaration.
4386 // A native function declaration starts with "native function" with
4387 // no line-terminator between the two words.
4388 if (extension_ != nullptr && peek() == Token::FUNCTION &&
4389 !scanner()->HasAnyLineTerminatorBeforeNext() && impl()->IsNative(expr)) {
4390 return impl()->ParseNativeDeclaration(ok);
4391 }
4392
4393 // Parsed expression statement, followed by semicolon.
4394 ExpectSemicolon(CHECK_OK);
4395 return factory()->NewExpressionStatement(expr, pos);
4396 }
4397
4398 template <typename Impl>
4399 typename ParserBase<Impl>::StatementT ParserBase<Impl>::ParseIfStatement(
4400 ZoneList<const AstRawString*>* labels, bool* ok) {
4401 // IfStatement ::
4402 // 'if' '(' Expression ')' Statement ('else' Statement)?
4403
4404 int pos = peek_position();
4405 Expect(Token::IF, CHECK_OK);
4406 Expect(Token::LPAREN, CHECK_OK);
4407 ExpressionT condition = ParseExpression(true, CHECK_OK);
4408 Expect(Token::RPAREN, CHECK_OK);
4409 StatementT then_statement = ParseScopedStatement(labels, false, CHECK_OK);
4410 StatementT else_statement = impl()->NullStatement();
4411 if (Check(Token::ELSE)) {
4412 else_statement = ParseScopedStatement(labels, false, CHECK_OK);
4413 } else {
4414 else_statement = factory()->NewEmptyStatement(kNoSourcePosition);
4415 }
4416 return factory()->NewIfStatement(condition, then_statement, else_statement,
4417 pos);
4418 }
4419
4420 template <typename Impl>
4421 typename ParserBase<Impl>::StatementT ParserBase<Impl>::ParseContinueStatement(
4422 bool* ok) {
4423 // ContinueStatement ::
4424 // 'continue' Identifier? ';'
4425
4426 int pos = peek_position();
4427 Expect(Token::CONTINUE, CHECK_OK);
4428 IdentifierT label = impl()->EmptyIdentifier();
4429 Token::Value tok = peek();
4430 if (!scanner()->HasAnyLineTerminatorBeforeNext() && tok != Token::SEMICOLON &&
4431 tok != Token::RBRACE && tok != Token::EOS) {
4432 // ECMA allows "eval" or "arguments" as labels even in strict mode.
4433 label = ParseIdentifier(kAllowRestrictedIdentifiers, CHECK_OK);
4434 }
4435 typename Types::IterationStatementT target =
4436 impl()->LookupContinueTarget(label, CHECK_OK);
4437 if (impl()->IsNullStatement(target)) {
4438 // Illegal continue statement.
4439 MessageTemplate::Template message = MessageTemplate::kIllegalContinue;
4440 if (!impl()->IsEmptyIdentifier(label)) {
4441 message = MessageTemplate::kUnknownLabel;
4442 }
4443 ReportMessage(message, label);
4444 *ok = false;
4445 return impl()->NullStatement();
4446 }
4447 ExpectSemicolon(CHECK_OK);
4448 return factory()->NewContinueStatement(target, pos);
4449 }
4450
4451 template <typename Impl>
4452 typename ParserBase<Impl>::StatementT ParserBase<Impl>::ParseBreakStatement(
4453 ZoneList<const AstRawString*>* labels, bool* ok) {
4454 // BreakStatement ::
4455 // 'break' Identifier? ';'
4456
4457 int pos = peek_position();
4458 Expect(Token::BREAK, CHECK_OK);
4459 IdentifierT label = impl()->EmptyIdentifier();
4460 Token::Value tok = peek();
4461 if (!scanner()->HasAnyLineTerminatorBeforeNext() && tok != Token::SEMICOLON &&
4462 tok != Token::RBRACE && tok != Token::EOS) {
4463 // ECMA allows "eval" or "arguments" as labels even in strict mode.
4464 label = ParseIdentifier(kAllowRestrictedIdentifiers, CHECK_OK);
4465 }
4466 // Parse labeled break statements that target themselves into
4467 // empty statements, e.g. 'l1: l2: l3: break l2;'
4468 if (!impl()->IsEmptyIdentifier(label) &&
4469 impl()->ContainsLabel(labels, label)) {
4470 ExpectSemicolon(CHECK_OK);
4471 return factory()->NewEmptyStatement(pos);
4472 }
4473 typename Types::BreakableStatementT target =
4474 impl()->LookupBreakTarget(label, CHECK_OK);
4475 if (impl()->IsNullStatement(target)) {
4476 // Illegal break statement.
4477 MessageTemplate::Template message = MessageTemplate::kIllegalBreak;
4478 if (!impl()->IsEmptyIdentifier(label)) {
4479 message = MessageTemplate::kUnknownLabel;
4480 }
4481 ReportMessage(message, label);
4482 *ok = false;
4483 return impl()->NullStatement();
4484 }
4485 ExpectSemicolon(CHECK_OK);
4486 return factory()->NewBreakStatement(target, pos);
4487 }
4488
4489 template <typename Impl>
4490 typename ParserBase<Impl>::StatementT ParserBase<Impl>::ParseReturnStatement(
4491 bool* ok) {
4492 // ReturnStatement ::
4493 // 'return' [no line terminator] Expression? ';'
4494
4495 // Consume the return token. It is necessary to do that before
4496 // reporting any errors on it, because of the way errors are
4497 // reported (underlining).
4498 Expect(Token::RETURN, CHECK_OK);
4499 Scanner::Location loc = scanner()->location();
4500
4501 Token::Value tok = peek();
4502 ExpressionT return_value = impl()->EmptyExpression();
4503 if (scanner()->HasAnyLineTerminatorBeforeNext() || tok == Token::SEMICOLON ||
4504 tok == Token::RBRACE || tok == Token::EOS) {
4505 if (IsSubclassConstructor(function_state_->kind())) {
4506 return_value = impl()->ThisExpression(loc.beg_pos);
4507 } else {
4508 return_value = impl()->GetLiteralUndefined(position());
4509 }
4510 } else {
4511 if (IsSubclassConstructor(function_state_->kind())) {
4512 // Because of the return code rewriting that happens in case of a subclass
4513 // constructor we don't want to accept tail calls, therefore we don't set
4514 // ReturnExprScope to kInsideValidReturnStatement here.
4515 return_value = ParseExpression(true, CHECK_OK);
4516 } else {
4517 ReturnExprScope maybe_allow_tail_calls(
4518 function_state_, ReturnExprContext::kInsideValidReturnStatement);
4519 return_value = ParseExpression(true, CHECK_OK);
4520
4521 if (allow_tailcalls() && !is_sloppy(language_mode()) && !is_resumable()) {
4522 // ES6 14.6.1 Static Semantics: IsInTailPosition
4523 function_state_->AddImplicitTailCallExpression(return_value);
4524 }
4525 }
4526 }
4527 ExpectSemicolon(CHECK_OK);
4528 return_value = impl()->RewriteReturn(return_value, loc.beg_pos);
4529
4530 DeclarationScope* decl_scope = GetDeclarationScope();
4531 if (decl_scope->is_script_scope() || decl_scope->is_eval_scope()) {
4532 impl()->ReportMessageAt(loc, MessageTemplate::kIllegalReturn);
4533 *ok = false;
4534 return impl()->NullStatement();
4535 }
4536 return factory()->NewReturnStatement(return_value, loc.beg_pos);
4537 }
4538
4539 template <typename Impl>
4540 typename ParserBase<Impl>::StatementT ParserBase<Impl>::ParseWithStatement(
4541 ZoneList<const AstRawString*>* labels, bool* ok) {
4542 // WithStatement ::
4543 // 'with' '(' Expression ')' Statement
4544
4545 Expect(Token::WITH, CHECK_OK);
4546 int pos = position();
4547
4548 if (is_strict(language_mode())) {
4549 ReportMessage(MessageTemplate::kStrictWith);
4550 *ok = false;
4551 return impl()->NullStatement();
4552 }
4553
4554 Expect(Token::LPAREN, CHECK_OK);
4555 ExpressionT expr = ParseExpression(true, CHECK_OK);
4556 Expect(Token::RPAREN, CHECK_OK);
4557
4558 Scope* with_scope = NewScope(WITH_SCOPE);
4559 StatementT body = impl()->NullStatement();
4560 {
4561 BlockState block_state(&scope_state_, with_scope);
4562 with_scope->set_start_position(scanner()->peek_location().beg_pos);
4563 body = ParseScopedStatement(labels, true, CHECK_OK);
4564 with_scope->set_end_position(scanner()->location().end_pos);
4565 }
4566 return factory()->NewWithStatement(with_scope, expr, body, pos);
4567 }
4568
4321 #undef CHECK_OK 4569 #undef CHECK_OK
4322 #undef CHECK_OK_CUSTOM 4570 #undef CHECK_OK_CUSTOM
4323 4571
4324 template <typename Impl> 4572 template <typename Impl>
4325 void ParserBase<Impl>::ObjectLiteralChecker::CheckDuplicateProto( 4573 void ParserBase<Impl>::ObjectLiteralChecker::CheckDuplicateProto(
4326 Token::Value property) { 4574 Token::Value property) {
4327 if (property == Token::SMI || property == Token::NUMBER) return; 4575 if (property == Token::SMI || property == Token::NUMBER) return;
4328 4576
4329 if (IsProto()) { 4577 if (IsProto()) {
4330 if (has_seen_proto_) { 4578 if (has_seen_proto_) {
(...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after
4369 has_seen_constructor_ = true; 4617 has_seen_constructor_ = true;
4370 return; 4618 return;
4371 } 4619 }
4372 } 4620 }
4373 4621
4374 4622
4375 } // namespace internal 4623 } // namespace internal
4376 } // namespace v8 4624 } // namespace v8
4377 4625
4378 #endif // V8_PARSING_PARSER_BASE_H 4626 #endif // V8_PARSING_PARSER_BASE_H
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698