-
Notifications
You must be signed in to change notification settings - Fork 12.7k
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
impl Trait returned from trait method without arguments not detected as 'static #119773
Comments
It's pretty silly, but the returned closure doesn't have to be impl<'a> Original for &'a str {
// This fails to compile if you add `+ 'static` bound to the RPITIT
fn f() -> impl Fn() {
let silly: &'a str = "";
move || { let _silly = silly; }
}
} |
@QuineDot Thanks for the interesting example. I think it's consistent with what was reported in the ticket, only provides the flip side of the coin, an example that compiles with the current behavior, but would no longer compiled if it were to change. In other words, this code compiles, and I wouldn't expect it to: trait Original {
fn f() -> impl Fn();
}
impl<'a> Original for &'a str {
fn f() -> impl Fn() {
|| {
let x: &'a str = "foo";
}
}
} It compiles because the lifetime bounds from the type that implements the trait are apparently baked into the return type, even when the method has no arguments. That seems wrong - I'd argue that, to get such behavior, one should be explicit about the lifetime carried around by the trait, as in: trait Original<'a>: 'a {
fn f() -> impl Fn() + 'a;
}
impl<'a> Original<'a> for &'a str {
fn f() -> impl Fn() {
|| {
let x: &'a str = "foo";
}
}
} The current behavior for methods that don't accept |
Anyway, let me present a less silly example for the more general pattern. This implementation exists: impl<T> Default for &mut [T] {
fn default() -> Self {
// (possible because empty slices are magic)
&mut []
}
} And this works for any sized impl<T> Default for &'static mut [T] { Moreover, this implementation is useful for things like this. struct MyIter<'a, T> {
slice: &'a mut [T],
}
impl<'a, T> Iterator for MyIter<'a, T> {
type Item = &'a mut T;
fn next<'short>(&'short mut self) -> Option<Self::Item> {
// We can't borrow a `&'a mut T` through `&'short mut Self`, so get
// the slice out from behind `&mut self`. Note that `'a` is invariant
// here as well (and that `T` has no `'static` requirement).
let slice = std::mem::take(&mut self.slice);
match slice {
[] => None,
[item, rest @ .. ] => {
// Restore the tail of the slice
self.slice = rest;
Some(item)
}
}
}
} A rule that a return value couldn't capture a lifetime from the implementing type only would require removing the |
When given ```rust trait Original { fn f() -> impl Fn(); } trait Erased { fn f(&self) -> Box<dyn Fn()>; } impl<T: Original> Erased for T { fn f(&self) -> Box<dyn Fn()> { Box::new(<T as Original>::f()) } } ``` emit a suggestion to further constrain the RPITIT, instead of what we did previously, suggest restricting the `Trait::{opaque}` type in a `where` clause: ``` error[E0310]: the associated type `<T as Original>::{opaque#0}` may not live long enough --> $DIR/missing-static-bound-from-impl.rs:11:9 | LL | Box::new(<T as Original>::f()) | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | | | the associated type `<T as Original>::{opaque#0}` must be valid for the static lifetime... | ...so that the type `impl Fn()` will meet its required lifetime bounds | help: consider adding an explicit lifetime bound | LL | fn f() -> impl Fn() + 'static; | +++++++++ ``` Fix rust-lang#119773.
When given ```rust trait Original { fn f() -> impl Fn(); } trait Erased { fn f(&self) -> Box<dyn Fn()>; } impl<T: Original> Erased for T { fn f(&self) -> Box<dyn Fn()> { Box::new(<T as Original>::f()) } } ``` avoid suggestion to restrict the `Trait::{opaque}` type in a `where` clause: ``` error[E0310]: the associated type `<T as Original>::{opaque#0}` may not live long enough --> $DIR/missing-static-bound-from-impl.rs:11:9 | LL | Box::new(<T as Original>::f()) | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | | | the associated type `<T as Original>::{opaque#0}` must be valid for the static lifetime... | ...so that the type `impl Fn()` will meet its required lifetime bounds ``` CC rust-lang#119773.
…ompiler-errors Account for RPITIT in E0310 explicit lifetime constraint suggestion When given ```rust trait Original { fn f() -> impl Fn(); } trait Erased { fn f(&self) -> Box<dyn Fn()>; } impl<T: Original> Erased for T { fn f(&self) -> Box<dyn Fn()> { Box::new(<T as Original>::f()) } } ``` emit do not emit an invalid suggestion restricting the `Trait::{opaque}` type in a `where` clause: ``` error[E0310]: the associated type `<T as Original>::{opaque#0}` may not live long enough --> $DIR/missing-static-bound-from-impl.rs:11:9 | LL | Box::new(<T as Original>::f()) | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | | | the associated type `<T as Original>::{opaque#0}` must be valid for the static lifetime... | ...so that the type `impl Fn()` will meet its required lifetime bounds ``` Partially address rust-lang#119773. Ideally we'd suggest modifying `Erased::f` instead. r? `@compiler-errors`
Rollup merge of rust-lang#121435 - estebank:rpitit-static-119773, r=compiler-errors Account for RPITIT in E0310 explicit lifetime constraint suggestion When given ```rust trait Original { fn f() -> impl Fn(); } trait Erased { fn f(&self) -> Box<dyn Fn()>; } impl<T: Original> Erased for T { fn f(&self) -> Box<dyn Fn()> { Box::new(<T as Original>::f()) } } ``` emit do not emit an invalid suggestion restricting the `Trait::{opaque}` type in a `where` clause: ``` error[E0310]: the associated type `<T as Original>::{opaque#0}` may not live long enough --> $DIR/missing-static-bound-from-impl.rs:11:9 | LL | Box::new(<T as Original>::f()) | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | | | the associated type `<T as Original>::{opaque#0}` must be valid for the static lifetime... | ...so that the type `impl Fn()` will meet its required lifetime bounds ``` Partially address rust-lang#119773. Ideally we'd suggest modifying `Erased::f` instead. r? `@compiler-errors`
I tried this code:
Playground
I expected the code to compile successfully, but it failed on Rust 1.75.0 with the following error:
I understand that
Box<dyn Fn()>
is sugar forBox<dyn Fn() + 'static>
, but in this case theimpl Fn()
return is'static
. It has to be, becauseOriginal::f()
takes no arguments, so the value it returns cannot reference any non-static lifetime.The workaround is to change the return type of
Erased::f()
fromBox<dyn Fn()>
toBox<dyn Fn() + '_>
, which is allows the code to compile. (Playground.) (Another workaround is to change the definition ofOriginal::f()
to returnimpl Fn() + 'static
, but in the original codeOriginal
is provided by a different crate and cannot be modified.)A second, and minor issue is that the compiler's suggestion is not actionable because the opaque type returned from
Original::f()
cannot be named.The text was updated successfully, but these errors were encountered: