-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[parser] Ensure that list and map pattern parsing always makes progress.
In rare circumstances involving syntax errors, the `parsePattern` method inserts a synthetic token but does not consume any tokens. Usually when this happens it's not a problem, because whatever method is calling `parsePattern` consumes some tokens, so the parser always makes progress. However, when parsing list patterns, after calling `parsePattern`, the parser would look for a `,`, and if it didn't find one, it would supply a synthetic `,` and call `parsePattern` again, resulting in an infinite loop. A similar situation happened with map patterns, though the situation was more complex because in between the calls to `parsePattern`, the parser would also create synthetic key expressions and `:`s. To fix the problem, when parsing a list or map pattern, after the call to `parsePattern`, the parser checks whether any tokens were consumed. If no tokens were consumed, it ignores the next token from the input stream in order to make progress. I also investigated whether there were similar issues with parenthesized/record patterns and switch expressions, since those constructs also consist of a sequence of patterns separated by tokens and other things that could in principle be supplied synthetically. Fortunately, parser recovery doesn't get into an infinite loop in those cases, so I didn't make any further changes. But I did include test cases to make sure. Fixes #52352. Bug: #52352 Change-Id: Idc8140236f6054deb1fd3c862036fe47dd84f30b Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/302803 Reviewed-by: Konstantin Shcheglov <[email protected]> Commit-Queue: Paul Berry <[email protected]>
- Loading branch information
1 parent
48346c4
commit 8d2b2a1
Showing
27 changed files
with
1,339 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
5 changes: 5 additions & 0 deletions
5
pkg/front_end/parser_testcases/patterns/syntheticIdentifier_insideListPattern.dart
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
void f(Object? x) { | ||
switch (x) { | ||
case [if]: | ||
}; | ||
} |
59 changes: 59 additions & 0 deletions
59
pkg/front_end/parser_testcases/patterns/syntheticIdentifier_insideListPattern.dart.expect
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,59 @@ | ||
Problems reported: | ||
|
||
parser/patterns/syntheticIdentifier_insideListPattern:3:11: Expected an identifier, but got 'if'. | ||
case [if]: | ||
^^ | ||
|
||
beginCompilationUnit(void) | ||
beginMetadataStar(void) | ||
endMetadataStar(0) | ||
beginTopLevelMember(void) | ||
beginTopLevelMethod(, null, null) | ||
handleVoidKeyword(void) | ||
handleIdentifier(f, topLevelFunctionDeclaration) | ||
handleNoTypeVariables(() | ||
beginFormalParameters((, MemberKind.TopLevelMethod) | ||
beginMetadataStar(Object) | ||
endMetadataStar(0) | ||
beginFormalParameter(Object, MemberKind.TopLevelMethod, null, null, null) | ||
handleIdentifier(Object, typeReference) | ||
handleNoTypeArguments(?) | ||
handleType(Object, ?) | ||
handleIdentifier(x, formalParameterDeclaration) | ||
handleFormalParameterWithoutValue()) | ||
endFormalParameter(null, null, null, x, null, null, FormalParameterKind.requiredPositional, MemberKind.TopLevelMethod) | ||
endFormalParameters(1, (, ), MemberKind.TopLevelMethod) | ||
handleAsyncModifier(null, null) | ||
beginBlockFunctionBody({) | ||
beginSwitchStatement(switch) | ||
handleIdentifier(x, expression) | ||
handleNoTypeArguments()) | ||
handleNoArguments()) | ||
handleSend(x, )) | ||
handleParenthesizedCondition((, null, null) | ||
beginSwitchBlock({) | ||
beginCaseExpression(case) | ||
beginPattern(case) | ||
handleNoTypeArguments([) | ||
beginPattern([) | ||
beginConstantPattern(null) | ||
handleRecoverableError(Message[ExpectedIdentifier, Expected an identifier, but got 'if'., Try inserting an identifier before 'if'., {lexeme: if}], if, if) | ||
handleIdentifier(, expression) | ||
handleNoTypeArguments(if) | ||
handleNoArguments(if) | ||
handleSend(, if) | ||
endConstantPattern(null) | ||
endPattern() | ||
handleListPattern(1, [, ]) | ||
endPattern(]) | ||
handleSwitchCaseNoWhenClause(]) | ||
endCaseExpression(case, null, :) | ||
beginSwitchCase(0, 1, case) | ||
endSwitchCase(0, 1, null, null, 0, case, }) | ||
endSwitchBlock(1, {, }) | ||
endSwitchStatement(switch, }) | ||
handleEmptyStatement(;) | ||
endBlockFunctionBody(2, {, }) | ||
endTopLevelMethod(void, null, }) | ||
endTopLevelDeclaration() | ||
endCompilationUnit(1, ) |
119 changes: 119 additions & 0 deletions
119
...d/parser_testcases/patterns/syntheticIdentifier_insideListPattern.dart.intertwined.expect
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
parseUnit(void) | ||
skipErrorTokens(void) | ||
listener: beginCompilationUnit(void) | ||
syntheticPreviousToken(void) | ||
parseTopLevelDeclarationImpl(, Instance of 'DirectiveContext') | ||
parseMetadataStar() | ||
listener: beginMetadataStar(void) | ||
listener: endMetadataStar(0) | ||
parseTopLevelMemberImpl() | ||
listener: beginTopLevelMember(void) | ||
parseTopLevelMethod(, null, null, , Instance of 'VoidType', null, f, false) | ||
listener: beginTopLevelMethod(, null, null) | ||
listener: handleVoidKeyword(void) | ||
ensureIdentifierPotentiallyRecovered(void, topLevelFunctionDeclaration, false) | ||
listener: handleIdentifier(f, topLevelFunctionDeclaration) | ||
parseMethodTypeVar(f) | ||
listener: handleNoTypeVariables(() | ||
parseGetterOrFormalParameters(f, f, false, MemberKind.TopLevelMethod) | ||
parseFormalParameters(f, MemberKind.TopLevelMethod) | ||
parseFormalParametersRest((, MemberKind.TopLevelMethod) | ||
listener: beginFormalParameters((, MemberKind.TopLevelMethod) | ||
parseFormalParameter((, FormalParameterKind.requiredPositional, MemberKind.TopLevelMethod) | ||
parseMetadataStar(() | ||
listener: beginMetadataStar(Object) | ||
listener: endMetadataStar(0) | ||
listener: beginFormalParameter(Object, MemberKind.TopLevelMethod, null, null, null) | ||
listener: handleIdentifier(Object, typeReference) | ||
listener: handleNoTypeArguments(?) | ||
listener: handleType(Object, ?) | ||
ensureIdentifier(?, formalParameterDeclaration) | ||
listener: handleIdentifier(x, formalParameterDeclaration) | ||
listener: handleFormalParameterWithoutValue()) | ||
listener: endFormalParameter(null, null, null, x, null, null, FormalParameterKind.requiredPositional, MemberKind.TopLevelMethod) | ||
listener: endFormalParameters(1, (, ), MemberKind.TopLevelMethod) | ||
parseAsyncModifierOpt()) | ||
listener: handleAsyncModifier(null, null) | ||
inPlainSync() | ||
parseFunctionBody(), false, false) | ||
listener: beginBlockFunctionBody({) | ||
notEofOrValue(}, switch) | ||
parseStatement({) | ||
parseStatementX({) | ||
parseSwitchStatement({) | ||
listener: beginSwitchStatement(switch) | ||
ensureParenthesizedCondition(switch, allowCase: false) | ||
parseExpressionInParenthesisRest((, allowCase: false) | ||
parseExpression(() | ||
looksLikeOuterPatternEquals(() | ||
skipOuterPattern(() | ||
skipObjectPatternRest(x) | ||
parsePrecedenceExpression((, 1, true, ConstantPatternContext.none) | ||
parseUnaryExpression((, true, ConstantPatternContext.none) | ||
parsePrimary((, expression, ConstantPatternContext.none) | ||
parseSendOrFunctionLiteral((, expression, ConstantPatternContext.none) | ||
parseSend((, expression, ConstantPatternContext.none) | ||
isNextIdentifier(() | ||
ensureIdentifier((, expression) | ||
listener: handleIdentifier(x, expression) | ||
listener: handleNoTypeArguments()) | ||
parseArgumentsOpt(x) | ||
listener: handleNoArguments()) | ||
listener: handleSend(x, )) | ||
ensureCloseParen(x, () | ||
listener: handleParenthesizedCondition((, null, null) | ||
parseSwitchBlock()) | ||
ensureBlock(), null, switch statement) | ||
listener: beginSwitchBlock({) | ||
notEofOrValue(}, case) | ||
peekPastLabels(case) | ||
listener: beginCaseExpression(case) | ||
parsePattern(case, PatternContext.matching, precedence: 1) | ||
listener: beginPattern(case) | ||
parsePrimaryPattern(case, PatternContext.matching) | ||
listener: handleNoTypeArguments([) | ||
parseListPatternSuffix(case, PatternContext.matching) | ||
parsePattern([, PatternContext.matching, precedence: 1) | ||
listener: beginPattern([) | ||
parsePrimaryPattern([, PatternContext.matching) | ||
listener: beginConstantPattern(null) | ||
parsePrecedenceExpression([, 7, false, ConstantPatternContext.implicit) | ||
parseUnaryExpression([, false, ConstantPatternContext.implicit) | ||
parsePrimary([, expression, ConstantPatternContext.implicit) | ||
inPlainSync() | ||
parseSend([, expression, ConstantPatternContext.implicit) | ||
isNextIdentifier([) | ||
ensureIdentifier([, expression) | ||
reportRecoverableErrorWithToken(if, Instance of 'Template<(Token) => Message>') | ||
listener: handleRecoverableError(Message[ExpectedIdentifier, Expected an identifier, but got 'if'., Try inserting an identifier before 'if'., {lexeme: if}], if, if) | ||
rewriter() | ||
listener: handleIdentifier(, expression) | ||
listener: handleNoTypeArguments(if) | ||
parseArgumentsOpt() | ||
listener: handleNoArguments(if) | ||
listener: handleSend(, if) | ||
listener: endConstantPattern(null) | ||
listener: endPattern() | ||
listener: handleListPattern(1, [, ]) | ||
listener: endPattern(]) | ||
listener: handleSwitchCaseNoWhenClause(]) | ||
ensureColon(]) | ||
listener: endCaseExpression(case, null, :) | ||
peekPastLabels(}) | ||
parseStatementsInSwitchCase(:, }, case, 0, 1, null, null) | ||
listener: beginSwitchCase(0, 1, case) | ||
listener: endSwitchCase(0, 1, null, null, 0, case, }) | ||
notEofOrValue(}, }) | ||
listener: endSwitchBlock(1, {, }) | ||
listener: endSwitchStatement(switch, }) | ||
notEofOrValue(}, ;) | ||
parseStatement(}) | ||
parseStatementX(}) | ||
parseEmptyStatement(}) | ||
listener: handleEmptyStatement(;) | ||
notEofOrValue(}, }) | ||
listener: endBlockFunctionBody(2, {, }) | ||
listener: endTopLevelMethod(void, null, }) | ||
listener: endTopLevelDeclaration() | ||
reportAllErrorTokens(void) | ||
listener: endCompilationUnit(1, ) |
15 changes: 15 additions & 0 deletions
15
...nt_end/parser_testcases/patterns/syntheticIdentifier_insideListPattern.dart.parser.expect
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
NOTICE: Stream was rewritten by parser! | ||
|
||
void f(Object? x) { | ||
switch (x) { | ||
case [*synthetic*if]: | ||
}; | ||
} | ||
|
||
|
||
void[KeywordToken] f[StringToken]([BeginToken]Object[StringToken]?[SimpleToken] x[StringToken])[SimpleToken] {[BeginToken] | ||
switch[KeywordToken] ([BeginToken]x[StringToken])[SimpleToken] {[BeginToken] | ||
case[KeywordToken] [[BeginToken][SyntheticStringToken]if[KeywordToken]][SimpleToken]:[SimpleToken] | ||
}[SimpleToken];[SimpleToken] | ||
}[SimpleToken] | ||
[SimpleToken] |
Oops, something went wrong.