-
Notifications
You must be signed in to change notification settings - Fork 4.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Extensions parsing #76867
base: features/extensions
Are you sure you want to change the base?
Extensions parsing #76867
Conversation
@@ -81,6 +81,7 @@ member_declaration | |||
| base_type_declaration | |||
| delegate_declaration | |||
| enum_member_declaration | |||
| extension_container |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
don't love that it's not called _declaration. #Resolved
@@ -348,6 +349,14 @@ delegate_declaration | |||
: attribute_list* modifier* 'delegate' type identifier_token type_parameter_list? parameter_list type_parameter_constraint_clause* ';' | |||
; | |||
|
|||
extension_container | |||
: attribute_list* modifier* syntax_token type_parameter_list? '(' receiver_parameter ')' type_parameter_constraint_clause* '{' member_declaration* '}' |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
syntax_token? not 'extension'
? #Resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: allow an identifier. people will try to write it, and it will allow for better error recovery.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
nit: allow a parameter_list, as people will write more, and it will enable better error recovery.
; | ||
|
||
receiver_parameter | ||
: attribute_list* modifier* type identifier_token? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
we should allow a parameter_syntax. it will make it much easier ot recover and represent throughout the system. (similar to how even a => a
uses a parameter_syntax).
overfitting syntax is often problematic. both for normal typing cases, and for expanding on this in the future. #Resolved
@@ -1732,7 +1732,7 @@ private TypeDeclarationSyntax ParseClassOrStructOrInterfaceDeclaration(SyntaxLis | |||
_termState |= TerminatorState.IsEndOfRecordOrClassOrStructOrInterfaceSignature; | |||
|
|||
var saveTerm = _termState; | |||
_termState |= TerminatorState.IsPossibleAggregateClauseStartOrStop; | |||
//_termState |= TerminatorState.IsPossibleAggregateClauseStartOrStop; // TODO2 looking for affected test |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
mark as prototype? so this doesn't get merged in? #Resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This PR is draft. Not ready for review. TODO2 comments are flagged by our build and are useful to track things that need to be handled in the PR ;-)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
oh. gmail didn't mark this as draft. sorry!
gtk about TODO2. i didn't realize that.
@@ -1962,6 +1963,151 @@ static TypeDeclarationSyntax constructTypeDeclaration(ContextAwareSyntax syntaxF | |||
} | |||
} | |||
|
|||
private MemberDeclarationSyntax ParseExtensionContainer(SyntaxList<AttributeListSyntax> attributes, SyntaxListBuilder modifiers) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
make return type strongly typed. #Resolved
var outerSaveTerm = _termState; | ||
_termState |= TerminatorState.IsPossibleAggregateClauseStartOrStop; // TODO2 add test affected | ||
|
||
TypeParameterListSyntax typeParameters = this.ParseTypeParameterList(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
check for identifier and recover gracefully. #Resolved
_termState |= TerminatorState.IsPossibleAggregateClauseStartOrStop; // TODO2 add test affected | ||
|
||
TypeParameterListSyntax typeParameters = this.ParseTypeParameterList(); | ||
ReceiverParameterSyntax receiverParameter = parseReceiverParameter(out var openParenToken, out var closeParenToken); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
parse a parameter list, with a requirement that at least a single parameter is required. This can be done by updating ParseParameterList to take a parameter stating if a single parameter is required (and passing that to hte hlper it calls).
you can optionally error at parse time or binding time if there is >1 parameter. I recommend binding time. #Resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
By requiring a single parameter, you can be sure in later phases that it is safe to access [0]
_termState |= TerminatorState.IsEndOfRecordOrClassOrStructOrInterfaceSignature; | ||
constraints = _pool.Allocate<TypeParameterConstraintClauseSyntax>(); | ||
this.ParseTypeParameterConstraintClauses(constraints); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
extract this into helper. otehrwise it can get out of sync with type parameter parsing elsewhere. #Resolved
parseMembers = false; | ||
} | ||
|
||
if (parseMembers) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
having a variable to just expand on if (!openBrace.IsMissing)
seems excessive. #Resolved
|
||
var modifiersList = (SyntaxList<SyntaxToken>)modifiers.ToList(); | ||
var membersList = (SyntaxList<MemberDeclarationSyntax>)members; | ||
var constraintsList = (SyntaxList<TypeParameterConstraintClauseSyntax>)constraints; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this is confusing. we have the .ToListAndFree helpers. Why not use those? then you don't need a try/finally either. #Resolved
} | ||
} | ||
|
||
if (openBrace.IsMissing) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
here you check the openBrace, instead of parseMembers #Resolved
{ | ||
members = _pool.Allocate<MemberDeclarationSyntax>(); | ||
|
||
while (true) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
afaict, this is a copy of ParseClassOrStructOrInterfaceDeclaration. Seems like we can just expand that to parse extensions, just with some extension-specific code. Note: as all teh rest support the primary cosntructor parameter list, this all falls out. The only difference that i can tell so far is simple that in the case of extensions we want at least one parameter (though this could be a binding-time check), and that we do not have an identifier. All of that seems similar to add to ParseClassOrStructOrInterfaceDeclaration vs this copy :) #Resolved
SyntaxToken? identifier; | ||
if (this.CurrentToken.Kind == SyntaxKind.CloseParenToken) | ||
{ | ||
identifier = null; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
ah, you want identifiers to be optional. mixed feelings. but that feels very doable still :) #Resolved
@@ -43,5 +43,6 @@ public enum CompilerFeature | |||
RecordStructs, | |||
RequiredMembers, | |||
RefLifetime, | |||
Extensions, | |||
} | |||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
incremental tests that changing class C(int i)
to extension(int i)
and vvice versa works would be good.
#Resolved
Overall, this looks quite good to me. My pref is to share more int eh parser. either higher level parsing functinos, or lower level shared routines :) #Resolved |
15cf64b
to
8e29f61
Compare
Done with review pass (commit 9) |
I think this may end up allowed, since emitted as static field. In reply to: 2611226090 Refers to: src/Compilers/CSharp/Test/Syntax/Parsing/ExtensionsParsingTests.cs:1713 in 51686c8. [](commit_id = 51686c8, deletion_comment = False) |
The spec doesn't allow fields and we didn't even think what would it mean for implementation. Also, is this going to be an extension field? In reply to: 2611341439 Refers to: src/Compilers/CSharp/Test/Syntax/Parsing/ExtensionsParsingTests.cs:1713 in 51686c8. [](commit_id = 51686c8, deletion_comment = False) |
I suggest sticking to the spec for now In reply to: 2611344424 Refers to: src/Compilers/CSharp/Test/Syntax/Parsing/ExtensionsParsingTests.cs:1713 in 51686c8. [](commit_id = 51686c8, deletion_comment = False) |
This may be allowed (TBD) In reply to: 2611227756 Refers to: src/Compilers/CSharp/Test/Syntax/Parsing/ExtensionsParsingTests.cs:1921 in 51686c8. [](commit_id = 51686c8, deletion_comment = False) |
I suggest sticking to the spec for now In reply to: 2611345773 Refers to: src/Compilers/CSharp/Test/Syntax/Parsing/ExtensionsParsingTests.cs:1921 in 51686c8. [](commit_id = 51686c8, deletion_comment = False) |
@jjonescz for second review. Thanks |
<Field Name="AttributeLists" Type="SyntaxList<AttributeListSyntax>" Override="true"/> | ||
<Field Name="Modifiers" Type="SyntaxList<SyntaxToken>" Override="true"/> | ||
<Field Name="Keyword" Type="SyntaxToken" Override="true"> | ||
<ContextualKind Name="ExtensionKeyword"/> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
i'm actually slightly surprised by this. i would have thought this would be <kind
#Resolved
<Field Name="CloseBraceToken" Type="SyntaxToken" Override="true" Optional="true"> | ||
<Kind Name="CloseBraceToken"/> | ||
</Field> | ||
<Field Name="SemicolonToken" Type="SyntaxToken" Optional="true" Override="true"> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why is the semicolon token allowed here? It's not in the speclet. #Resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I kept it for consistency and simplicity with the other declaration kinds. Will update the speclet.
|
||
default: | ||
throw ExceptionUtilities.UnexpectedValue(this.CurrentToken.Kind); | ||
} | ||
} | ||
|
||
private TypeDeclarationSyntax ParseClassOrStructOrInterfaceDeclaration(SyntaxList<AttributeListSyntax> attributes, SyntaxListBuilder modifiers) | ||
private TypeDeclarationSyntax ParseClassOrStructOrInterfaceOrExtensionDeclaration(SyntaxList<AttributeListSyntax> attributes, SyntaxListBuilder modifiers) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks like this could be renamed to ParseTypeDeclaration as well #Resolved
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That name would conflict. Shortened as ParseMainTypeDeclaration
public override SyntaxToken Identifier => default; | ||
|
||
internal override BaseTypeDeclarationSyntax WithIdentifierCore(SyntaxToken identifier) | ||
=> throw new System.NotImplementedException(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am not sure why this thread was marked as resolved.
// as all normal parameters are legal extension parameters. | ||
if (parameter.Identifier.Kind() == SyntaxKind.None) | ||
{ | ||
return false; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It looks like there are legitimate test failures in IOperation leg. #Closed |
Done with review pass (commit 13). |
public partial class ExtensionDeclarationSyntax | ||
{ | ||
public override SyntaxToken Identifier | ||
=> throw new System.NotImplementedException(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It is, however, Ok to throw when one tries to set identifier.
=> throw new System.NotImplementedException(); | ||
|
||
public override BaseListSyntax? BaseList | ||
=> throw new System.NotImplementedException(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Done with review pass (commit 17) |
// We can only reuse parameters without identifiers (found in extension declarations) in context that allow optional identifiers. | ||
// The reverse is fine though. Normal parameters (from non extensions) can be re-used into an extension declaration | ||
// as all normal parameters are legal extension parameters. | ||
if (!allowOptionalIdentifier && parameter.Identifier.Kind() == SyntaxKind.None) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Relates to test plan #76130