From b6686611be9ee6770abae7923078cf3083998114 Mon Sep 17 00:00:00 2001 From: Andy Gocke Date: Tue, 23 Apr 2019 15:10:08 -0700 Subject: [PATCH 1/2] Fix crash in pattern matching When we relaxed the requirement for pattern matching open types to a constant pattern to not require a conversion from the pattern expression to the open type, but the pattern expression should be required to have a constant value. Fixes #34980 --- .../CSharp/Portable/Binder/Binder_Patterns.cs | 2 +- .../Semantics/PatternMatchingTests4.cs | 128 ++++++++++++++++++ 2 files changed, 129 insertions(+), 1 deletion(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs index 9ec9f7dd06054..947cd408a0962 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs @@ -188,7 +188,7 @@ internal BoundExpression ConvertPatternExpression( if (inputType.ContainsTypeParameter()) { convertedExpression = expression; - if (!hasErrors) + if (!hasErrors && expression.ConstantValue is {}) { HashSet useSiteDiagnostics = null; if (expression.ConstantValue == ConstantValue.Null) diff --git a/src/Compilers/CSharp/Test/Semantic/Semantics/PatternMatchingTests4.cs b/src/Compilers/CSharp/Test/Semantic/Semantics/PatternMatchingTests4.cs index 16a44674b8022..c7299e96fec67 100644 --- a/src/Compilers/CSharp/Test/Semantic/Semantics/PatternMatchingTests4.cs +++ b/src/Compilers/CSharp/Test/Semantic/Semantics/PatternMatchingTests4.cs @@ -12,6 +12,134 @@ namespace Microsoft.CodeAnalysis.CSharp.UnitTests.Semantics [CompilerTrait(CompilerFeature.Patterns)] public class PatternMatchingTests4 : PatternMatchingTestBase { + [Fact] + [WorkItem(34980, "https://github.com/dotnet/roslyn/issues/34980")] + public void PatternMatchOpenTypeCaseDefault() + { + var comp = CreateCompilation(@" +class C +{ + public void M(T t) + { + switch (t) + { + case default: + break; + } + } +}"); + comp.VerifyDiagnostics( + // (8,18): error CS0150: A constant value is expected + // case default: + Diagnostic(ErrorCode.ERR_ConstantExpected, "default").WithLocation(8, 18), + // (8,18): error CS8313: A default literal 'default' is not valid as a case constant. Use another literal (e.g. '0' or 'null') as appropriate. If you intended to write the default label, use 'default:' without 'case'. + // case default: + Diagnostic(ErrorCode.ERR_DefaultInSwitch, "default").WithLocation(8, 18)); + } + + [Fact] + [WorkItem(34980, "https://github.com/dotnet/roslyn/issues/34980")] + public void PatternMatchOpenTypeCaseDefaultT() + { + var comp = CreateCompilation(@" +class C +{ + public void M(T t) + { + switch (t) + { + case default(T): + break; + } + } +}"); + comp.VerifyDiagnostics( + // (8,18): error CS0150: A constant value is expected + // case default(T): + Diagnostic(ErrorCode.ERR_ConstantExpected, "default(T)").WithLocation(8, 18)); + } + + [Fact] + [WorkItem(34980, "https://github.com/dotnet/roslyn/issues/34980")] + public void PatternMatchGenericParameterToMethodGroup() + { + var comp = CreateCompilation(@" +class C +{ + public void M1(object o) + { + _ = o is M1; + switch (o) + { + case M1: + break; + } + } + public void M2(T t) + { + _ = t is M2; + switch (t) + { + case M2: + break; + } + } +}"); + comp.VerifyDiagnostics( + // (6,18): error CS0428: Cannot convert method group 'M1' to non-delegate type 'object'. Did you intend to invoke the method? + // _ = o is M1; + Diagnostic(ErrorCode.ERR_MethGrpToNonDel, "M1").WithArguments("M1", "object").WithLocation(6, 18), + // (9,18): error CS0428: Cannot convert method group 'M1' to non-delegate type 'object'. Did you intend to invoke the method? + // case M1: + Diagnostic(ErrorCode.ERR_MethGrpToNonDel, "M1").WithArguments("M1", "object").WithLocation(9, 18), + // (15,18): error CS0150: A constant value is expected + // _ = t is M2; + Diagnostic(ErrorCode.ERR_ConstantExpected, "M2").WithLocation(15, 18), + // (18,18): error CS0150: A constant value is expected + // case M2: + Diagnostic(ErrorCode.ERR_ConstantExpected, "M2").WithLocation(18, 18) + ); + } + + [Fact] + [WorkItem(34980, "https://github.com/dotnet/roslyn/issues/34980")] + public void PatternMatchGenericParameterToNonConstantExprs() + { + var comp = CreateCompilation(@" +class C +{ + public void M(T t) + { + switch (t) + { + case (() => 0): + break; + case stackalloc int[1] { 0 }: + break; + case new { X = 0 }: + break; + } + } +}"); + comp.VerifyDiagnostics( + // (8,18): error CS8129: No suitable 'Deconstruct' instance or extension method was found for type 'T', with 2 out parameters and a void return type. + // case (() => 0): + Diagnostic(ErrorCode.ERR_MissingDeconstruct, "(() => 0)").WithArguments("T", "2").WithLocation(8, 18), + // (8,22): error CS1003: Syntax error, ',' expected + // case (() => 0): + Diagnostic(ErrorCode.ERR_SyntaxError, "=>").WithArguments(",", "=>").WithLocation(8, 22), + // (8,25): error CS1003: Syntax error, ',' expected + // case (() => 0): + Diagnostic(ErrorCode.ERR_SyntaxError, "0").WithArguments(",", "").WithLocation(8, 25), + // (10,18): error CS1525: Invalid expression term 'stackalloc' + // case stackalloc int[1] { 0 }: + Diagnostic(ErrorCode.ERR_InvalidExprTerm, "stackalloc").WithArguments("stackalloc").WithLocation(10, 18), + // (12,18): error CS0150: A constant value is expected + // case new { X = 0 }: + Diagnostic(ErrorCode.ERR_ConstantExpected, "new { X = 0 }").WithLocation(12, 18) + ); + } + [Fact] public void TestPresenceOfITuple() { From a0bf84b4c3863c592df62565e6cccee411537b6a Mon Sep 17 00:00:00 2001 From: Andy Gocke Date: Thu, 25 Apr 2019 09:33:32 -0700 Subject: [PATCH 2/2] Respond to PR comments --- src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs b/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs index 947cd408a0962..c783ddb97c617 100644 --- a/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs +++ b/src/Compilers/CSharp/Portable/Binder/Binder_Patterns.cs @@ -188,7 +188,8 @@ internal BoundExpression ConvertPatternExpression( if (inputType.ContainsTypeParameter()) { convertedExpression = expression; - if (!hasErrors && expression.ConstantValue is {}) + // If the expression does not have a constant value, an error will be reported in the caller + if (!hasErrors && expression.ConstantValue is object) { HashSet useSiteDiagnostics = null; if (expression.ConstantValue == ConstantValue.Null)