Skip to content

Commit

Permalink
Reduce allocations from creating MissingTokenWithTrivia objects (#75404)
Browse files Browse the repository at this point in the history
* Reduce allocations from creating MissingTokenWithTrivia objects

This method accounts for 0.7% of allocations in the roslyn csharp editing speedometer test profile I'm looking at. In addition to caching tokens < LastTokenWithWellKnownText similar to what is already done, the SyntaxKind.IdentifierToken case is handled as it's very commonly hit too. Local testing of opening LanguageParser and typing in a new method reduced the number of MissingTokenWithTrivia from over 1 million to zero.
  • Loading branch information
ToddGrun authored Oct 8, 2024
1 parent c625f52 commit a656d1a
Show file tree
Hide file tree
Showing 2 changed files with 20 additions and 2 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -146,7 +146,7 @@ internal static SyntaxToken Token(GreenNode leading, SyntaxKind kind, string tex

internal static SyntaxToken MissingToken(SyntaxKind kind)
{
return SyntaxToken.CreateMissing(kind, null, null);
return SyntaxToken.CreateMissing(kind);
}

internal static SyntaxToken MissingToken(GreenNode leading, SyntaxKind kind, GreenNode trailing)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -76,7 +76,7 @@ internal static SyntaxToken Create(SyntaxKind kind)
throw new ArgumentException(string.Format(CSharpResources.ThisMethodCanOnlyBeUsedToCreateTokens, kind), nameof(kind));
}

return CreateMissing(kind, null, null);
return CreateMissing(kind);
}

return s_tokensWithNoTrivia[(int)kind].Value;
Expand Down Expand Up @@ -118,6 +118,20 @@ internal static SyntaxToken Create(SyntaxKind kind, GreenNode leading, GreenNode
return new SyntaxTokenWithTrivia(kind, leading, trailing);
}

internal static SyntaxToken CreateMissing(SyntaxKind kind)
{
if (kind <= LastTokenWithWellKnownText)
{
return s_missingTokensWithNoTrivia[(int)kind].Value;
}
else if (kind == SyntaxKind.IdentifierToken)
{
return s_missingIdentifierTokenWithNoTrivia;
}

return new MissingTokenWithTrivia(kind, leading: null, trailing: null);
}

internal static SyntaxToken CreateMissing(SyntaxKind kind, GreenNode leading, GreenNode trailing)
{
return new MissingTokenWithTrivia(kind, leading, trailing);
Expand All @@ -131,6 +145,9 @@ internal static SyntaxToken CreateMissing(SyntaxKind kind, GreenNode leading, Gr
private static readonly ArrayElement<SyntaxToken>[] s_tokensWithElasticTrivia = new ArrayElement<SyntaxToken>[(int)LastTokenWithWellKnownText + 1];
private static readonly ArrayElement<SyntaxToken>[] s_tokensWithSingleTrailingSpace = new ArrayElement<SyntaxToken>[(int)LastTokenWithWellKnownText + 1];
private static readonly ArrayElement<SyntaxToken>[] s_tokensWithSingleTrailingCRLF = new ArrayElement<SyntaxToken>[(int)LastTokenWithWellKnownText + 1];
private static readonly ArrayElement<SyntaxToken>[] s_missingTokensWithNoTrivia = new ArrayElement<SyntaxToken>[(int)LastTokenWithWellKnownText + 1];

private static readonly SyntaxToken s_missingIdentifierTokenWithNoTrivia = new MissingTokenWithTrivia(SyntaxKind.IdentifierToken, leading: null, trailing: null);

static SyntaxToken()
{
Expand All @@ -140,6 +157,7 @@ static SyntaxToken()
s_tokensWithElasticTrivia[(int)kind].Value = new SyntaxTokenWithTrivia(kind, SyntaxFactory.ElasticZeroSpace, SyntaxFactory.ElasticZeroSpace);
s_tokensWithSingleTrailingSpace[(int)kind].Value = new SyntaxTokenWithTrivia(kind, null, SyntaxFactory.Space);
s_tokensWithSingleTrailingCRLF[(int)kind].Value = new SyntaxTokenWithTrivia(kind, null, SyntaxFactory.CarriageReturnLineFeed);
s_missingTokensWithNoTrivia[(int)kind].Value = new MissingTokenWithTrivia(kind, leading: null, trailing: null);
}
}

Expand Down

0 comments on commit a656d1a

Please sign in to comment.