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

Side by Side Diff: src/parsing/preparser.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
« src/parsing/preparser.h ('K') | « src/parsing/preparser.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2011 the V8 project authors. All rights reserved. 1 // Copyright 2011 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 <cmath> 5 #include <cmath>
6 6
7 #include "src/allocation.h" 7 #include "src/allocation.h"
8 #include "src/base/logging.h" 8 #include "src/base/logging.h"
9 #include "src/conversions-inl.h" 9 #include "src/conversions-inl.h"
10 #include "src/conversions.h" 10 #include "src/conversions.h"
(...skipping 182 matching lines...) Expand 10 before | Expand all | Expand 10 after
193 if (allow_harmony_restrictive_declarations()) { 193 if (allow_harmony_restrictive_declarations()) {
194 ReportMessageAt(scanner()->location(), 194 ReportMessageAt(scanner()->location(),
195 MessageTemplate::kGeneratorInLegacyContext); 195 MessageTemplate::kGeneratorInLegacyContext);
196 *ok = false; 196 *ok = false;
197 return Statement::Default(); 197 return Statement::Default();
198 } 198 }
199 } 199 }
200 return ParseHoistableDeclaration(pos, flags, nullptr, false, ok); 200 return ParseHoistableDeclaration(pos, flags, nullptr, false, ok);
201 } 201 }
202 202
203 PreParser::Statement PreParser::ParseExpressionOrLabelledStatement(
204 ZoneList<const AstRawString*>* names,
205 AllowLabelledFunctionStatement allow_function, bool* ok) {
206 // ExpressionStatement | LabelledStatement ::
207 // Expression ';'
208 // Identifier ':' Statement
209
210 switch (peek()) {
211 case Token::FUNCTION:
212 case Token::LBRACE:
213 UNREACHABLE(); // Always handled by the callers.
214 case Token::CLASS:
215 ReportUnexpectedToken(Next());
216 *ok = false;
217 return Statement::Default();
218
219 default:
220 break;
221 }
222
223 bool starts_with_identifier = peek_any_identifier();
224 ExpressionClassifier classifier(this);
225 Expression expr = ParseExpressionCoverGrammar(true, CHECK_OK);
226 ValidateExpression(CHECK_OK);
227
228 // Even if the expression starts with an identifier, it is not necessarily an
229 // identifier. For example, "foo + bar" starts with an identifier but is not
230 // an identifier.
231 if (starts_with_identifier && expr.IsIdentifier() && peek() == Token::COLON) {
232 // Expression is a single identifier, and not, e.g., a parenthesized
233 // identifier.
234 DCHECK(!expr.AsIdentifier().IsEnum());
235 DCHECK(!parsing_module_ || !expr.AsIdentifier().IsAwait());
236 DCHECK(is_sloppy(language_mode()) ||
237 !IsFutureStrictReserved(expr.AsIdentifier()));
238 Consume(Token::COLON);
239 // ES#sec-labelled-function-declarations Labelled Function Declarations
240 if (peek() == Token::FUNCTION && is_sloppy(language_mode())) {
241 if (allow_function == kAllowLabelledFunctionStatement) {
242 return ParseFunctionDeclaration(ok);
243 } else {
244 return ParseScopedStatement(names, true, ok);
245 }
246 }
247 Statement statement =
248 ParseStatement(nullptr, kDisallowLabelledFunctionStatement, ok);
249 return statement.IsJumpStatement() ? Statement::Default() : statement;
250 // Preparsing is disabled for extensions (because the extension details
251 // aren't passed to lazily compiled functions), so we don't
252 // accept "native function" in the preparser.
253 }
254 // Parsed expression statement.
255 ExpectSemicolon(CHECK_OK);
256 return Statement::ExpressionStatement(expr);
257 }
258
259 PreParser::Statement PreParser::ParseIfStatement(
260 ZoneList<const AstRawString*>* labels, bool* ok) {
261 // IfStatement ::
262 // 'if' '(' Expression ')' Statement ('else' Statement)?
263
264 Expect(Token::IF, CHECK_OK);
265 Expect(Token::LPAREN, CHECK_OK);
266 ParseExpression(true, CHECK_OK);
267 Expect(Token::RPAREN, CHECK_OK);
268 Statement stat = ParseScopedStatement(labels, false, CHECK_OK);
269 if (peek() == Token::ELSE) {
270 Next();
271 Statement else_stat = ParseScopedStatement(labels, false, CHECK_OK);
272 stat = (stat.IsJumpStatement() && else_stat.IsJumpStatement()) ?
273 Statement::Jump() : Statement::Default();
274 } else {
275 stat = Statement::Default();
276 }
277 return stat;
278 }
279
280
281 PreParser::Statement PreParser::ParseContinueStatement(bool* ok) {
282 // ContinueStatement ::
283 // 'continue' [no line terminator] Identifier? ';'
284
285 Expect(Token::CONTINUE, CHECK_OK);
286 Token::Value tok = peek();
287 if (!scanner()->HasAnyLineTerminatorBeforeNext() &&
288 tok != Token::SEMICOLON &&
289 tok != Token::RBRACE &&
290 tok != Token::EOS) {
291 // ECMA allows "eval" or "arguments" as labels even in strict mode.
292 ParseIdentifier(kAllowRestrictedIdentifiers, CHECK_OK);
293 }
294 ExpectSemicolon(CHECK_OK);
295 return Statement::Jump();
296 }
297
298 PreParser::Statement PreParser::ParseBreakStatement(
299 ZoneList<const AstRawString*>* labels, bool* ok) {
300 // BreakStatement ::
301 // 'break' [no line terminator] Identifier? ';'
302
303 Expect(Token::BREAK, CHECK_OK);
304 Token::Value tok = peek();
305 if (!scanner()->HasAnyLineTerminatorBeforeNext() &&
306 tok != Token::SEMICOLON &&
307 tok != Token::RBRACE &&
308 tok != Token::EOS) {
309 // ECMA allows "eval" or "arguments" as labels even in strict mode.
310 ParseIdentifier(kAllowRestrictedIdentifiers, CHECK_OK);
311 }
312 ExpectSemicolon(CHECK_OK);
313 return Statement::Jump();
314 }
315
316
317 PreParser::Statement PreParser::ParseReturnStatement(bool* ok) {
318 // ReturnStatement ::
319 // 'return' [no line terminator] Expression? ';'
320
321 // Consume the return token. It is necessary to do before
322 // reporting any errors on it, because of the way errors are
323 // reported (underlining).
324 Expect(Token::RETURN, CHECK_OK);
325
326 // An ECMAScript program is considered syntactically incorrect if it
327 // contains a return statement that is not within the body of a
328 // function. See ECMA-262, section 12.9, page 67.
329 // This is not handled during preparsing.
330
331 Token::Value tok = peek();
332 if (!scanner()->HasAnyLineTerminatorBeforeNext() &&
333 tok != Token::SEMICOLON &&
334 tok != Token::RBRACE &&
335 tok != Token::EOS) {
336 // Because of the return code rewriting that happens in case of a subclass
337 // constructor we don't want to accept tail calls, therefore we don't set
338 // ReturnExprScope to kInsideValidReturnStatement here.
339 ReturnExprContext return_expr_context =
340 IsSubclassConstructor(function_state_->kind())
341 ? function_state_->return_expr_context()
342 : ReturnExprContext::kInsideValidReturnStatement;
343
344 ReturnExprScope maybe_allow_tail_calls(function_state_,
345 return_expr_context);
346 ParseExpression(true, CHECK_OK);
347 }
348 ExpectSemicolon(CHECK_OK);
349 return Statement::Jump();
350 }
351
352 PreParser::Statement PreParser::ParseWithStatement(
353 ZoneList<const AstRawString*>* labels, bool* ok) {
354 // WithStatement ::
355 // 'with' '(' Expression ')' Statement
356 Expect(Token::WITH, CHECK_OK);
357 if (is_strict(language_mode())) {
358 ReportMessageAt(scanner()->location(), MessageTemplate::kStrictWith);
359 *ok = false;
360 return Statement::Default();
361 }
362 Expect(Token::LPAREN, CHECK_OK);
363 ParseExpression(true, CHECK_OK);
364 Expect(Token::RPAREN, CHECK_OK);
365
366 Scope* with_scope = NewScope(WITH_SCOPE);
367 BlockState block_state(&scope_state_, with_scope);
368 ParseScopedStatement(labels, true, CHECK_OK);
369 return Statement::Default();
370 }
371
372 PreParser::Statement PreParser::ParseSwitchStatement( 203 PreParser::Statement PreParser::ParseSwitchStatement(
373 ZoneList<const AstRawString*>* labels, bool* ok) { 204 ZoneList<const AstRawString*>* labels, bool* ok) {
374 // SwitchStatement :: 205 // SwitchStatement ::
375 // 'switch' '(' Expression ')' '{' CaseClause* '}' 206 // 'switch' '(' Expression ')' '{' CaseClause* '}'
376 207
377 Expect(Token::SWITCH, CHECK_OK); 208 Expect(Token::SWITCH, CHECK_OK);
378 Expect(Token::LPAREN, CHECK_OK); 209 Expect(Token::LPAREN, CHECK_OK);
379 ParseExpression(true, CHECK_OK); 210 ParseExpression(true, CHECK_OK);
380 Expect(Token::RPAREN, CHECK_OK); 211 Expect(Token::RPAREN, CHECK_OK);
381 212
(...skipping 454 matching lines...) Expand 10 before | Expand all | Expand 10 after
836 667
837 body->Add(PreParserStatement::ExpressionStatement(return_value), zone()); 668 body->Add(PreParserStatement::ExpressionStatement(return_value), zone());
838 } 669 }
839 670
840 #undef CHECK_OK 671 #undef CHECK_OK
841 #undef CHECK_OK_CUSTOM 672 #undef CHECK_OK_CUSTOM
842 673
843 674
844 } // namespace internal 675 } // namespace internal
845 } // namespace v8 676 } // namespace v8
OLDNEW
« src/parsing/preparser.h ('K') | « src/parsing/preparser.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698