Skip to content

Commit d837407

Browse files
authored
Rollup merge of #80319 - jyn514:async-lifetimes, r=tmandry
Fix elided lifetimes shown as `'_` on async functions Closes #63037. r? `@tmandry` on the implementation, `@Darksonn` on the test cases.
2 parents c24fcad + ceb66ad commit d837407

File tree

2 files changed

+67
-2
lines changed

2 files changed

+67
-2
lines changed

src/librustdoc/clean/mod.rs

+23-2
Original file line numberDiff line numberDiff line change
@@ -638,6 +638,18 @@ impl Clean<Generics> for hir::Generics<'_> {
638638
_ => false,
639639
}
640640
}
641+
/// This can happen for `async fn`, e.g. `async fn f<'_>(&'_ self)`.
642+
///
643+
/// See [`lifetime_to_generic_param`] in [`rustc_ast_lowering`] for more information.
644+
///
645+
/// [`lifetime_to_generic_param`]: rustc_ast_lowering::LoweringContext::lifetime_to_generic_param
646+
fn is_elided_lifetime(param: &hir::GenericParam<'_>) -> bool {
647+
match param.kind {
648+
hir::GenericParamKind::Lifetime { kind: hir::LifetimeParamKind::Elided } => true,
649+
_ => false,
650+
}
651+
}
652+
641653
let impl_trait_params = self
642654
.params
643655
.iter()
@@ -656,7 +668,7 @@ impl Clean<Generics> for hir::Generics<'_> {
656668
.collect::<Vec<_>>();
657669

658670
let mut params = Vec::with_capacity(self.params.len());
659-
for p in self.params.iter().filter(|p| !is_impl_trait(p)) {
671+
for p in self.params.iter().filter(|p| !is_impl_trait(p) && !is_elided_lifetime(p)) {
660672
let p = p.clean(cx);
661673
params.push(p);
662674
}
@@ -1437,7 +1449,16 @@ impl Clean<Type> for hir::Ty<'_> {
14371449
TyKind::Never => Never,
14381450
TyKind::Ptr(ref m) => RawPointer(m.mutbl, box m.ty.clean(cx)),
14391451
TyKind::Rptr(ref l, ref m) => {
1440-
let lifetime = if l.is_elided() { None } else { Some(l.clean(cx)) };
1452+
// There are two times a `Fresh` lifetime can be created:
1453+
// 1. For `&'_ x`, written by the user. This corresponds to `lower_lifetime` in `rustc_ast_lowering`.
1454+
// 2. For `&x` as a parameter to an `async fn`. This corresponds to `elided_ref_lifetime in `rustc_ast_lowering`.
1455+
// See #59286 for more information.
1456+
// Ideally we would only hide the `'_` for case 2., but I don't know a way to distinguish it.
1457+
// Turning `fn f(&'_ self)` into `fn f(&self)` isn't the worst thing in the world, though;
1458+
// there's no case where it could cause the function to fail to compile.
1459+
let elided =
1460+
l.is_elided() || matches!(l.name, LifetimeName::Param(ParamName::Fresh(_)));
1461+
let lifetime = if elided { None } else { Some(l.clean(cx)) };
14411462
BorrowedRef { lifetime, mutability: m.mutbl, type_: box m.ty.clean(cx) }
14421463
}
14431464
TyKind::Slice(ref ty) => Slice(box ty.clean(cx)),

src/test/rustdoc/async-fn.rs

+44
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
// ignore-tidy-linelength
12
// edition:2018
23
#![feature(min_const_generics)]
34

@@ -48,7 +49,50 @@ impl Foo {
4849
pub async fn mut_self(mut self, mut first: usize) {}
4950
}
5051

52+
pub trait Pattern<'a> {}
53+
5154
pub trait Trait<const N: usize> {}
5255
// @has async_fn/fn.const_generics.html
5356
// @has - '//pre[@class="rust fn"]' 'pub async fn const_generics<const N: usize>(_: impl Trait<N>)'
5457
pub async fn const_generics<const N: usize>(_: impl Trait<N>) {}
58+
59+
// test that elided lifetimes are properly elided and not displayed as `'_`
60+
// regression test for #63037
61+
// @has async_fn/fn.elided.html
62+
// @has - '//pre[@class="rust fn"]' 'pub async fn elided(foo: &str) -> &str'
63+
pub async fn elided(foo: &str) -> &str {}
64+
// This should really be shown as written, but for implementation reasons it's difficult.
65+
// See `impl Clean for TyKind::Rptr`.
66+
// @has async_fn/fn.user_elided.html
67+
// @has - '//pre[@class="rust fn"]' 'pub async fn user_elided(foo: &str) -> &str'
68+
pub async fn user_elided(foo: &'_ str) -> &str {}
69+
// @has async_fn/fn.static_trait.html
70+
// @has - '//pre[@class="rust fn"]' 'pub async fn static_trait(foo: &str) -> Box<dyn Bar>'
71+
pub async fn static_trait(foo: &str) -> Box<dyn Bar> {}
72+
// @has async_fn/fn.lifetime_for_trait.html
73+
// @has - '//pre[@class="rust fn"]' "pub async fn lifetime_for_trait(foo: &str) -> Box<dyn Bar + '_>"
74+
pub async fn lifetime_for_trait(foo: &str) -> Box<dyn Bar + '_> {}
75+
// @has async_fn/fn.elided_in_input_trait.html
76+
// @has - '//pre[@class="rust fn"]' "pub async fn elided_in_input_trait(t: impl Pattern<'_>)"
77+
pub async fn elided_in_input_trait(t: impl Pattern<'_>) {}
78+
79+
struct AsyncFdReadyGuard<'a, T> { x: &'a T }
80+
81+
impl Foo {
82+
// @has async_fn/struct.Foo.html
83+
// @has - '//h4[@class="method"]' 'pub async fn complicated_lifetimes( &self, context: &impl Bar) -> impl Iterator<Item = &usize>'
84+
pub async fn complicated_lifetimes(&self, context: &impl Bar) -> impl Iterator<Item = &usize> {}
85+
// taken from `tokio` as an example of a method that was particularly bad before
86+
// @has - '//h4[@class="method"]' "pub async fn readable<T>(&self) -> Result<AsyncFdReadyGuard<'_, T>, ()>"
87+
pub async fn readable<T>(&self) -> Result<AsyncFdReadyGuard<'_, T>, ()> {}
88+
// @has - '//h4[@class="method"]' "pub async fn mut_self(&mut self)"
89+
pub async fn mut_self(&mut self) {}
90+
}
91+
92+
// test named lifetimes, just in case
93+
// @has async_fn/fn.named.html
94+
// @has - '//pre[@class="rust fn"]' "pub async fn named<'a, 'b>(foo: &'a str) -> &'b str"
95+
pub async fn named<'a, 'b>(foo: &'a str) -> &'b str {}
96+
// @has async_fn/fn.named_trait.html
97+
// @has - '//pre[@class="rust fn"]' "pub async fn named_trait<'a, 'b>(foo: impl Pattern<'a>) -> impl Pattern<'b>"
98+
pub async fn named_trait<'a, 'b>(foo: impl Pattern<'a>) -> impl Pattern<'b> {}

0 commit comments

Comments
 (0)