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

Side by Side Diff: src/parsing/parser.cc

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 #include "src/parsing/parser.h" 5 #include "src/parsing/parser.h"
6 6
7 #include <memory> 7 #include <memory>
8 8
9 #include "src/api.h" 9 #include "src/api.h"
10 #include "src/ast/ast-expression-rewriter.h" 10 #include "src/ast/ast-expression-rewriter.h"
(...skipping 1651 matching lines...) Expand 10 before | Expand all | Expand 10 after
1662 1662
1663 void Parser::DeclareAndInitializeVariables( 1663 void Parser::DeclareAndInitializeVariables(
1664 Block* block, const DeclarationDescriptor* declaration_descriptor, 1664 Block* block, const DeclarationDescriptor* declaration_descriptor,
1665 const DeclarationParsingResult::Declaration* declaration, 1665 const DeclarationParsingResult::Declaration* declaration,
1666 ZoneList<const AstRawString*>* names, bool* ok) { 1666 ZoneList<const AstRawString*>* names, bool* ok) {
1667 DCHECK_NOT_NULL(block); 1667 DCHECK_NOT_NULL(block);
1668 PatternRewriter::DeclareAndInitializeVariables( 1668 PatternRewriter::DeclareAndInitializeVariables(
1669 this, block, declaration_descriptor, declaration, names, ok); 1669 this, block, declaration_descriptor, declaration, names, ok);
1670 } 1670 }
1671 1671
1672 static bool ContainsLabel(ZoneList<const AstRawString*>* labels, 1672 ZoneList<const AstRawString*>* Parser::DeclareLabel(
1673 const AstRawString* label) { 1673 ZoneList<const AstRawString*>* labels, Expression* expr, bool* ok) {
1674 DCHECK(label != NULL); 1674 VariableProxy* var = expr->AsVariableProxy();
1675 if (labels != NULL) { 1675 const AstRawString* label = var->raw_name();
1676 for (int i = labels->length(); i-- > 0; ) { 1676 // TODO(1240780): We don't check for redeclaration of labels
1677 if (labels->at(i) == label) { 1677 // during preparsing since keeping track of the set of active
1678 return true; 1678 // labels requires nontrivial changes to the way scopes are
1679 } 1679 // structured. However, these are probably changes we want to
1680 // make later anyway so we should go back and fix this then.
1681 if (ContainsLabel(labels, label) || TargetStackContainsLabel(label)) {
1682 ReportMessage(MessageTemplate::kLabelRedeclaration, label);
1683 *ok = false;
1684 return nullptr;
1685 }
1686 if (labels == nullptr) {
1687 labels = new (zone()) ZoneList<const AstRawString*>(1, zone());
1688 }
1689 labels->Add(label, zone());
1690 // Remove the "ghost" variable that turned out to be a label
1691 // from the top scope. This way, we don't try to resolve it
1692 // during the scope processing.
1693 scope()->RemoveUnresolved(var);
1694 return labels;
1695 }
1696
1697 bool Parser::ContainsLabel(ZoneList<const AstRawString*>* labels,
1698 const AstRawString* label) {
1699 DCHECK_NOT_NULL(label);
1700 if (labels != nullptr) {
1701 for (int i = labels->length(); i-- > 0;) {
1702 if (labels->at(i) == label) return true;
1680 } 1703 }
1681 } 1704 }
1682 return false; 1705 return false;
1683 } 1706 }
1684 1707
1708 Expression* Parser::RewriteReturn(Expression* return_value, int pos) {
1709 if (IsSubclassConstructor(function_state_->kind())) {
1710 // For subclass constructors we need to return this in case of undefined
1711 // return a Smi (transformed into an exception in the ConstructStub)
1712 // for a non object.
1713 //
1714 // return expr;
1715 //
1716 // Is rewritten as:
1717 //
1718 // return (temp = expr) === undefined ? this :
1719 // %_IsJSReceiver(temp) ? temp : 1;
1720
1721 // temp = expr
1722 Variable* temp = NewTemporary(ast_value_factory()->empty_string());
1723 Assignment* assign = factory()->NewAssignment(
1724 Token::ASSIGN, factory()->NewVariableProxy(temp), return_value, pos);
1725
1726 // %_IsJSReceiver(temp)
1727 ZoneList<Expression*>* is_spec_object_args =
1728 new (zone()) ZoneList<Expression*>(1, zone());
1729 is_spec_object_args->Add(factory()->NewVariableProxy(temp), zone());
1730 Expression* is_spec_object_call = factory()->NewCallRuntime(
1731 Runtime::kInlineIsJSReceiver, is_spec_object_args, pos);
1732
1733 // %_IsJSReceiver(temp) ? temp : 1;
1734 Expression* is_object_conditional = factory()->NewConditional(
1735 is_spec_object_call, factory()->NewVariableProxy(temp),
1736 factory()->NewSmiLiteral(1, pos), pos);
1737
1738 // temp === undefined
1739 Expression* is_undefined = factory()->NewCompareOperation(
1740 Token::EQ_STRICT, assign,
1741 factory()->NewUndefinedLiteral(kNoSourcePosition), pos);
1742
1743 // is_undefined ? this : is_object_conditional
1744 return_value = factory()->NewConditional(is_undefined, ThisExpression(pos),
1745 is_object_conditional, pos);
1746 }
1747 if (is_generator()) {
1748 return_value = impl()->BuildIteratorResult(return_value, true);
marja 2016/09/09 08:44:59 Why impl() here?
nickie 2016/09/09 09:42:26 Right, my bad. I tried to keep this in parser bas
1749 } else if (is_async_function()) {
1750 return_value =
1751 impl()->BuildResolvePromise(return_value, return_value->position());
marja 2016/09/09 08:44:59 Ditto
nickie 2016/09/09 09:42:26 Done.
1752 }
1753 return return_value;
1754 }
1755
1685 Statement* Parser::ParseFunctionDeclaration(bool* ok) { 1756 Statement* Parser::ParseFunctionDeclaration(bool* ok) {
1686 Consume(Token::FUNCTION); 1757 Consume(Token::FUNCTION);
1687 int pos = position(); 1758 int pos = position();
1688 ParseFunctionFlags flags = ParseFunctionFlags::kIsNormal; 1759 ParseFunctionFlags flags = ParseFunctionFlags::kIsNormal;
1689 if (Check(Token::MUL)) { 1760 if (Check(Token::MUL)) {
1690 flags |= ParseFunctionFlags::kIsGenerator; 1761 flags |= ParseFunctionFlags::kIsGenerator;
1691 if (allow_harmony_restrictive_declarations()) { 1762 if (allow_harmony_restrictive_declarations()) {
1692 ReportMessageAt(scanner()->location(), 1763 ReportMessageAt(scanner()->location(),
1693 MessageTemplate::kGeneratorInLegacyContext); 1764 MessageTemplate::kGeneratorInLegacyContext);
1694 *ok = false; 1765 *ok = false;
1695 return nullptr; 1766 return nullptr;
1696 } 1767 }
1697 } 1768 }
1698 1769
1699 return ParseHoistableDeclaration(pos, flags, nullptr, false, CHECK_OK); 1770 return ParseHoistableDeclaration(pos, flags, nullptr, false, CHECK_OK);
1700 } 1771 }
1701 1772
1702 Statement* Parser::ParseExpressionOrLabelledStatement(
1703 ZoneList<const AstRawString*>* labels,
1704 AllowLabelledFunctionStatement allow_function, bool* ok) {
1705 // ExpressionStatement | LabelledStatement ::
1706 // Expression ';'
1707 // Identifier ':' Statement
1708 //
1709 // ExpressionStatement[Yield] :
1710 // [lookahead ∉ {{, function, class, let [}] Expression[In, ?Yield] ;
1711
1712 int pos = peek_position();
1713
1714 switch (peek()) {
1715 case Token::FUNCTION:
1716 case Token::LBRACE:
1717 UNREACHABLE(); // Always handled by the callers.
1718 case Token::CLASS:
1719 ReportUnexpectedToken(Next());
1720 *ok = false;
1721 return nullptr;
1722 default:
1723 break;
1724 }
1725
1726 bool starts_with_idenfifier = peek_any_identifier();
1727 Expression* expr = ParseExpression(true, CHECK_OK);
1728 if (peek() == Token::COLON && starts_with_idenfifier && expr != NULL &&
1729 expr->AsVariableProxy() != NULL &&
1730 !expr->AsVariableProxy()->is_this()) {
1731 // Expression is a single identifier, and not, e.g., a parenthesized
1732 // identifier.
1733 VariableProxy* var = expr->AsVariableProxy();
1734 const AstRawString* label = var->raw_name();
1735 // TODO(1240780): We don't check for redeclaration of labels
1736 // during preparsing since keeping track of the set of active
1737 // labels requires nontrivial changes to the way scopes are
1738 // structured. However, these are probably changes we want to
1739 // make later anyway so we should go back and fix this then.
1740 if (ContainsLabel(labels, label) || TargetStackContainsLabel(label)) {
1741 ReportMessage(MessageTemplate::kLabelRedeclaration, label);
1742 *ok = false;
1743 return NULL;
1744 }
1745 if (labels == NULL) {
1746 labels = new(zone()) ZoneList<const AstRawString*>(4, zone());
1747 }
1748 labels->Add(label, zone());
1749 // Remove the "ghost" variable that turned out to be a label
1750 // from the top scope. This way, we don't try to resolve it
1751 // during the scope processing.
1752 scope()->RemoveUnresolved(var);
1753 Expect(Token::COLON, CHECK_OK);
1754 // ES#sec-labelled-function-declarations Labelled Function Declarations
1755 if (peek() == Token::FUNCTION && is_sloppy(language_mode())) {
1756 if (allow_function == kAllowLabelledFunctionStatement) {
1757 return ParseFunctionDeclaration(ok);
1758 } else {
1759 return ParseScopedStatement(labels, true, ok);
1760 }
1761 }
1762 return ParseStatement(labels, kDisallowLabelledFunctionStatement, ok);
1763 }
1764
1765 // If we have an extension, we allow a native function declaration.
1766 // A native function declaration starts with "native function" with
1767 // no line-terminator between the two words.
1768 if (extension_ != NULL && peek() == Token::FUNCTION &&
1769 !scanner()->HasAnyLineTerminatorBeforeNext() && expr != NULL &&
1770 expr->AsVariableProxy() != NULL &&
1771 expr->AsVariableProxy()->raw_name() ==
1772 ast_value_factory()->native_string() &&
1773 !scanner()->literal_contains_escapes()) {
1774 return ParseNativeDeclaration(ok);
1775 }
1776
1777 // Parsed expression statement, followed by semicolon.
1778 ExpectSemicolon(CHECK_OK);
1779 return factory()->NewExpressionStatement(expr, pos);
1780 }
1781
1782
1783 IfStatement* Parser::ParseIfStatement(ZoneList<const AstRawString*>* labels,
1784 bool* ok) {
1785 // IfStatement ::
1786 // 'if' '(' Expression ')' Statement ('else' Statement)?
1787
1788 int pos = peek_position();
1789 Expect(Token::IF, CHECK_OK);
1790 Expect(Token::LPAREN, CHECK_OK);
1791 Expression* condition = ParseExpression(true, CHECK_OK);
1792 Expect(Token::RPAREN, CHECK_OK);
1793 Statement* then_statement = ParseScopedStatement(labels, false, CHECK_OK);
1794 Statement* else_statement = NULL;
1795 if (peek() == Token::ELSE) {
1796 Next();
1797 else_statement = ParseScopedStatement(labels, false, CHECK_OK);
1798 } else {
1799 else_statement = factory()->NewEmptyStatement(kNoSourcePosition);
1800 }
1801 return factory()->NewIfStatement(
1802 condition, then_statement, else_statement, pos);
1803 }
1804
1805
1806 Statement* Parser::ParseContinueStatement(bool* ok) {
1807 // ContinueStatement ::
1808 // 'continue' Identifier? ';'
1809
1810 int pos = peek_position();
1811 Expect(Token::CONTINUE, CHECK_OK);
1812 const AstRawString* label = NULL;
1813 Token::Value tok = peek();
1814 if (!scanner()->HasAnyLineTerminatorBeforeNext() &&
1815 tok != Token::SEMICOLON && tok != Token::RBRACE && tok != Token::EOS) {
1816 // ECMA allows "eval" or "arguments" as labels even in strict mode.
1817 label = ParseIdentifier(kAllowRestrictedIdentifiers, CHECK_OK);
1818 }
1819 IterationStatement* target = LookupContinueTarget(label, CHECK_OK);
1820 if (target == NULL) {
1821 // Illegal continue statement.
1822 MessageTemplate::Template message = MessageTemplate::kIllegalContinue;
1823 if (label != NULL) {
1824 message = MessageTemplate::kUnknownLabel;
1825 }
1826 ReportMessage(message, label);
1827 *ok = false;
1828 return NULL;
1829 }
1830 ExpectSemicolon(CHECK_OK);
1831 return factory()->NewContinueStatement(target, pos);
1832 }
1833
1834
1835 Statement* Parser::ParseBreakStatement(ZoneList<const AstRawString*>* labels,
1836 bool* ok) {
1837 // BreakStatement ::
1838 // 'break' Identifier? ';'
1839
1840 int pos = peek_position();
1841 Expect(Token::BREAK, CHECK_OK);
1842 const AstRawString* label = NULL;
1843 Token::Value tok = peek();
1844 if (!scanner()->HasAnyLineTerminatorBeforeNext() &&
1845 tok != Token::SEMICOLON && tok != Token::RBRACE && tok != Token::EOS) {
1846 // ECMA allows "eval" or "arguments" as labels even in strict mode.
1847 label = ParseIdentifier(kAllowRestrictedIdentifiers, CHECK_OK);
1848 }
1849 // Parse labeled break statements that target themselves into
1850 // empty statements, e.g. 'l1: l2: l3: break l2;'
1851 if (label != NULL && ContainsLabel(labels, label)) {
1852 ExpectSemicolon(CHECK_OK);
1853 return factory()->NewEmptyStatement(pos);
1854 }
1855 BreakableStatement* target = NULL;
1856 target = LookupBreakTarget(label, CHECK_OK);
1857 if (target == NULL) {
1858 // Illegal break statement.
1859 MessageTemplate::Template message = MessageTemplate::kIllegalBreak;
1860 if (label != NULL) {
1861 message = MessageTemplate::kUnknownLabel;
1862 }
1863 ReportMessage(message, label);
1864 *ok = false;
1865 return NULL;
1866 }
1867 ExpectSemicolon(CHECK_OK);
1868 return factory()->NewBreakStatement(target, pos);
1869 }
1870
1871
1872 Statement* Parser::ParseReturnStatement(bool* ok) {
1873 // ReturnStatement ::
1874 // 'return' Expression? ';'
1875
1876 // Consume the return token. It is necessary to do that before
1877 // reporting any errors on it, because of the way errors are
1878 // reported (underlining).
1879 Expect(Token::RETURN, CHECK_OK);
1880 Scanner::Location loc = scanner()->location();
1881
1882 Token::Value tok = peek();
1883 Statement* result;
1884 Expression* return_value;
1885 if (scanner()->HasAnyLineTerminatorBeforeNext() ||
1886 tok == Token::SEMICOLON ||
1887 tok == Token::RBRACE ||
1888 tok == Token::EOS) {
1889 if (IsSubclassConstructor(function_state_->kind())) {
1890 return_value = ThisExpression(loc.beg_pos);
1891 } else {
1892 return_value = GetLiteralUndefined(position());
1893 }
1894 } else {
1895 int pos = peek_position();
1896
1897 if (IsSubclassConstructor(function_state_->kind())) {
1898 // Because of the return code rewriting that happens in case of a subclass
1899 // constructor we don't want to accept tail calls, therefore we don't set
1900 // ReturnExprScope to kInsideValidReturnStatement here.
1901 return_value = ParseExpression(true, CHECK_OK);
1902
1903 // For subclass constructors we need to return this in case of undefined
1904 // return a Smi (transformed into an exception in the ConstructStub)
1905 // for a non object.
1906 //
1907 // return expr;
1908 //
1909 // Is rewritten as:
1910 //
1911 // return (temp = expr) === undefined ? this :
1912 // %_IsJSReceiver(temp) ? temp : 1;
1913
1914 // temp = expr
1915 Variable* temp = NewTemporary(ast_value_factory()->empty_string());
1916 Assignment* assign = factory()->NewAssignment(
1917 Token::ASSIGN, factory()->NewVariableProxy(temp), return_value, pos);
1918
1919 // %_IsJSReceiver(temp)
1920 ZoneList<Expression*>* is_spec_object_args =
1921 new (zone()) ZoneList<Expression*>(1, zone());
1922 is_spec_object_args->Add(factory()->NewVariableProxy(temp), zone());
1923 Expression* is_spec_object_call = factory()->NewCallRuntime(
1924 Runtime::kInlineIsJSReceiver, is_spec_object_args, pos);
1925
1926 // %_IsJSReceiver(temp) ? temp : 1;
1927 Expression* is_object_conditional = factory()->NewConditional(
1928 is_spec_object_call, factory()->NewVariableProxy(temp),
1929 factory()->NewSmiLiteral(1, pos), pos);
1930
1931 // temp === undefined
1932 Expression* is_undefined = factory()->NewCompareOperation(
1933 Token::EQ_STRICT, assign,
1934 factory()->NewUndefinedLiteral(kNoSourcePosition), pos);
1935
1936 // is_undefined ? this : is_object_conditional
1937 return_value = factory()->NewConditional(
1938 is_undefined, ThisExpression(pos), is_object_conditional, pos);
1939 } else {
1940 ReturnExprScope maybe_allow_tail_calls(
1941 function_state_, ReturnExprContext::kInsideValidReturnStatement);
1942 return_value = ParseExpression(true, CHECK_OK);
1943
1944 if (allow_tailcalls() && !is_sloppy(language_mode()) && !is_resumable()) {
1945 // ES6 14.6.1 Static Semantics: IsInTailPosition
1946 function_state_->AddImplicitTailCallExpression(return_value);
1947 }
1948 }
1949 }
1950 ExpectSemicolon(CHECK_OK);
1951
1952 if (is_generator()) {
1953 return_value = BuildIteratorResult(return_value, true);
1954 } else if (is_async_function()) {
1955 return_value = BuildResolvePromise(return_value, return_value->position());
1956 }
1957
1958 result = factory()->NewReturnStatement(return_value, loc.beg_pos);
1959
1960 DeclarationScope* decl_scope = GetDeclarationScope();
1961 if (decl_scope->is_script_scope() || decl_scope->is_eval_scope()) {
1962 ReportMessageAt(loc, MessageTemplate::kIllegalReturn);
1963 *ok = false;
1964 return NULL;
1965 }
1966 return result;
1967 }
1968
1969
1970 Statement* Parser::ParseWithStatement(ZoneList<const AstRawString*>* labels,
1971 bool* ok) {
1972 // WithStatement ::
1973 // 'with' '(' Expression ')' Statement
1974
1975 Expect(Token::WITH, CHECK_OK);
1976 int pos = position();
1977
1978 if (is_strict(language_mode())) {
1979 ReportMessage(MessageTemplate::kStrictWith);
1980 *ok = false;
1981 return NULL;
1982 }
1983
1984 Expect(Token::LPAREN, CHECK_OK);
1985 Expression* expr = ParseExpression(true, CHECK_OK);
1986 Expect(Token::RPAREN, CHECK_OK);
1987
1988 Scope* with_scope = NewScope(WITH_SCOPE);
1989 Statement* body;
1990 {
1991 BlockState block_state(&scope_state_, with_scope);
1992 with_scope->set_start_position(scanner()->peek_location().beg_pos);
1993 body = ParseScopedStatement(labels, true, CHECK_OK);
1994 with_scope->set_end_position(scanner()->location().end_pos);
1995 }
1996 return factory()->NewWithStatement(with_scope, expr, body, pos);
1997 }
1998
1999
2000 CaseClause* Parser::ParseCaseClause(bool* default_seen_ptr, bool* ok) { 1773 CaseClause* Parser::ParseCaseClause(bool* default_seen_ptr, bool* ok) {
2001 // CaseClause :: 1774 // CaseClause ::
2002 // 'case' Expression ':' StatementList 1775 // 'case' Expression ':' StatementList
2003 // 'default' ':' StatementList 1776 // 'default' ':' StatementList
2004 1777
2005 Expression* label = NULL; // NULL expression indicates default case 1778 Expression* label = NULL; // NULL expression indicates default case
2006 if (peek() == Token::CASE) { 1779 if (peek() == Token::CASE) {
2007 Expect(Token::CASE, CHECK_OK); 1780 Expect(Token::CASE, CHECK_OK);
2008 label = ParseExpression(true, CHECK_OK); 1781 label = ParseExpression(true, CHECK_OK);
2009 } else { 1782 } else {
(...skipping 3997 matching lines...) Expand 10 before | Expand all | Expand 10 after
6007 node->Print(Isolate::Current()); 5780 node->Print(Isolate::Current());
6008 } 5781 }
6009 #endif // DEBUG 5782 #endif // DEBUG
6010 5783
6011 #undef CHECK_OK 5784 #undef CHECK_OK
6012 #undef CHECK_OK_VOID 5785 #undef CHECK_OK_VOID
6013 #undef CHECK_FAILED 5786 #undef CHECK_FAILED
6014 5787
6015 } // namespace internal 5788 } // namespace internal
6016 } // namespace v8 5789 } // namespace v8
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698