Skip to content

Commit 72b6f70

Browse files
committed
Use collect_in_band_defs for async lifetime captures.
1 parent 6e7ec0c commit 72b6f70

File tree

2 files changed

+55
-54
lines changed

2 files changed

+55
-54
lines changed

compiler/rustc_ast_lowering/src/lib.rs

+44-54
Original file line numberDiff line numberDiff line change
@@ -648,15 +648,13 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
648648
&mut self,
649649
f: impl FnOnce(&mut Self) -> T,
650650
) -> (Vec<(Span, ParamName)>, T) {
651-
assert!(!self.is_collecting_in_band_lifetimes);
652-
assert!(self.lifetimes_to_define.is_empty());
653-
self.is_collecting_in_band_lifetimes = true;
651+
let was_collecting = std::mem::replace(&mut self.is_collecting_in_band_lifetimes, true);
652+
let len = self.lifetimes_to_define.len();
654653

655654
let res = f(self);
656655

657-
self.is_collecting_in_band_lifetimes = false;
658-
659-
let lifetimes_to_define = std::mem::take(&mut self.lifetimes_to_define);
656+
let lifetimes_to_define = self.lifetimes_to_define.split_off(len);
657+
self.is_collecting_in_band_lifetimes = was_collecting;
660658
(lifetimes_to_define, res)
661659
}
662660

@@ -1688,18 +1686,29 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
16881686
// this is because the elided lifetimes from the return type
16891687
// should be figured out using the ordinary elision rules, and
16901688
// this desugaring achieves that.
1689+
1690+
debug!("lower_async_fn_ret_ty: in_scope_lifetimes={:#?}", self.in_scope_lifetimes);
1691+
debug!("lower_async_fn_ret_ty: lifetimes_to_define={:#?}", self.lifetimes_to_define);
1692+
1693+
// Calculate all the lifetimes that should be captured
1694+
// by the opaque type. This should include all in-scope
1695+
// lifetime parameters, including those defined in-band.
16911696
//
1692-
// The variable `input_lifetimes_count` tracks the number of
1693-
// lifetime parameters to the opaque type *not counting* those
1694-
// lifetimes elided in the return type. This includes those
1695-
// that are explicitly declared (`in_scope_lifetimes`) and
1696-
// those elided lifetimes we found in the arguments (current
1697-
// content of `lifetimes_to_define`). Next, we will process
1698-
// the return type, which will cause `lifetimes_to_define` to
1699-
// grow.
1700-
let input_lifetimes_count = self.in_scope_lifetimes.len() + self.lifetimes_to_define.len();
1701-
1702-
let mut lifetime_params = Vec::new();
1697+
// `lifetime_params` is a vector of tuple (span, parameter name, lifetime name).
1698+
1699+
// Input lifetime like `'a` or `'1`:
1700+
let mut lifetime_params: Vec<_> = self
1701+
.in_scope_lifetimes
1702+
.iter()
1703+
.cloned()
1704+
.map(|name| (name.ident().span, name, hir::LifetimeName::Param(name)))
1705+
.chain(
1706+
self.lifetimes_to_define
1707+
.iter()
1708+
.map(|&(span, name)| (span, name, hir::LifetimeName::Param(name))),
1709+
)
1710+
.collect();
1711+
17031712
self.with_hir_id_owner(opaque_ty_node_id, |this| {
17041713
// We have to be careful to get elision right here. The
17051714
// idea is that we create a lifetime parameter for each
@@ -1709,34 +1718,26 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
17091718
//
17101719
// Then, we will create `fn foo(..) -> Foo<'_, '_>`, and
17111720
// hence the elision takes place at the fn site.
1712-
let future_bound = this
1713-
.with_anonymous_lifetime_mode(AnonymousLifetimeMode::CreateParameter, |this| {
1714-
this.lower_async_fn_output_type_to_future_bound(output, fn_def_id, span)
1721+
let (lifetimes_to_define, future_bound) =
1722+
this.with_anonymous_lifetime_mode(AnonymousLifetimeMode::CreateParameter, |this| {
1723+
this.collect_in_band_defs(|this| {
1724+
this.lower_async_fn_output_type_to_future_bound(output, fn_def_id, span)
1725+
})
17151726
});
1716-
17171727
debug!("lower_async_fn_ret_ty: future_bound={:#?}", future_bound);
1728+
debug!("lower_async_fn_ret_ty: lifetimes_to_define={:#?}", lifetimes_to_define);
17181729

1719-
// Calculate all the lifetimes that should be captured
1720-
// by the opaque type. This should include all in-scope
1721-
// lifetime parameters, including those defined in-band.
1722-
//
1723-
// Note: this must be done after lowering the output type,
1724-
// as the output type may introduce new in-band lifetimes.
1725-
lifetime_params = this
1726-
.in_scope_lifetimes
1727-
.iter()
1728-
.cloned()
1729-
.map(|name| (name.ident().span, name))
1730-
.chain(this.lifetimes_to_define.iter().cloned())
1731-
.collect();
1732-
1733-
debug!("lower_async_fn_ret_ty: in_scope_lifetimes={:#?}", this.in_scope_lifetimes);
1734-
debug!("lower_async_fn_ret_ty: lifetimes_to_define={:#?}", this.lifetimes_to_define);
1730+
lifetime_params.extend(
1731+
// Output lifetime like `'_`:
1732+
lifetimes_to_define
1733+
.into_iter()
1734+
.map(|(span, name)| (span, name, hir::LifetimeName::Implicit(false))),
1735+
);
17351736
debug!("lower_async_fn_ret_ty: lifetime_params={:#?}", lifetime_params);
17361737

17371738
let generic_params =
1738-
this.arena.alloc_from_iter(lifetime_params.iter().map(|(span, hir_name)| {
1739-
this.lifetime_to_generic_param(*span, *hir_name, opaque_ty_def_id)
1739+
this.arena.alloc_from_iter(lifetime_params.iter().map(|&(span, hir_name, _)| {
1740+
this.lifetime_to_generic_param(span, hir_name, opaque_ty_def_id)
17401741
}));
17411742

17421743
let opaque_ty_item = hir::OpaqueTy {
@@ -1770,25 +1771,14 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
17701771
//
17711772
// For the "output" lifetime parameters, we just want to
17721773
// generate `'_`.
1773-
let mut generic_args = Vec::with_capacity(lifetime_params.len());
1774-
generic_args.extend(lifetime_params[..input_lifetimes_count].iter().map(
1775-
|&(span, hir_name)| {
1776-
// Input lifetime like `'a` or `'1`:
1774+
let generic_args =
1775+
self.arena.alloc_from_iter(lifetime_params.into_iter().map(|(span, _, name)| {
17771776
GenericArg::Lifetime(hir::Lifetime {
17781777
hir_id: self.next_id(),
17791778
span: self.lower_span(span),
1780-
name: hir::LifetimeName::Param(hir_name),
1779+
name,
17811780
})
1782-
},
1783-
));
1784-
generic_args.extend(lifetime_params[input_lifetimes_count..].iter().map(|&(span, _)|
1785-
// Output lifetime like `'_`.
1786-
GenericArg::Lifetime(hir::Lifetime {
1787-
hir_id: self.next_id(),
1788-
span: self.lower_span(span),
1789-
name: hir::LifetimeName::Implicit(false),
1790-
})));
1791-
let generic_args = self.arena.alloc_from_iter(generic_args);
1781+
}));
17921782

17931783
// Create the `Foo<...>` reference itself. Note that the `type
17941784
// Foo = impl Trait` is, internally, created as a child of the

src/test/ui/async-await/generics-and-bounds.rs

+11
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22
// edition:2018
33
// compile-flags: --crate-type lib
44

5+
#![feature(in_band_lifetimes)]
6+
57
use std::future::Future;
68

79
pub async fn simple_generic<T>() {}
@@ -71,6 +73,10 @@ pub fn call_with_ref_block<'a>(f: &'a (impl Foo + 'a)) -> impl Future<Output = (
7173
async move { f.foo() }
7274
}
7375

76+
pub fn call_with_ref_block_in_band(f: &'a (impl Foo + 'a)) -> impl Future<Output = ()> + 'a {
77+
async move { f.foo() }
78+
}
79+
7480
pub fn async_block_with_same_generic_params_unifies() {
7581
let mut a = call_generic_bound_block(FooType);
7682
a = call_generic_bound_block(FooType);
@@ -85,4 +91,9 @@ pub fn async_block_with_same_generic_params_unifies() {
8591
let f_two = FooType;
8692
let mut d = call_with_ref_block(&f_one);
8793
d = call_with_ref_block(&f_two);
94+
95+
let f_one = FooType;
96+
let f_two = FooType;
97+
let mut d = call_with_ref_block_in_band(&f_one);
98+
d = call_with_ref_block_in_band(&f_two);
8899
}

0 commit comments

Comments
 (0)