-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
Associated type behind a *mut unecessarily results in invariant subtyping #68375
Comments
Using associated types always results in an invariant type bound, so that's where the invaraince is coming from. Also To get around this: pub trait Data {
type Elem;
}
pub struct ViewRepr<A>(A);
impl<'a, A> Data for ViewRepr<&'a A> {
type Elem = A;
}
type ArrayBase<D> = ArrayBaseInner<D, <D as Data>::Elem>;
pub struct ArrayBaseInner<D: Data<Elem = Elem>, Elem> {
ptr: *mut Elem,
d: D,
}
pub fn cast_array_view<'shorter, 'longer: 'shorter>(
input: ArrayBase<ViewRepr<&'longer f32>>
) -> ArrayBase<ViewRepr<&'shorter f32>> {
input
} This allows Rust to reason about the variance of each parameter separately, which allows the code to pass |
Huh, ok. That leaves two questions
|
Because there is no general way to figure out the lifetime relationships between the associated type and
I'm not sure, I just came across this in much the same way you did. It should be added to the nomicon (and the reference while we're at it). |
Oh, you don't need the type alias. This works pub struct ArrayBase<D: Data<Elem = Elem>, Elem = <D as Data>::Elem> {
ptr: *mut Elem,
d: D,
} This leads me to believe that we maybe could allow for a more lenient variance for associated types. |
I tend to agree with your conclusion that that suggests it would be possible to do better. Actually trying to use that in real code (ndarray) results in me running into a different issue, which I believe is a version of #28895. I've added a minimized example to that issue. |
Consider the following code (playpen)
cast_array_view
fails to compile with the first definition ofptr
, and succeeds with the second, despite the two functions being apparently identical.I think the reason this is happening is most likely related to the fact that subtyping of
*mut
pointers is invariant on the type of their parameter, and structs containing*mut T
pointers are invariant with respect toT
. SinceD::Elem
is being stored behind a*mut
pointer rustc has decided thatArrayBase
is invariant with respect toD
, instead ofD::Elem
.PS: Thanks to /u/Patryk27 on reddit for looking at this and pointing me in the direction of the variance docs.
The text was updated successfully, but these errors were encountered: