diff --git a/ExternalProjects/BizHawk.Analyzer/HawkSourceAnalyzer.cs b/ExternalProjects/BizHawk.Analyzer/HawkSourceAnalyzer.cs index 33eeda8dff..c5e38c007b 100644 --- a/ExternalProjects/BizHawk.Analyzer/HawkSourceAnalyzer.cs +++ b/ExternalProjects/BizHawk.Analyzer/HawkSourceAnalyzer.cs @@ -1,6 +1,5 @@ namespace BizHawk.Analyzers; -using System; using System.Collections.Immutable; using Microsoft.CodeAnalysis; @@ -66,9 +65,16 @@ public class HawkSourceAnalyzer : DiagnosticAnalyzer { context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None); context.EnableConcurrentExecution(); + INamedTypeSymbol? invalidOperationExceptionSym = null; + INamedTypeSymbol? switchExpressionExceptionSym = null; context.RegisterSyntaxNodeAction( - static snac => + snac => { + if (invalidOperationExceptionSym is null) + { + invalidOperationExceptionSym = snac.Compilation.GetTypeByMetadataName("System.InvalidOperationException")!; + switchExpressionExceptionSym = snac.Compilation.GetTypeByMetadataName("System.Runtime.CompilerServices.SwitchExpressionException"); + } switch (snac.Node) { case AnonymousMethodExpressionSyntax: @@ -84,21 +90,9 @@ public class HawkSourceAnalyzer : DiagnosticAnalyzer snac.ReportDiagnostic(Diagnostic.Create(DiagNoQueryExpression, snac.Node.GetLocation())); break; case SwitchExpressionArmSyntax { WhenClause: null, Pattern: DiscardPatternSyntax, Expression: ThrowExpressionSyntax tes }: - if (tes.Expression is ObjectCreationExpressionSyntax oces) + var thrownExceptionType = snac.SemanticModel.GetThrownExceptionType(tes); + if (thrownExceptionType is null) { - // to resolve edge-cases involving aliases, you're supposed to use `snac.SemanticModel.GetTypeInfo(oces.Type).ConvertedType?.Name`, but I couldn't get it to work - if (((oces.Type as IdentifierNameSyntax)?.Identifier)?.ToString() is "SwitchExpressionException" or nameof(InvalidOperationException)) - { - // correct usage, do not flag - } - else - { - snac.ReportDiagnostic(Diagnostic.Create(DiagSwitchShouldThrowIOE, tes.GetLocation(), ERR_MSG_SWITCH_THROWS_WRONG_TYPE)); - } - } - else - { - // code reads `throw ` snac.ReportDiagnostic(Diagnostic.Create( DiagSwitchShouldThrowIOE, tes.GetLocation(), @@ -107,6 +101,11 @@ public class HawkSourceAnalyzer : DiagnosticAnalyzer properties: null, ERR_MSG_SWITCH_THROWS_UNKNOWN)); } + else if (!invalidOperationExceptionSym.Matches(thrownExceptionType) && switchExpressionExceptionSym?.Matches(thrownExceptionType) != true) + { + snac.ReportDiagnostic(Diagnostic.Create(DiagSwitchShouldThrowIOE, tes.GetLocation(), ERR_MSG_SWITCH_THROWS_WRONG_TYPE)); + } + // else correct usage, do not flag break; } }, diff --git a/ExternalProjects/BizHawk.Analyzer/RoslynUtils.cs b/ExternalProjects/BizHawk.Analyzer/RoslynUtils.cs new file mode 100644 index 0000000000..deaa57b34d --- /dev/null +++ b/ExternalProjects/BizHawk.Analyzer/RoslynUtils.cs @@ -0,0 +1,18 @@ +namespace BizHawk.Analyzers; + +using Microsoft.CodeAnalysis; +using Microsoft.CodeAnalysis.CSharp.Syntax; + +internal static class RoslynUtils +{ + private static ITypeSymbol? GetThrownExceptionType(this SemanticModel model, ExpressionSyntax exprSyn) + => exprSyn is ObjectCreationExpressionSyntax oces + ? model.GetTypeInfo(exprSyn).Type + : null; // code reads `throw ` + + public static ITypeSymbol? GetThrownExceptionType(this SemanticModel model, ThrowExpressionSyntax tes) + => model.GetThrownExceptionType(tes.Expression); + + public static bool Matches(this ITypeSymbol expected, ITypeSymbol? actual) + => SymbolEqualityComparer.Default.Equals(expected, actual); +} diff --git a/References/BizHawk.Analyzer.dll b/References/BizHawk.Analyzer.dll index 1d654f89a2..4c53a0a2e0 100644 Binary files a/References/BizHawk.Analyzer.dll and b/References/BizHawk.Analyzer.dll differ