-
Notifications
You must be signed in to change notification settings - Fork 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
Consider silly cyclic assignment in scoped variance #8766
Consider silly cyclic assignment in scoped variance #8766
Conversation
- 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 | ||
- The method has at least one additional `ref`, `in`, or `out` parameter, or a parameter of `ref struct` type. | ||
- The method has a `ref` or `out` parameter of `ref struct` type with a mismatch of adding `[UnscopedRef]` (not removing `scoped`). | ||
(In this case, a [silly cyclic assignment](#cyclic-assignment) is possible, hence no other parameters are necessary.) |
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 don't think it's so much that it allows silly cyclic assignment as it's just a basic type system violation. Consider that it allows you to do things like:
interface I1 {
ref int M();
}
struct S {
int field;
[UnscopedRef] ref int M() => ref field;
}
This is a similar case where introducing [UnscopedRef]
in the implementing type leads to lifetime violation.
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.
Yes, but that example is already covered by the existing rules and is an error today (I fixed it so S
actually implements I1
). Those rules are
- Or both of these are true:
- The method returns a
ref struct
or returns aref
orref readonly
, or the method has aref
orout
parameter ofref struct
type.- The method has at least one additional
ref
,in
, orout
parameter, or a parameter ofref struct
type.
This new bullet point
- The method has a
ref
orout
parameter ofref struct
type with a mismatch of adding[UnscopedRef]
(not removingscoped
).
is only necessary because of the silly cyclic assignment.
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.
Discussed offline and it's not clear why the paragraph starting with
The compiler will report a diagnostic for unsafe scoped mismatches across overrides, interface implementations, and delegate conversions when:
is in the speclet in the first place.
In other words, it seems that adding [UnscopedRef]
or removing scoped
should always be a mismatch error without exceptions. cc @cston
We should also investigate this sentence in the light of ref structs being allowed to implement interfaces:
The rules above ignore
this
parameters becauseref struct
instance methods cannot be used for overrides, interface implementations, or delegate conversions.
Co-authored-by: Jared Parsons <[email protected]>
The diagnostic is not reported in other cases because: | ||
- The methods with such signatures cannot capture the refs passed in, so any scoped mismatch is not dangerous. | ||
- These include very common and simple scenarios (e.g., plain old `out` parameters which are used in `TryParse` method signatures) | ||
and reporting scoped mismatches just because they are used across LangVersion 11 (and hence the `out` parameter is differently scoped) would be confusing. |
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.
Consider using "language version" rather than "LangVersion".
This brings back the scope variance exceptions removed from the speclet in #8777 and updates them.
Corresponding implementation bugfix: dotnet/roslyn#76261