Skip to content
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

Open
gmorenz opened this issue Jan 19, 2020 · 5 comments
Open
Labels
A-associated-items Area: Associated items (types, constants & functions) A-type-system Area: Type system T-lang Relevant to the language team, which will review and decide on the PR/issue. T-types Relevant to the types team, which will review and decide on the PR/issue.

Comments

@gmorenz
Copy link

gmorenz commented Jan 19, 2020

Consider the following code (playpen)

pub trait Data {
    type Elem;
}

pub struct ViewRepr<A>(A);

impl<'a, A> Data for ViewRepr<&'a A> {
    type Elem = A;
}

pub struct ArrayBase<D: Data> {
    ptr: *mut D::Elem,
    // ptr: *mut f32,
    d: D,
}

pub fn cast_array_view<'shorter, 'longer: 'shorter>(
    input: ArrayBase<ViewRepr<&'longer f32>>
) -> ArrayBase<ViewRepr<&'shorter f32>> {
    input
}

cast_array_view fails to compile with the first definition of ptr, 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 to T. Since D::Elem is being stored behind a *mut pointer rustc has decided that ArrayBase is invariant with respect to D, instead of D::Elem.

PS: Thanks to /u/Patryk27 on reddit for looking at this and pointing me in the direction of the variance docs.

@jonas-schievink jonas-schievink added A-associated-items Area: Associated items (types, constants & functions) T-lang Relevant to the language team, which will review and decide on the PR/issue. A-type-system Area: Type system labels Jan 19, 2020
@RustyYato
Copy link
Contributor

RustyYato commented Jan 20, 2020

Using associated types always results in an invariant type bound, so that's where the invaraince is coming from. Also *mut T is invariant in T, as you noted, but this doesn't make ArrayBase<D> invariant in D, only invariant in D::Elem. But the fact that you used D::Elem makes it invariant in D.

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

@gmorenz
Copy link
Author

gmorenz commented Jan 20, 2020

Using associated types always results in an invariant type bound

Huh, ok. That leaves two questions

@RustyYato
Copy link
Contributor

Why?

Because there is no general way to figure out the lifetime relationships between the associated type and Self. So invariance is the only safe variance.

Is this documented somewhere? Maybe it should be added to the variance section of the nomicon?

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).

@RustyYato
Copy link
Contributor

RustyYato commented Jan 20, 2020

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.

@gmorenz
Copy link
Author

gmorenz commented Jan 20, 2020

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.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-associated-items Area: Associated items (types, constants & functions) A-type-system Area: Type system T-lang Relevant to the language team, which will review and decide on the PR/issue. T-types Relevant to the types team, which will review and decide on the PR/issue.
Projects
None yet
Development

No branches or pull requests

4 participants