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

libcore: DST-ify AsSlice, Equiv, and ops traits #18638

Merged
merged 2 commits into from
Nov 20, 2014

Conversation

aturon
Copy link
Member

@aturon aturon commented Nov 4, 2014

This PR changes AsSlice to work on unsized types, and changes the
impl for &[T] to [T]. Aside from making the trait more general,
this also helps some ongoing work with method resolution changes.

This is a breaking change: code that uses generics bounded by AsSlice
will have to change. In particular, such code previously often took
arguments of type V where V: AsSlice<T> by value. These should now
be taken by reference:

fn foo<Sized? V: AsSlice<T>>(v: &V) { .. }

A few std lib functions have been changed accordingly.

The PR also relaxes constraints on generics and traits within the
core::ops module and for the Equiv trait.

[breaking-change]

r? @nikomatsakis
cc @japaric

@japaric
Copy link
Member

japaric commented Nov 5, 2014

@aturon These impls probably need an extra Sized? as well:

libcollections/vec.rs:impl<T: PartialEq, V: AsSlice<T>> Equiv<V> for Vec<T> {
libcollections/vec.rs:impl<T: Clone, V: AsSlice<T>> Add<V, Vec<T>> for Vec<T> {
libcore/slice.rs:impl<T: PartialEq, V: AsSlice<T>> Equiv<V> for [T] {
libcore/slice.rs:impl<'a,T:PartialEq, V: AsSlice<T>> Equiv<V> for &'a mut [T] {
libgraphviz/maybe_owned_vec.rs:impl<'a, T: PartialEq, V: AsSlice<T>> Equiv<V> for MaybeOwnedVector<'a, T> {

@@ -343,7 +344,7 @@ impl Path {

/// Returns a normalized byte vector representation of a path, by removing all empty
/// components, and unnecessary . and .. components.
fn normalize<V: AsSlice<u8>>(v: V) -> Vec<u8> {
fn normalize<Sized? V: AsSlice<u8>>(v: &V) -> Vec<u8> {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(Last time I looked at this code, I had the impression that this method could be changed to normalize(&[u8]) (no need for generics), but I didn't look further)

@nikomatsakis
Copy link
Contributor

From my part, r+. These changes look fine to me. But @japaric is presumably right about the other impls.

@aturon aturon changed the title libcore: DST-ify AsSlice libcore: DST-ify AsSlice, Equiv, and ops traits Nov 5, 2014
@aturon
Copy link
Member Author

aturon commented Nov 5, 2014

I pushed an update to add Sized? to the impls that @japaric suggested. Doing so required DST-ifying more of core::ops and the Equiv trait; I went ahead and added Sized? throughout those traits.

I've looked for generic code over these traits for further places to add Sized?, but the only ones that might apply are slated for removal due to collections or numerics reform.

@japaric Want to take a look as well and make sure I didn't miss something?

@japaric
Copy link
Member

japaric commented Nov 5, 2014

LGTM

@@ -119,7 +119,7 @@ pub trait VectorVector<T> for Sized? {
fn connect_vec(&self, sep: &T) -> Vec<T>;
}

impl<T: Clone, V: AsSlice<T>> VectorVector<T> for [V] {
impl<'a, T: Clone, Sized? V: AsSlice<T>> VectorVector<T> for [&'a V] {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This impl breaks concat_vec for &[Vec<T>] (breaking tests below).

@aturon
Copy link
Member Author

aturon commented Nov 5, 2014

OK, this turns out to be somewhat subtle, because sometimes generic code really needs to work with AsSlice by value rather than by reference, and that in turn means we need impls for references to slices, not just slices. Ideally we'd do something like:

impl<'a, T, Sized? U: AsSlice<T>> AsSlice<T> for &'a U {
    #[inline(always)]
    fn as_slice<'a>(&'a self) -> &'a [T] { self.as_slice() }
}

impl<'a, T, Sized? U: AsSlice<T>> AsSlice<T> for &'a mut U {
    #[inline(always)]
    fn as_slice<'a>(&'a self) -> &'a [T] { self.as_slice() }
}

but this is currently not working due to, I think, some weakness in method resolution that @nikomatsakis is fixing. Will investigate.

@japaric
Copy link
Member

japaric commented Nov 5, 2014

FWIW, I usually write such methods as:

impl<'a, T, Sized? U: AsSlice<T>> AsSlice<T> for &'a U {
    #[inline(always)]
    fn as_slice<'a>(&'a self) -> &'a [T] { AsSlice::as_slice(*self) }
}

Otherwise, I end up hitting infinite recursion errors.

@aturon
Copy link
Member Author

aturon commented Nov 5, 2014

(In particular, adding these impls causes the .as_slice() method to no longer be available on String.)

@aturon
Copy link
Member Author

aturon commented Nov 5, 2014

@japaric Ah yes, of course, thanks.

@alexcrichton
Copy link
Member

ping @aturon, needs a rebase

@aturon
Copy link
Member Author

aturon commented Nov 16, 2014

@alexcrichton This is blocked on @nikomatsakis's method dispatch changes landing.

@nikomatsakis
Copy link
Contributor

@aturon unblocked now, I guess?

@nikomatsakis
Copy link
Contributor

I'm going to unassign myself and assign to @aturon since it just needs rebasing and tender love and care at this point.

@nikomatsakis nikomatsakis assigned aturon and unassigned nikomatsakis Nov 18, 2014
This commit changes `AsSlice` to work on unsized types, and changes the
`impl` for `&[T]` to `[T]`. Aside from making the trait more general,
this also helps some ongoing work with method resolution changes.

This is a breaking change: code that uses generics bounded by `AsSlice`
will have to change. In particular, such code previously often took
arguments of type `V` where `V: AsSlice<T>` by value. These should now
be taken by reference:

```rust
fn foo<Sized? V: AsSlice<T>>(v: &V) { .. }
```

A few std lib functions have been changed accordingly.

[breaking-change]
This commit relaxes constraints on generics and traits within the
`core::ops` module and for the `Equiv` trait.
bors added a commit that referenced this pull request Nov 20, 2014
This PR changes `AsSlice` to work on unsized types, and changes the
`impl` for `&[T]` to `[T]`. Aside from making the trait more general,
this also helps some ongoing work with method resolution changes.

This is a breaking change: code that uses generics bounded by `AsSlice`
will have to change. In particular, such code previously often took
arguments of type `V` where `V: AsSlice<T>` by value. These should now
be taken by reference:

```rust
fn foo<Sized? V: AsSlice<T>>(v: &V) { .. }
```

A few std lib functions have been changed accordingly.

The PR also relaxes constraints on generics and traits within the
`core::ops` module and for the `Equiv` trait.

[breaking-change]

r? @nikomatsakis 
cc @japaric
@bors bors closed this Nov 20, 2014
@bors bors merged commit c287afb into rust-lang:master Nov 20, 2014
@nikomatsakis
Copy link
Contributor

Hmm, I think this means we can remove the horrible "auto-ref-ref" hack in the method resolution code.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants