-
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
Remove scope variance exceptions #76296
Merged
jjonescz
merged 7 commits into
dotnet:main
from
jjonescz:76100-ScopeVariance-NoExceptions
Jan 6, 2025
Merged
Changes from 6 commits
Commits
Show all changes
7 commits
Select commit
Hold shift + click to select a range
f0d3f46
Remove scope variance exceptions
jjonescz 62910ca
Add UnscopedRef example
jjonescz 0539a70
Report new scoped mismatches as warnings in older lang versions
jjonescz 810f1e3
Merge branch 'main' into 76100-ScopeVariance-NoExceptions
jjonescz 1d15cfb
Fixup a test
jjonescz dc450fc
Fixup another test
jjonescz 88b5b10
Merge branch 'main' into 76100-ScopeVariance-NoExceptions
jjonescz File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
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
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 |
---|---|---|
|
@@ -1149,25 +1149,22 @@ static void checkValidMethodOverride( | |
MethodSymbol overridingMethod, | ||
BindingDiagnosticBag diagnostics) | ||
{ | ||
if (RequiresValidScopedOverrideForRefSafety(overriddenMethod)) | ||
{ | ||
CheckValidScopedOverride( | ||
overriddenMethod, | ||
overridingMethod, | ||
diagnostics, | ||
static (diagnostics, overriddenMethod, overridingMethod, overridingParameter, _, location) => | ||
{ | ||
diagnostics.Add( | ||
ReportInvalidScopedOverrideAsError(overriddenMethod, overridingMethod) ? | ||
ErrorCode.ERR_ScopedMismatchInParameterOfOverrideOrImplementation : | ||
ErrorCode.WRN_ScopedMismatchInParameterOfOverrideOrImplementation, | ||
location, | ||
new FormattedSymbol(overridingParameter, SymbolDisplayFormat.ShortFormat)); | ||
}, | ||
overridingMemberLocation, | ||
allowVariance: true, | ||
invokedAsExtensionMethod: false); | ||
} | ||
CheckValidScopedOverride( | ||
overriddenMethod, | ||
overridingMethod, | ||
diagnostics, | ||
static (diagnostics, overriddenMethod, overridingMethod, overridingParameter, _, location) => | ||
{ | ||
diagnostics.Add( | ||
ReportInvalidScopedOverrideAsError(overriddenMethod, overridingMethod) ? | ||
ErrorCode.ERR_ScopedMismatchInParameterOfOverrideOrImplementation : | ||
ErrorCode.WRN_ScopedMismatchInParameterOfOverrideOrImplementation, | ||
location, | ||
new FormattedSymbol(overridingParameter, SymbolDisplayFormat.ShortFormat)); | ||
}, | ||
overridingMemberLocation, | ||
allowVariance: true, | ||
invokedAsExtensionMethod: false); | ||
|
||
CheckValidNullableMethodOverride(overridingMethod.DeclaringCompilation, overriddenMethod, overridingMethod, diagnostics, | ||
ReportBadReturn, | ||
|
@@ -1369,61 +1366,56 @@ static bool isValidNullableConversion( | |
} | ||
|
||
#nullable enable | ||
/// <summary> | ||
/// Returns true if the method signature must match, with respect to scoped for ref safety, | ||
/// in overrides, interface implementations, or delegate conversions. | ||
/// </summary> | ||
internal static bool RequiresValidScopedOverrideForRefSafety(MethodSymbol? method) | ||
{ | ||
if (method is null) | ||
{ | ||
return false; | ||
} | ||
|
||
var parameters = method.Parameters; | ||
|
||
// https://github.com/dotnet/csharplang/blob/main/proposals/csharp-11.0/low-level-struct-improvements.md#scoped-mismatch | ||
// The compiler will report a diagnostic for _unsafe scoped mismatches_ across overrides, interface implementations, and delegate conversions when: | ||
// - The method returns a `ref struct` or returns a `ref` or `ref readonly`, or the method has a `ref` or `out` parameter of `ref struct` type, and | ||
// ... | ||
int nRefParametersRequired; | ||
if (method.ReturnType.IsRefLikeOrAllowsRefLikeType() || | ||
(method.RefKind is RefKind.Ref or RefKind.RefReadOnly)) | ||
{ | ||
nRefParametersRequired = 1; | ||
} | ||
else if (parameters.Any(p => (p.RefKind is RefKind.Ref or RefKind.Out) && p.Type.IsRefLikeOrAllowsRefLikeType())) | ||
{ | ||
nRefParametersRequired = 2; // including the parameter found above | ||
} | ||
else | ||
{ | ||
return false; | ||
} | ||
|
||
// ... | ||
// - The method has at least one additional `ref`, `in`, `ref readonly`, or `out` parameter, or a parameter of `ref struct` type. | ||
int nRefParameters = parameters.Count(p => p.RefKind is RefKind.Ref or RefKind.In or RefKind.RefReadOnlyParameter or RefKind.Out); | ||
if (nRefParameters >= nRefParametersRequired) | ||
{ | ||
return true; | ||
} | ||
else if (parameters.Any(p => p.RefKind == RefKind.None && p.Type.IsRefLikeOrAllowsRefLikeType())) | ||
{ | ||
return true; | ||
} | ||
|
||
return false; | ||
} | ||
|
||
/// <summary> | ||
/// Returns true if a scoped mismatch should be reported as an error rather than a warning. | ||
/// </summary> | ||
internal static bool ReportInvalidScopedOverrideAsError(MethodSymbol baseMethod, MethodSymbol overrideMethod) | ||
{ | ||
// https://github.com/dotnet/csharplang/blob/main/proposals/csharp-11.0/low-level-struct-improvements.md#scoped-mismatch | ||
// The diagnostic is reported as an error if the mismatched signatures are both using C#11 ref safety rules; otherwise, the diagnostic is a warning. | ||
return baseMethod.UseUpdatedEscapeRules && overrideMethod.UseUpdatedEscapeRules; | ||
return baseMethod.UseUpdatedEscapeRules && overrideMethod.UseUpdatedEscapeRules && | ||
// We have removed exceptions to the scoped mismatch error reporting, but to avoid breaks | ||
// we report the new scenarios (previously exempted) as warnings in C# 12 and earlier. | ||
// https://github.com/dotnet/roslyn/issues/76100 | ||
(overrideMethod.DeclaringCompilation.LanguageVersion > LanguageVersion.CSharp12 || usedToBeReported(baseMethod)); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Downgraded the new errors to warnings in C# 12 and lower to lessen the break per offline discussion with Jared. (We think breaking in C# 13 is fine since it's new.) |
||
|
||
static bool usedToBeReported(MethodSymbol method) | ||
{ | ||
var parameters = method.Parameters; | ||
|
||
// https://github.com/dotnet/csharplang/blob/1f7f23f/proposals/csharp-11.0/low-level-struct-improvements.md#scoped-mismatch | ||
// The compiler will report a diagnostic for _unsafe scoped mismatches_ across overrides, interface implementations, and delegate conversions when: | ||
// - The method returns a `ref struct` or returns a `ref` or `ref readonly`, or the method has a `ref` or `out` parameter of `ref struct` type, and | ||
// ... | ||
int nRefParametersRequired; | ||
if (method.ReturnType.IsRefLikeOrAllowsRefLikeType() || | ||
(method.RefKind is RefKind.Ref or RefKind.RefReadOnly)) | ||
{ | ||
nRefParametersRequired = 1; | ||
} | ||
else if (parameters.Any(p => (p.RefKind is RefKind.Ref or RefKind.Out) && p.Type.IsRefLikeOrAllowsRefLikeType())) | ||
{ | ||
nRefParametersRequired = 2; // including the parameter found above | ||
} | ||
else | ||
{ | ||
return false; | ||
} | ||
|
||
// ... | ||
// - The method has at least one additional `ref`, `in`, `ref readonly`, or `out` parameter, or a parameter of `ref struct` type. | ||
int nRefParameters = parameters.Count(p => p.RefKind is RefKind.Ref or RefKind.In or RefKind.RefReadOnlyParameter or RefKind.Out); | ||
if (nRefParameters >= nRefParametersRequired) | ||
{ | ||
return true; | ||
} | ||
else if (parameters.Any(p => p.RefKind == RefKind.None && p.Type.IsRefLikeOrAllowsRefLikeType())) | ||
{ | ||
return true; | ||
} | ||
|
||
return false; | ||
} | ||
} | ||
|
||
/// <summary> | ||
|
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
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
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.
Note: I found these breaks when building the BCL with this change: dotnet/runtime@fae33d2e25a
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.
@stephentoub FYI