Join GitHub today
GitHub is home to over 28 million developers working together to host and review code, manage projects, and build software together.
Sign upTracking issue for `impl Trait` (RFC 1522, RFC 1951, RFC 2071) #34511
Comments
aturon
added
B-unstable
B-RFC-approved
labels
Jun 27, 2016
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
pthariensflame
Jul 15, 2016
Contributor
@aturon Can we actually put the RFC in the repository? (@mbrubeck commented there that this was a problem.)
|
@aturon Can we actually put the RFC in the repository? (@mbrubeck commented there that this was a problem.) |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
|
Done. |
eddyb
referenced this issue
Jul 28, 2016
Merged
Implement `impl Trait` in return type position by anonymization. #35091
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
eddyb
Jul 31, 2016
Member
First attempt at implementation is #35091 (second, if you count my branch from last year).
One problem I ran into is with lifetimes. Type inference likes to put region variables everywhere and without any region-checking changes, those variables don't infer to anything other than local scopes.
However, the concrete type must be exportable, so I restricted it to 'static and explicitly named early-bound lifetime parameters, but it's never any of those if any function is involved - even a string literal doesn't infer to 'static, it's pretty much completely useless.
One thing I thought of, that would have 0 impact on region-checking itself, is to erase lifetimes:
- nothing exposing the concrete type of an
impl Traitshould care about lifetimes - a quick search forReveal::Allsuggests that's already the case in the compiler - a bound needs to be placed on all concrete types of
impl Traitin the return type of a function, that it outlives the call of that function - this means that any lifetime is, by necessity, either'staticor one of the lifetime parameters of the function - even if we can't know which (e.g. "shortest of'aand'b") - we must choose a variance for the implicit lifetime parametrism of
impl Trait(i.e. on all lifetime parameters in scope, same as with type parameters): invariance is easiest and gives more control to the callee, while contravariance lets the caller do more and would require checking that every lifetime in the return type is in a contravariant position (same with covariant type parametrism instead of invariant) - the auto trait leaking mechanism requires that a trait bound may be put on the concrete type, in another function - since we've erased the lifetimes and have no idea what lifetime goes where, every erased lifetime in the concrete type will have to be substituted with a fresh inference variable that is guaranteed to not be shorter than the shortest lifetime out of all actual lifetime parameters; the problem lies in the fact that trait impls can end up requiring stronger lifetime relationships (e.g.
X<'a, 'a>orX<'static>), which must be detected and errored on, as they can't be proven for those lifetimes
That last point about auto trait leakage is my only worry, everything else seems straight-forward.
It's not entirely clear at this point how much of region-checking we can reuse as-is. Hopefully all.
|
First attempt at implementation is #35091 (second, if you count my branch from last year). One problem I ran into is with lifetimes. Type inference likes to put region variables everywhere and without any region-checking changes, those variables don't infer to anything other than local scopes. One thing I thought of, that would have 0 impact on region-checking itself, is to erase lifetimes:
That last point about auto trait leakage is my only worry, everything else seems straight-forward. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
arielb1
Jul 31, 2016
Contributor
But lifetimes are important with impl Trait - e.g.
fn get_debug_str(s: &str) -> impl fmt::Debug {
s
}
fn get_debug_string(s: &str) -> impl fmt::Debug {
s.to_string()
}
fn good(s: &str) -> Box<fmt::Debug+'static> {
// if this does not compile, that would be quite annoying
Box::new(get_debug_string())
}
fn bad(s: &str) -> Box<fmt::Debug+'static> {
// if this *does* compile, we have a problem
Box::new(get_debug_str())
}I mentioned that several times in the RFC threads
|
But lifetimes are important with fn get_debug_str(s: &str) -> impl fmt::Debug {
s
}
fn get_debug_string(s: &str) -> impl fmt::Debug {
s.to_string()
}
fn good(s: &str) -> Box<fmt::Debug+'static> {
// if this does not compile, that would be quite annoying
Box::new(get_debug_string())
}
fn bad(s: &str) -> Box<fmt::Debug+'static> {
// if this *does* compile, we have a problem
Box::new(get_debug_str())
}I mentioned that several times in the RFC threads |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
arielb1
Jul 31, 2016
Contributor
trait-object-less version:
fn as_debug(s: &str) -> impl fmt::Debug;
fn example() {
let mut s = String::new("hello");
let debug = as_debug(&s);
s.truncate(0);
println!("{:?}", debug);
}This is either UB or not depending on the definition of as_debug.
|
trait-object-less version: fn as_debug(s: &str) -> impl fmt::Debug;
fn example() {
let mut s = String::new("hello");
let debug = as_debug(&s);
s.truncate(0);
println!("{:?}", debug);
}This is either UB or not depending on the definition of |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
eddyb
Jul 31, 2016
Member
@arielb1 Ah, right, I forgot that one of the reasons I did what I did was to only capture lifetime parameters, not anonymous late-bound ones, except it doesn't really work.
|
@arielb1 Ah, right, I forgot that one of the reasons I did what I did was to only capture lifetime parameters, not anonymous late-bound ones, except it doesn't really work. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
eddyb
Jul 31, 2016
Member
@arielb1 Do we have a strict outlives relation we can put between lifetimes found in the concrete type pre-erasure and late-bound lifetimes in the signature? Otherwise, it might not be a bad idea to just look at lifetime relationships and insta-fail any direct or indirect 'a outlives 'b where 'a is anything other than 'static or a lifetime parameter and 'b appears in the concrete type of an impl Trait.
|
@arielb1 Do we have a strict outlives relation we can put between lifetimes found in the concrete type pre-erasure and late-bound lifetimes in the signature? Otherwise, it might not be a bad idea to just look at lifetime relationships and insta-fail any direct or indirect |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
nikomatsakis
Aug 3, 2016
Contributor
Sorry for taking a while to write back here. So I've been thinking
about this problem. My feeling is that we do, ultimately, have to (and
want to) extend regionck with a new kind of constraint -- I'll call it
an \in constraint, because it allows you to say something like '0 \in {'a, 'b, 'c}, meaning that the region used for '0 must be
either 'a, 'b, or 'c. I'm not sure of the best way to integrate
this into solving itself -- certainly if the \in set is a singleton
set, it's just an equate relation (which we don't currently have as a
first-class thing, but which can be composed out of two bounds), but
otherwise it makes things complicated.
This all relates to my desire to make the set of region constraints
more expressive than what we have today. Certainly one could compose a
\in constraint out of OR and == constraints. But of course more
expressive constraints are harder to solve and \in is no different.
Anyway, let me just lay out a bit of my thinking here. Let's work with this
example:
pub fn foo<'a,'b>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> {...}I think the most accurate desugaring for a impl Trait is probably a
new type:
pub struct FooReturn<'a, 'b> {
field: XXX // for some suitable type XXX
}
impl<'a,'b> Iterator for FooReturn<'a,'b> {
type Item = <XXX as Iterator>::Item;
}Now the impl Iterator<Item=u32> in foo should behave the same as
FooReturn<'a,'b> would behave. It's not a perfect match though. One
difference, for example, is variance, as eddyb brought up -- I am
assuming we will make impl Foo-like types invariant over the type
parameters of foo. The auto trait behavior works out, however.
(Another area where the match might not be ideal is if we ever add the
ability to "pierce" the impl Iterator abstraction, so that code
"inside" the abstraction knows the precise type -- then it would sort
of have an implicit "unwrap" operation taking place.)
In some ways a better match is to consider a kind of synthetic trait:
trait FooReturn<'a,'b> {
type Type: Iterator<Item=u32>;
}
impl<'a,'b> FooReturn<'a,'b> for () {
type Type = XXX;
}Now we could consider the impl Iterator type to be like <() as FooReturn<'a,'b>>::Type. This is also not a perfect match, because we
would ordinarily normalize it away. You might imagine using specialization
to prevent that though:
trait FooReturn<'a,'b> {
type Type: Iterator<Item=u32>;
}
impl<'a,'b> FooReturn<'a,'b> for () {
default type Type = XXX; // can't really be specialized, but wev
}In this case, <() as FooReturn<'a,'b>>::Type would not normalize,
and we have a much closer match. The variance, in particular, behaves
right; if we ever wanted to have some type that are "inside" the
abstraction, they would be the same but they are allowed to
normalize. However, there is a catch: the auto trait stuff doesn't
quite work. (We may want to consider harmonizing things here,
actually.)
Anyway, my point in exploring these potential desugarings is not to
suggest that we implement "impl Trait" as an actual desugaring
(though it might be nice...) but to give an intuition for our job. I
think that the second desugaring -- in terms of projections -- is a
pretty helpful one for guiding us forward.
One place that this projection desugaring is a really useful guide is
the "outlives" relation. If we wanted to check whether <() as FooReturn<'a,'b>>::Type: 'x, RFC 1214 tells us that we can prove this
so long as 'a: 'x and 'b: 'x holds. This is I think how we want
to handle things for impl trait as well.
At trans time, and for auto-traits, we will have to know what XXX
is, of course. The basic idea here, I assume, is to create a type
variable for XXX and check that the actual values which are returned
can all be unified with XXX. That type variable should, in theory,
tell us our answer. But of course the problem is that this type
variable may refer to a lot of regions which are not in scope in the
fn signature -- e.g., the regions of the fn body. (This same problem
does not occur with types; even though, technically, you could put
e.g. a struct declaration in the fn body and it would be unnameable,
that's a kind of artificial restriction -- one could just as well move
the struct outside the fn.)
If you look both at the struct desugaring or the impl, there is an
(implicit in the lexical structure of Rust) restriction that XXX can
only name either 'static or lifetimes like 'a and 'b, which
appear in the function signature. That is the thing we are not
modeling here. I'm not sure the best way to do it -- some type
inference schemes have a more direct representation of scoping, and
I've always wanted to add that to Rust, to help us with closures. But
let's think about smaller deltas first I guess.
This is where the \in constraint comes from. One can imagine adding
a type-check rule that (basically) FR(XXX) \subset {'a, 'b} --
meaning that the "free regions" appearing in XXX can only be 'a and
'b. This would wind up translating to \in requirements for the
various regions that appear in XXX.
Let's look at an actual example:
fn foo<'a,'b>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> {
if condition { x.iter().cloned() } else { y.iter().cloned() }
}Here, the type if condition is true would be something like
Cloned<SliceIter<'a, i32>>. But if condition is false, we would
want Cloned<SliceIter<'b, i32>>. Of course in both cases we would
wind up with something like (using numbers for type/region variables):
Cloned<SliceIter<'0, i32>> <: 0
'a: '0 // because the source is x.iter()
Cloned<SliceIter<'1, i32>> <: 0
'b: '1 // because the source is y.iter()
If we then instantiate the variable 0 to Cloned<SliceIter<'2, i32>>,
we have '0: '2 and '1: '2, or a total set of region relations
like:
'a: '0
'0: '2
'b: '1
'1: '2
'2: 'body // the lifetime of the fn body
So what value should we use for '2? We have also the additional
constraint that '2 in {'a, 'b}. With the fn as written, I think we
would have to report an error, since neither 'a nor 'b is a
correct choice. Interestingly, though, if we added the constraint 'a: 'b, then there would be a correct value ('b).
Note that if we just run the normal algorithm, we would wind up with
'2 being 'body. I'm not sure how to handle the \in relations
except for exhaustive search (though I can imagine some special
cases).
OK, that's as far as I've gotten. =)
|
Sorry for taking a while to write back here. So I've been thinking This all relates to my desire to make the set of region constraints Anyway, let me just lay out a bit of my thinking here. Let's work with this pub fn foo<'a,'b>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> {...}I think the most accurate desugaring for a pub struct FooReturn<'a, 'b> {
field: XXX // for some suitable type XXX
}
impl<'a,'b> Iterator for FooReturn<'a,'b> {
type Item = <XXX as Iterator>::Item;
}Now the In some ways a better match is to consider a kind of synthetic trait: trait FooReturn<'a,'b> {
type Type: Iterator<Item=u32>;
}
impl<'a,'b> FooReturn<'a,'b> for () {
type Type = XXX;
}Now we could consider the trait FooReturn<'a,'b> {
type Type: Iterator<Item=u32>;
}
impl<'a,'b> FooReturn<'a,'b> for () {
default type Type = XXX; // can't really be specialized, but wev
}In this case, Anyway, my point in exploring these potential desugarings is not to One place that this projection desugaring is a really useful guide is At trans time, and for auto-traits, we will have to know what If you look both at the struct desugaring or the impl, there is an This is where the Let's look at an actual example: fn foo<'a,'b>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> {
if condition { x.iter().cloned() } else { y.iter().cloned() }
}Here, the type if
If we then instantiate the variable 0 to
So what value should we use for Note that if we just run the normal algorithm, we would wind up with OK, that's as far as I've gotten. =) |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
nikomatsakis
Aug 10, 2016
Contributor
On the PR #35091, @arielb1 wrote:
I don't like the "capture all lifetimes in the impl trait" approach and would prefer something more like lifetime elision.
I thought it would make more sense to discuss here. @arielb1, can you elaborate more on what you have in mind? In terms of the analogies I made above, I guess you are fundamentally talking about "pruning" the set of lifetimes that would appear either as parameters on the newtype or in the projection (i.e., <() as FooReturn<'a>>::Type instead of <() as FooReturn<'a,'b>>::Type or something?
I don't think that the lifetime elision rules as they exist would be a good guide in this respect: if we just picked the lifetime of &self to include only, then we wouldn't necessarily be able to include the type parameters from the Self struct, nor type parameters from the method, since they may have WF conditions that require us to name some of the other lifetimes.
Anyway, it'd be great to see some examples that illustrate the rules you have in mind, and perhaps any advantages thereof. :) (Also, I guess we would need some syntax to override the choice.) All other things being equal, if we can avoid having to pick from N lifetimes, I'd prefer that.
|
On the PR #35091, @arielb1 wrote:
I thought it would make more sense to discuss here. @arielb1, can you elaborate more on what you have in mind? In terms of the analogies I made above, I guess you are fundamentally talking about "pruning" the set of lifetimes that would appear either as parameters on the newtype or in the projection (i.e., I don't think that the lifetime elision rules as they exist would be a good guide in this respect: if we just picked the lifetime of Anyway, it'd be great to see some examples that illustrate the rules you have in mind, and perhaps any advantages thereof. :) (Also, I guess we would need some syntax to override the choice.) All other things being equal, if we can avoid having to pick from N lifetimes, I'd prefer that. |
added a commit
that referenced
this issue
Aug 12, 2016
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
petrochenkov
Aug 12, 2016
Contributor
I haven't seen interactions of impl Trait with privacy discussed anywhere.
Now fn f() -> impl Trait can return a private type S: Trait similarly to trait objects fn f() -> Box<Trait>. I.e. objects of private types can walk freely outside of their module in anonymized form.
This seems reasonable and desirable - the type itself is an implementation detail, only its interface, available through a public trait Trait is public.
However there's one difference between trait objects and impl Trait. With trait objects alone all trait methods of private types can get internal linkage, they will still be callable through function pointers. With impl Traits trait methods of private types are directly callable from other translation units. The algorithm doing "internalization" of symbols will have to try harder to internalize methods only for types not anonymized with impl Trait, or to be very pessimistic.
|
I haven't seen interactions of |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
arielb1
Aug 14, 2016
Contributor
The "explicit" way to write foo would be
fn foo<'a: 'c,'b: 'c,'c>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> + 'c {
if condition { x.iter().cloned() } else { y.iter().cloned() }
}Here there is no question about the lifetime bound. Obviously, having to write the lifetime bound each time would be quite repetitive. However, the way we deal with that kind of repetition is generally through lifetime elision. In the case of foo, elision would fail and force the programmer to explicitly specify lifetimes.
I am opposed to adding explicitness-sensitive lifetime elision as @eddyb did only in the specific case of impl Trait and not otherwise.
|
The "explicit" way to write fn foo<'a: 'c,'b: 'c,'c>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> + 'c {
if condition { x.iter().cloned() } else { y.iter().cloned() }
}Here there is no question about the lifetime bound. Obviously, having to write the lifetime bound each time would be quite repetitive. However, the way we deal with that kind of repetition is generally through lifetime elision. In the case of I am opposed to adding explicitness-sensitive lifetime elision as @eddyb did only in the specific case of |
jamesmunns
referenced this issue
in Covertness/coap-rs
Aug 14, 2016
Open
static handler brings problems #10
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
nikomatsakis
Aug 15, 2016
Contributor
@arielb1 hmm, I'm not 100% sure how to think about this proposed syntax in terms of the "desugarings" that I discussed. It allows you to specify what appears to be a lifetime bound, but the thing we are trying to infer is mostly what lifetimes appear in the hidden type. Does this suggest that at most one lifetime could be "hidden" (and that it would have to be specified exactly?)
It seems like it's not always the case that a "single lifetime parameter" suffices:
fn foo<'a, 'b>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> {
x.iter().chain(y).cloned()
}In this case, the hidden iterator type refers to both 'a and 'b (although it is variant in both of them; but I guess we could come up with an example that is invariant).
|
@arielb1 hmm, I'm not 100% sure how to think about this proposed syntax in terms of the "desugarings" that I discussed. It allows you to specify what appears to be a lifetime bound, but the thing we are trying to infer is mostly what lifetimes appear in the hidden type. Does this suggest that at most one lifetime could be "hidden" (and that it would have to be specified exactly?) It seems like it's not always the case that a "single lifetime parameter" suffices: fn foo<'a, 'b>(x: &'a [u32], y: &'b [u32]) -> impl Iterator<Item=u32> {
x.iter().chain(y).cloned()
}In this case, the hidden iterator type refers to both |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
nikomatsakis
Aug 18, 2016
Contributor
So @aturon and I discussed this issue somewhat and I wanted to share. There are really a couple of orthogonal questions here and I want to separate them out. The first question is "what type/lifetime parameters can potentially be used in the hidden type?" In terms of the (quasi-)desugaring into a default type, this comes down to "what type parameters appear on the trait we introduce". So, for example, if this function:
fn foo<'a, 'b, T>() -> impl Trait { ... }would get desugared to something like:
fn foo<'a, 'b, T>() -> <() as Foo<...>>::Type { ... }
trait Foo<...> {
type Type: Trait;
}
impl<...> Foo<...> for () {
default type Type = /* inferred */;
}then this question comes down to "what type parameters appear on the trait Foo and its impl"? Basically, the ... here. Clearly this include include the set of type parameters that appear are used by Trait itself, but what additional type parameters? (As I noted before, this desugaring is 100% faithful except for the leakage of auto traits, and I would argue that we should leak auto traits also for specializable impls.)
The default answer we've been using is "all of them", so here ... would be 'a, 'b, T (along with any anonymous parameters that may appear). This may be a reasonable default, but it's not necessarily the best default. (As @arielb1 pointed out.)
This has an effect on the outlives relation, since, in order to determine that <() as Foo<...>>::Type (referring to some particular, opaque instantiation of impl Trait) outlives 'x, we effectively must show that ...: 'x (that is, every lifetime and type parameter).
This is why I say it is not enough to consider lifetime parameters: imagine that we have some call to foo like foo::<'a0, 'b0, &'c0 i32>. This implies that all three lifetimes, '[abc]0, must outlive 'x -- in other words, so long as the return value is in use, this will prolog the loans of all data given into the function. But, as @arielb1 poitned out, elision suggests that this will usually be longer than necessary.
So I imagine that what we need is:
- to settle on a reasonable default, perhaps using intution from elision;
- to have an explicit syntax for when the default is not appropriate.
@aturon spitballed something like impl<...> Trait as the explicit syntax, which seems reasonable. Therefore, one could write:
fn foo<'a, 'b, T>(...) -> impl<T> Trait { }to indicate that the hidden type does not in fact refer to 'a or 'b but only T. Or one might write impl<'a> Trait to indicate that neither 'b nor T are captured.
As for the defaults, it seems like having more data would be pretty useful -- but the general logic of elision suggests that we would do well to capture all the parameters named in the type of self, when applicable. E.g., if you have fn foo<'a,'b>(&'a self, v: &'b [u8]) and the type is Bar<'c, X>, then the type of self would be &'a Bar<'c, X> and hence we would capture 'a, 'c, and X by default, but not 'b.
Another related note is what the meaning of a lifetime bound is. I think that sound lifetime bounds have an existing meaning that should not be changed: if we write impl (Trait+'a) that means that the hidden type T outlives 'a. Similarly one can write impl (Trait+'static) to indicate that there are no borrowed pointers present (even if some lifetimes are captured). When inferring the hidden type T, this would imply a lifetime bound like $T: 'static, where $T is the inference variable we create for the hidden type. This would be handled in the usual way. From a caller's perspective, where the hidden type is, well, hidden, the 'static bound would allow us to conclude that impl (Trait+'static) outlives 'static even if there are lifetime parameters captured.
Here it just behaves exactly as the desugaring would behave:
fn foo<'a, 'b, T>() -> <() as Foo<'a, 'b, 'T>>::Type { ... }
trait Foo<'a, 'b, T> {
type Type: Trait + 'static; // <-- note the `'static` bound appears here
}
impl<'a, 'b, T> Foo<...> for () {
default type Type = /* something that doesn't reference `'a`, `'b`, or `T` */;
}All of this is orthogonal from inference. We still want (I think) to add the notion of a "choose from" constraint and modify inference with some heuristics and, possibly, exhaustive search (the experience from RFC 1214 suggests that heuristics with a conservative fallback can actually get us very far; I'm not aware of people running into limitations in this respect, though there is probably an issue somewhere). Certainly, adding lifetime bounds like 'static or 'a` may influence inference, and thus be helpful, but that is not a perfect solution: for one thing, they are visible to the caller and become part of the API, which may not be desired.
|
So @aturon and I discussed this issue somewhat and I wanted to share. There are really a couple of orthogonal questions here and I want to separate them out. The first question is "what type/lifetime parameters can potentially be used in the hidden type?" In terms of the (quasi-)desugaring into a fn foo<'a, 'b, T>() -> impl Trait { ... }would get desugared to something like: fn foo<'a, 'b, T>() -> <() as Foo<...>>::Type { ... }
trait Foo<...> {
type Type: Trait;
}
impl<...> Foo<...> for () {
default type Type = /* inferred */;
}then this question comes down to "what type parameters appear on the trait The default answer we've been using is "all of them", so here This has an effect on the outlives relation, since, in order to determine that This is why I say it is not enough to consider lifetime parameters: imagine that we have some call to So I imagine that what we need is:
@aturon spitballed something like fn foo<'a, 'b, T>(...) -> impl<T> Trait { }to indicate that the hidden type does not in fact refer to As for the defaults, it seems like having more data would be pretty useful -- but the general logic of elision suggests that we would do well to capture all the parameters named in the type of Another related note is what the meaning of a lifetime bound is. I think that sound lifetime bounds have an existing meaning that should not be changed: if we write Here it just behaves exactly as the desugaring would behave: fn foo<'a, 'b, T>() -> <() as Foo<'a, 'b, 'T>>::Type { ... }
trait Foo<'a, 'b, T> {
type Type: Trait + 'static; // <-- note the `'static` bound appears here
}
impl<'a, 'b, T> Foo<...> for () {
default type Type = /* something that doesn't reference `'a`, `'b`, or `T` */;
}All of this is orthogonal from inference. We still want (I think) to add the notion of a "choose from" constraint and modify inference with some heuristics and, possibly, exhaustive search (the experience from RFC 1214 suggests that heuristics with a conservative fallback can actually get us very far; I'm not aware of people running into limitations in this respect, though there is probably an issue somewhere). Certainly, adding lifetime bounds like |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
arielb1
Aug 18, 2016
Contributor
Possible options:
Explicit lifetime bound with output parameter elision
Like trait objects today, impl Trait objects have a single lifetime bound parameter, which is inferred using the elision rules.
Disadvantage: unergonomic
Advantage: clear
Explicit lifetime bounds with "all generic" elision
Like trait objects today, impl Trait objects have a single lifetime bound parameter.
However, elision creates a new early-bound parameters that outlives all explicit parameters:
fn foo<T>(&T) -> impl Foo
-->
fn foo<'total, T: 'total>(&T) -> impl Foo + 'totalDisadvantage: adds an early-bound parameter
more.
|
Possible options: Explicit lifetime bound with output parameter elisionLike trait objects today, Disadvantage: unergonomic Explicit lifetime bounds with "all generic" elisionLike trait objects today, However, elision creates a new early-bound parameters that outlives all explicit parameters: fn foo<T>(&T) -> impl Foo
-->
fn foo<'total, T: 'total>(&T) -> impl Foo + 'totalDisadvantage: adds an early-bound parameter more. |
nrc
added
the
T-lang
label
Aug 19, 2016
nrc
added
B-RFC-implemented
and removed
B-RFC-approved
labels
Aug 29, 2016
Mark-Simulacrum
referenced this issue
in rust-lang/rfcs
Sep 11, 2016
Closed
RFC: Add unboxed, abstract return types #105
dimbleby
referenced this issue
in dimbleby/rust-c-ares
Oct 15, 2016
Closed
Remove boxing from futures example #26
frewsxcv
referenced this issue
in zargony/atom-language-rust
Oct 26, 2016
Open
Recognize minimal impl traits (Rust RFC 1522) #86
WaDelma
referenced this issue
in WaDelma/poisson
Nov 5, 2016
Open
Allow non-uniform point densities #5
frewsxcv
referenced this issue
in georust/geo
Nov 10, 2016
Open
Introduce trait for accessing geometries #67
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
Boscop
commented
Nov 15, 2016
|
I ran into this issue with impl Trait +'a and borrowing: #37790 |
jeanphilippeD
referenced this issue
in rust-lang-nursery/rust-bindgen
Nov 21, 2016
Closed
Replace clang.rs iterator with generic boxed Map<Range<>> #213
spinda
referenced this issue
Nov 29, 2016
Closed
Infinite size checker doesn't see through impl Trait, leading to rustc overflow #38064
Popog
referenced this issue
in bluss/indexmap
Dec 7, 2016
Closed
Move mutable key access to a trait #7
dimbleby
referenced this issue
in dimbleby/c-ares-resolver
Dec 20, 2016
Closed
unbox returned futures #1
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
IanWhitney
Jan 9, 2017
If I'm understanding this change correctly (and the chance of that is probably low!), then I think this playground code should work:
https://play.rust-lang.org/?gist=496ec05e6fa9d3a761df09c95297aa2a&version=nightly&backtrace=0
Both ThingOne and ThingTwo implement the Thing trait. build says it will return something that implements Thing, which it does. Yet it does not compile. So I'm clearly misunderstanding something.
IanWhitney
commented
Jan 9, 2017
|
If I'm understanding this change correctly (and the chance of that is probably low!), then I think this playground code should work: https://play.rust-lang.org/?gist=496ec05e6fa9d3a761df09c95297aa2a&version=nightly&backtrace=0 Both |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
eddyb
Jan 9, 2017
Member
That "something" must have a type, but in your case you have two conflicting types. @nikomatsakis has previously suggested making this work in general by creating e.g. ThingOne | ThingTwo as type mismatches appear.
|
That "something" must have a type, but in your case you have two conflicting types. @nikomatsakis has previously suggested making this work in general by creating e.g. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
WiSaGaN
Jan 9, 2017
Contributor
@eddyb could you elaborate on ThingOne | ThingTwo? Don't you need to have Box if we only know the type at run-time? Or is it a kind of enum?
|
@eddyb could you elaborate on |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
eddyb
Jan 9, 2017
Member
Yeah it could be an ad-hoc enum-like type that delegated trait method calls, where possible, to its variants.
|
Yeah it could be an ad-hoc |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
glaebhoerl
Jan 9, 2017
Contributor
I've wanted that kind of thing before too. The anonymous enums RFC: rust-lang/rfcs#1154
|
I've wanted that kind of thing before too. The anonymous enums RFC: rust-lang/rfcs#1154 |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
eddyb
Jan 9, 2017
Member
It's a rare case of something that works better if it's inference-driven, because if you only create these types on a mismatch, the variants are different (which is a problem with the generalized form).
Also you can get something out of not having pattern-matching (except in obviously disjoint cases?).
But IMO delegation sugar would "just work" in all relevant cases, even if you manage to get a T | T.
|
It's a rare case of something that works better if it's inference-driven, because if you only create these types on a mismatch, the variants are different (which is a problem with the generalized form). |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
glaebhoerl
Jan 9, 2017
Contributor
Could you spell out the other, implicit halves of those sentences? I don't understand most of it, and suspect I'm missing some context. Were you implicitly responding to the problems with union types? That RFC is simply anonymous enums, not union types - (T|T) would be exactly as problematic as Result<T, T>.
|
Could you spell out the other, implicit halves of those sentences? I don't understand most of it, and suspect I'm missing some context. Were you implicitly responding to the problems with union types? That RFC is simply anonymous enums, not union types - |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
Centril
Jul 4, 2018
Contributor
@lnicola You could just as well go with const bindings instead of let bindings, where the former is referentially transparent. Picking let (which happens to not be referentially transparent inside fn) is an arbitrary choice that I don't think is particularly intuitive. I think the intuitive view of type aliases is that they are referentially transparent (even if that word is not used) because they are aliases.
I'm also not looking at type as C preprocessor substitution because it has to be capture avoiding and respect generics (no SFINAE). Instead, I'm thinking of type precisely as I would a binding in a language like Idris or Agda where all bindings are pure.
I imagine beginners will be less likely to think of
impl Traitas existential vs. universtal types, but as "a thing that impls a Trait
That seems like a distinction without a difference to me. The jargon "existential" is not used, but I believe the user is intuitively linking it to the same concept as that of an existential type (which is nothing more than "some type Foo that impls Bar" in the context of Rust).
Re
type Foo = _, it overloads_with a completely unrelated meaning.
How so? type Foo = _; here aligns with the use of _ in other contexts where a type is expected.
It means "infer the real type", just as when you write .collect::<Vec<_>>().
It can also seem tricky to find in the documentation and/or Google.
Shouldn't be that difficult? "type alias underscore" should hopefully bring up the wanted result...?
Doesn't seem any different than searching for "type alias impl trait".
|
@lnicola You could just as well go with I'm also not looking at
That seems like a distinction without a difference to me. The jargon "existential" is not used, but I believe the user is intuitively linking it to the same concept as that of an existential type (which is nothing more than "some type Foo that impls Bar" in the context of Rust).
How so?
Shouldn't be that difficult? "type alias underscore" should hopefully bring up the wanted result...? |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
iopq
Jul 4, 2018
Google doesn't index special characters. If my StackOverflow question has an underscore in it, Google won't automatically index that for queries that contain the word underscore
iopq
commented
Jul 4, 2018
|
Google doesn't index special characters. If my StackOverflow question has an underscore in it, Google won't automatically index that for queries that contain the word underscore |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
rkruppe
Jul 4, 2018
Contributor
How so?
type Foo = _;here aligns with the use of _ in other contexts where a type is expected.
It means "infer the real type", just as when you write .collect::<Vec<_>>().
But this feature doesn't infer the type and give you a type alias for it, it creates an existential type which (outside of some limited scope like module or crate) doesn't unify with "the real type".
But this feature doesn't infer the type and give you a type alias for it, it creates an existential type which (outside of some limited scope like module or crate) doesn't unify with "the real type". |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
varkor
Jul 4, 2018
Contributor
Google doesn't index special characters.
This is no longer true (though possibly whitespace-dependent..?).
But this feature doesn't infer the type and give you a type alias for it, it creates an existential type which (outside of some limited scope like module or crate) doesn't unify with "the real type".
The suggested semantics of type Foo = _; is as an alternative to having an existential type alias, based entirely on inference. If that wasn't entirely clear, I'm going to follow up soon with something that should explain the intentions a bit better.
This is no longer true (though possibly whitespace-dependent..?).
The suggested semantics of |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
iopq
Jul 4, 2018
You still won't get good results with _ in your query, for whatever reason. If you search underscore, you get things with the word underscore in them. If you search _ you get everything that has an underscore, so I don't even know if it's relevant
iopq
commented
Jul 4, 2018
|
You still won't get good results with _ in your query, for whatever reason. If you search underscore, you get things with the word underscore in them. If you search _ you get everything that has an underscore, so I don't even know if it's relevant |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
stjepang
Jul 4, 2018
Contributor
Picking let (which happens to not be referentially transparent inside fn) is an arbitrary choice that I don't think is particularly intuitive. I think the intuitive view of type aliases is that they are referentially transparent (even if that word is not used) because they are aliases.
Sorry, I still can't wrap my head around this because my intuition is completely backwards.
For example, if we have type Foo = Bar, my intuition says:
"We're declaring Foo, which becomes the same type as Bar."
Then, if we write type Foo = impl Bar, my intuition says:
"We're declaring Foo, which becomes a type that implements Bar."
If Foo is just a textual alias for impl Bar, then that'd be super unintuitive to me. I like thinking of this as textual vs semantic aliases.
So if Foo can be replaced with impl Bar anywhere it appears, that's a textual alias, to me most reminiscent of macros and metaprogramming. But if Foo was assigned a meaning at the point of declaration and can be used in multiple places with that original meaning (not contextual meaning!), that's a semantic alias.
Also, I fail to understand the motivation behind contextual existential types anyhow. Would they ever be useful, considering that trait aliases can achieve the exact same thing?
Perhaps I find referential transparency unintuitive because of my non-Haskell background, who knows... :) But in any case, it's definitely not the kind of behavior I'd expect in Rust.
Sorry, I still can't wrap my head around this because my intuition is completely backwards. For example, if we have Then, if we write If So if Also, I fail to understand the motivation behind contextual existential types anyhow. Would they ever be useful, considering that trait aliases can achieve the exact same thing? Perhaps I find referential transparency unintuitive because of my non-Haskell background, who knows... :) But in any case, it's definitely not the kind of behavior I'd expect in Rust. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
rpjohnst
Jul 4, 2018
Contributor
If
Foowere really a type alias for an existential type(emphasis mine). I read that 'an' as 'a specific' which means
fandgwould not support different concrete return types, since they refer to the same existential type.
This is a misuse of the term "existential type," or at least a way that is at odds with @varkor's post. type Foo = impl Bar can appear to make Foo an alias for the type ∃ T. T: Trait- and if you substitute ∃ T. T: Trait everywhere you use Foo, even non-textually, you can get a different concrete type in each position.
The scoping of this ∃ T quantifier (expressed in your example as existential type _0) is the thing in question. It's tight like this in APIT- the caller can pass any value that satisfies ∃ T. T: Trait. But it's not in RPIT, and not in RFC 2071's existential type declarations, and not in your desugaring example- there, the quantifier is farther out, at the whole-function or whole-module level, and you deal with the same T everywhere.
Thus the ambiguity- we already have impl Trait placing its quantifier in different places depending on its position, so which one should we expect for type T = impl Trait? Some informal polls, as well as some after-the-fact-realizations by participants in the RFC 2071 thread, prove that it's not clear one way or the other.
This is why we want to move away from the interpretation of impl Trait as anything at all to do with existentials, and instead describe its semantics in terms of type inference. type T = _ does not have the same sort of ambiguity- there's still the surface-level "can't copy-paste the _ in place of T," but there's no longer "the single type that T is an alias of can mean multiple concrete types." (The opaque/doesn't-unify behavior is the thing @varkor is talking about following up on.)
This is a misuse of the term "existential type," or at least a way that is at odds with @varkor's post. The scoping of this Thus the ambiguity- we already have This is why we want to move away from the interpretation of |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
kennytm
Jul 4, 2018
Member
referential transparency
Just because a type alias currently being compatible with referential transparency, doesn't mean people expect the feature to follow it.
As an example, the const item is referential transparent (mentioned in #34511 (comment)), and that actually caused confusion to new and old users (rust-lang-nursery/rust-clippy#1560).
So I think for a Rust programmer referential transparency isn't the first thing they would think of.
Just because a type alias currently being compatible with referential transparency, doesn't mean people expect the feature to follow it. As an example, the So I think for a Rust programmer referential transparency isn't the first thing they would think of. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
Centril
Jul 4, 2018
Contributor
@stjepang @kennytm I'm not saying that everyone will expect that type aliases with type Foo = impl Trait; will act in a referentially transparent manner. But I think a non-trivial amount of users will, as evidenced by confusions in this thread and elsewhere (what @rpjohnst is referring to...). This is a problem, but perhaps not an insurmountable one. It is something to keep in mind tho as we move forward.
My current thinking on what should be done in this matter has moved in line with @varkor and @rpjohnst.
|
@stjepang @kennytm I'm not saying that everyone will expect that type aliases with My current thinking on what should be done in this matter has moved in line with @varkor and @rpjohnst. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
ExpHP
Jul 4, 2018
Contributor
re: referential transparency
type Foo<T> = (T, T);
type Bar = Foo<impl Copy>; // not equivalent to (impl Copy, impl Copy)that is to say, even generating new types at every instance is not referentially transparent in the context of generic type aliases.
|
re: referential transparency type Foo<T> = (T, T);
type Bar = Foo<impl Copy>; // not equivalent to (impl Copy, impl Copy)that is to say, even generating new types at every instance is not referentially transparent in the context of generic type aliases. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
alexreg
Jul 4, 2018
Contributor
@Centril I raise my hand when it comes to expecting referential transparency for Foo in type Foo = impl Bar;. With type Foo: Bar = _; however, I would not expect referential transparency.
|
@Centril I raise my hand when it comes to expecting referential transparency for |
bergus
referenced this issue
in rust-lang/rfcs
Jul 9, 2018
Open
RFC: add futures and task system to libcore #2418
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
rpjohnst
Jul 11, 2018
Contributor
It's also possible that we could extend return-position impl Trait to support multiple types, without any sort of enum impl Trait-like mechanism, by monomorphizing (pieces of) the caller. This strengthens the "impl Trait is always existential" interpretation, brings it closer in line with dyn Trait, and suggests an abstract type syntax that doesn't use impl Trait at all.
I wrote this up on internals here: https://internals.rust-lang.org/t/extending-impl-trait-to-allow-multiple-return-types/7921
|
It's also possible that we could extend return-position I wrote this up on internals here: https://internals.rust-lang.org/t/extending-impl-trait-to-allow-multiple-return-types/7921 |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
nrc
Jul 19, 2018
Member
Just a note for when we stabilise the new existential types - "existential" was always intended to be a temporary keyword (according to the RFC) and (IMO) is terrible. We must come up with something better before stabilising.
|
Just a note for when we stabilise the new existential types - "existential" was always intended to be a temporary keyword (according to the RFC) and (IMO) is terrible. We must come up with something better before stabilising. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
jan-hudec
Jul 22, 2018
The talk about “existential” types does not seem to be clearing things. I would say that impl Trait stands for a specific, inferred type implementing Trait. Described that way, type Foo = impl Bar is clearly a specific, always the same, type—and that is also the only interpretation that is actually useful: so it can be used in other contexts besides the one from which it was inferred, like in structs.
In this sense, it would make sense to also write impl Trait as _ : Trait.
It's also possible that we could extend return-position
impl Traitto support multiple types
That would make it strictly less useful IMO. The point of aliases to impl types is that a function can be defined as returning impl Foo, but the specific type still propagated through the program in other structs and stuff. That would work if the compiler implicitly generated suitable enum, but not with monomorphisation.
jan-hudec
commented
Jul 22, 2018
|
The talk about “existential” types does not seem to be clearing things. I would say that In this sense, it would make sense to also write
That would make it strictly less useful IMO. The point of aliases to |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
rpjohnst
Jul 22, 2018
Contributor
@jan-hudec Those ideas have come up in discussion on Discord, and there are some issues, primarily based around the fact that the current interpretation of return-position and argument-position impl Trait are inconsistent.
Making impl Trait stand for a specific inferred type is a good option, but for it to fix that inconsistency, it must be a different kind of type inference than Rust has today- it must infer polymorphic types so that it can preserve the current behavior of argument-position impl Trait. This is probably the most straightforward way to go, but it's not as simple as you say.
For example, once impl Trait means "use this new type of inference to find a polymorphic-as-possible type that implements Trait," type Foo = impl Bar starts to imply things about modules. The RFC 2071 rules around how to infer an abstract type say that all uses must independently infer the same type, but this polymorphic inference would at least imply that more is possible. And if we ever got parametrized modules (even just over lifetimes, a far more plausible idea), there would be questions around that interaction.
There's also the fact that some people will always interpret the type Foo = impl Bar syntax as an alias for an existential, regardless of whether they understand the word "existential" and regardless of how we teach it. So picking an alternative syntax, even if it happens to work out with the inference-based interpretation, is probably still a good idea.
Further, while the _: Trait syntax is actually what inspired the discussion around the inference-based interpretation in the first place, it does not do what we want. First, the inference implied by _ is not polymorphic, so that's a bad analogy to the rest of the language. Second, _ implies that the actual type is visible elsewhere, while impl Trait is specifically designed to hide the actual type.
Finally, the reason I wrote that monomorphization proposal was from the angle of finding another way to unify the meaning of argument and return-position impl Trait. And while yes, it does mean that -> impl Trait no longer guarantees a single concrete type, we don't currently have a way to take advantage of that anyway. And the proposed solutions are all annoying workarounds- extra boilerplate abstract type tricks, typeof, etc. Forcing everyone who wants to rely on single-type behavior to also name that single type via the abstract type syntax (whatever it may be) is arguably a benefit overall.
|
@jan-hudec Those ideas have come up in discussion on Discord, and there are some issues, primarily based around the fact that the current interpretation of return-position and argument-position Making For example, once There's also the fact that some people will always interpret the Further, while the Finally, the reason I wrote that monomorphization proposal was from the angle of finding another way to unify the meaning of argument and return-position |
This was referenced Jul 22, 2018
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
KodrAus
Jul 23, 2018
Those ideas have come up in discussion on Discord, and there are some issues, primarily based around the fact that the current interpretation of return-position and argument-position
impl Traitare inconsistent.
Personally, I don't find this inconsistency to be a problem in practice. The scope in which concrete types are determined for argument position vs return position vs type position seem to work out fairly intuitively.
KodrAus
commented
Jul 23, 2018
Personally, I don't find this inconsistency to be a problem in practice. The scope in which concrete types are determined for argument position vs return position vs type position seem to work out fairly intuitively. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
iopq
Jul 23, 2018
I have a function where the caller decides its return type. Of course, I can't use impl Trait there. It's not as intuitive as you imply until you understand the difference.
iopq
commented
Jul 23, 2018
|
I have a function where the caller decides its return type. Of course, I can't use impl Trait there. It's not as intuitive as you imply until you understand the difference. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
rpjohnst
Jul 23, 2018
Contributor
Personally, I don't find this inconsistency to be a problem in practice.
Indeed. What this suggests to me is not that we should ignore the inconsistency, but that we should re-explain the design so that it's consistent (for example, by explaining it as polymorphic type inference). This way, future extensions (RFC 2071, etc.) can be checked against the new, consistent interpretation to prevent things from becoming confusing.
Indeed. What this suggests to me is not that we should ignore the inconsistency, but that we should re-explain the design so that it's consistent (for example, by explaining it as polymorphic type inference). This way, future extensions (RFC 2071, etc.) can be checked against the new, consistent interpretation to prevent things from becoming confusing. |
dtolnay
referenced this issue
Jul 23, 2018
Open
Accept let-binding as a defining use of existential type #52653
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
mikeyhew
Jul 28, 2018
Contributor
Forcing everyone who wants to rely on single-type behavior to also name that single type via the abstract type syntax (whatever it may be) is arguably a benefit overall.
For some cases I agree with that sentiment, but it doesn't work with closures or generators, and is unergonomic for a lot of cases where you don't care what the type is and all you care about is that it implements a certain trait, e.g. with iterator combinators.
For some cases I agree with that sentiment, but it doesn't work with closures or generators, and is unergonomic for a lot of cases where you don't care what the type is and all you care about is that it implements a certain trait, e.g. with iterator combinators. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
rpjohnst
Jul 28, 2018
Contributor
@mikeyhew You misunderstand me- it works fine for closures or other unnameable types, because I'm talking about inventing a name via RFC 2071 abstract type syntax. You have to invent a name regardless if you're going to use the single type anywhere else.
|
@mikeyhew You misunderstand me- it works fine for closures or other unnameable types, because I'm talking about inventing a name via RFC 2071 |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
|
@rpjohnst oh I see, thanks for clarifying |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
fasihrana
commented
Jul 29, 2018
|
Waiting for |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
Nemo157
Jul 30, 2018
Contributor
As another vote for let x: impl Trait it'll simplify some of the futures examples, here's an example example, currently it is using a function just to get the ability to use impl Trait:
fn make_sink_async() -> impl Future<Output = Result<
impl Sink<SinkItem = T, SinkError = E>,
E,
>> { // ... }instead this could be written as a normal let binding:
let future_sink: impl Future<Output = Result<
impl Sink<SinkItem = T, SinkError = E>,
E,
>> = // ...;|
As another vote for fn make_sink_async() -> impl Future<Output = Result<
impl Sink<SinkItem = T, SinkError = E>,
E,
>> { // ... }instead this could be written as a normal let binding: let future_sink: impl Future<Output = Result<
impl Sink<SinkItem = T, SinkError = E>,
E,
>> = // ...; |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
oli-obk
Jul 30, 2018
Contributor
I can mentor someone through implementing let x: impl Trait if desired. It's not impossibly hard to do, but definitely not easy either. An entry point:
Similarly to how we visit the return type impl Trait in https://github.com/rust-lang/rust/blob/master/src/librustc/hir/lowering.rs#L3159 we need to visit the type of locals in https://github.com/rust-lang/rust/blob/master/src/librustc/hir/lowering.rs#L3159 and make sure their newly generated existential items are returned together with the local.
Then, when visiting the type of locals, be sure to set ExistentialContext to Return to actually enable it.
This should already get us very far. Not sure if all the way, it's not 100% like return position impl trait, but mostly should behave like it.
|
I can mentor someone through implementing Similarly to how we visit the return type impl Trait in https://github.com/rust-lang/rust/blob/master/src/librustc/hir/lowering.rs#L3159 we need to visit the type of locals in https://github.com/rust-lang/rust/blob/master/src/librustc/hir/lowering.rs#L3159 and make sure their newly generated existential items are returned together with the local. Then, when visiting the type of locals, be sure to set This should already get us very far. Not sure if all the way, it's not 100% like return position impl trait, but mostly should behave like it. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
jan-hudec
Jul 30, 2018
Those ideas have come up in discussion on Discord, and there are some issues, primarily based around the fact that the current interpretation of return-position and argument-position impl Trait are inconsistent.
Takes us back to the scopes you talked about in your article. And I think they actually correspond to the enclosing “parenthesis”: for argument position it is the argument list, for return position it is the function—and for the alias it would be the scope in which the alias is defined.
jan-hudec
commented
Jul 30, 2018
Takes us back to the scopes you talked about in your article. And I think they actually correspond to the enclosing “parenthesis”: for argument position it is the argument list, for return position it is the function—and for the alias it would be the scope in which the alias is defined. |
burdges
referenced this issue
in rust-random/rand
Aug 1, 2018
Closed
redesign SampleUniform for custom types to use impl Trait #577
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
varkor
Aug 5, 2018
Contributor
I've opened an RFC proposing a resolution to the existential type concrete syntax, based on the discussion in this thread, the original RFC and synchronous discussions: rust-lang/rfcs#2515.
|
I've opened an RFC proposing a resolution to the |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
Nemo157
Aug 5, 2018
Contributor
The current existential type implementation can't be used to represent all current return position impl Trait definitions, since impl Trait captures every generic type argument even if unused it should be possible to do the same with existential type, but you get unused type parameter warnings: (playground)
fn foo<T>(_: T) -> impl ::std::fmt::Display {
5
}
existential type Bar<T>: ::std::fmt::Display;
fn bar<T>(_: T) -> Bar<T> {
5
}This can matter because the type parameters can have internal lifetimes that restrict the lifetime of the returned impl Trait even though the value itself is unused, remove the <T> from Bar in the playground above to see that the call to foo fails but bar works.
|
The current existential type implementation can't be used to represent all current return position fn foo<T>(_: T) -> impl ::std::fmt::Display {
5
}
existential type Bar<T>: ::std::fmt::Display;
fn bar<T>(_: T) -> Bar<T> {
5
}This can matter because the type parameters can have internal lifetimes that restrict the lifetime of the returned |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
oli-obk
Aug 6, 2018
Contributor
The current existential type implementation can't be used to represent all current return position impl Trait definitions
you can, it's just very inconvenient. You can return a newtype with a PhantomData field + actual data field and implement the trait as forwarding to the actual data field
you can, it's just very inconvenient. You can return a newtype with a |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
alexreg
Aug 6, 2018
Contributor
@oli-obk Thanks for the additional advice. With your previous advice and some from @cramertj, I could probably have a go at it shortly.
@fasihrana @Nemo157 See above. Maybe in a few weeks! :-)
|
@oli-obk Thanks for the additional advice. With your previous advice and some from @cramertj, I could probably have a go at it shortly. @fasihrana @Nemo157 See above. Maybe in a few weeks! :-) |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
Arnavion
commented
Aug 6, 2018
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
|
I implemented it this way very much on purpose |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
cramertj
Aug 6, 2018
Member
@Arnavion Yes, this is intentional, and matches the way that other item declarations (e.g. nested functions) work in Rust.
|
@Arnavion Yes, this is intentional, and matches the way that other item declarations (e.g. nested functions) work in Rust. |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
vi
Aug 8, 2018
Contributor
Was interaction between existential_type and never_type already discussed?
Maybe ! should be able to fill in any existential type regardless of traits involved.
existential type Mystery : TraitThatIsHardToEvenStartImplementing;
fn hack_to_make_it_compile() -> Mystery { unimplemented!() }Or shall there be some special untouchable type serving as type-level unimplemented!() that is able to automatically satisfy any existential type?
|
Was interaction between Maybe existential type Mystery : TraitThatIsHardToEvenStartImplementing;
fn hack_to_make_it_compile() -> Mystery { unimplemented!() }Or shall there be some special untouchable type serving as type-level |
This comment has been minimized.
Show comment
Hide comment
This comment has been minimized.
daboross
Aug 9, 2018
Contributor
@vi I think that would fall under the general "never type should implement all traits without any non-default non-self methods or associated types". I don't know where that would be tracked, though.
|
@vi I think that would fall under the general "never type should implement all traits without any non-default non-self methods or associated types". I don't know where that would be tracked, though. |
aturon commentedJun 27, 2016
•
edited by nikomatsakis
Edited 28 times
-
nikomatsakis
edited Jul 25, 2018 (most recent)
-
Centril
edited Jun 8, 2018
-
nikomatsakis
edited May 21, 2018
-
withoutboats
edited May 20, 2018
-
pietroalbini
edited Apr 29, 2018
-
jrozner
edited Apr 7, 2018
-
jrozner
edited Apr 7, 2018
-
cramertj
edited Mar 22, 2018
-
Centril
edited Feb 27, 2018
-
cramertj
edited Feb 27, 2018
-
cramertj
edited Feb 27, 2018
-
nikomatsakis
edited Feb 22, 2018
-
cramertj
edited Feb 7, 2018
-
cramertj
edited Jan 22, 2018
-
cramertj
edited Jan 12, 2018
-
cramertj
edited Jan 12, 2018
-
shepmaster
edited Dec 20, 2017
-
nikomatsakis
edited Dec 12, 2017
-
nikomatsakis
edited Dec 11, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Dec 6, 2017
-
nikomatsakis
edited Nov 15, 2017
-
nikomatsakis
edited Oct 23, 2017
Implementation status
The basic feature as specified in RFC 1522 is implemented, however there have been revisions that are still in need of work:
let x: impl Traitstaticandconst T: impl Traitabstract typeRFCs
There have been a number of RFCs regarding impl trait, all of which are tracked by this central tracking issue.
abstract typein modules and implslet,const, andstaticpositionsimpl Traitanddyn Traitwith multiple boundsUnresolved questions
The implementation has raised a number of interesting questions as well:
implkeyword when parsing types? Discussion: 1Sendforwhere F: Fn() -> impl Foo + Send?impl Traitafter->infntypes or parentheses sugar? #45994fn foo<T>(x: impl Iterator<Item = T>>)?impl Traitas arguments in the list, permitting migrationexistential type Foo: Barortype Foo = impl Bar? (see here for discussion)existential typein an impl be just items of the impl, or include nested items within the impl functions etc? (see here for example)