Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit ae4a479

Browse files
committedMar 14, 2025
Improve upvar analysis for deref of child capture
1 parent ecade53 commit ae4a479

File tree

4 files changed

+128
-6
lines changed

4 files changed

+128
-6
lines changed
 

‎compiler/rustc_hir_typeck/src/upvar.rs

+19-6
Original file line numberDiff line numberDiff line change
@@ -1862,8 +1862,9 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
18621862
///
18631863
/// (1.) Are we borrowing data owned by the parent closure? We can determine if
18641864
/// that is the case by checking if the parent capture is by move, EXCEPT if we
1865-
/// apply a deref projection, which means we're reborrowing a reference that we
1866-
/// captured by move.
1865+
/// apply a deref projection of an immutable reference, reborrows of immutable
1866+
/// references which aren't restricted to the LUB of the lifetimes of the deref
1867+
/// chain. This is why `&'short mut &'long T` can be reborrowed as `&'long T`.
18671868
///
18681869
/// ```rust
18691870
/// let x = &1i32; // Let's call this lifetime `'1`.
@@ -1902,10 +1903,22 @@ fn should_reborrow_from_env_of_parent_coroutine_closure<'tcx>(
19021903
) -> bool {
19031904
// (1.)
19041905
(!parent_capture.is_by_ref()
1905-
&& !matches!(
1906-
child_capture.place.projections.get(parent_capture.place.projections.len()),
1907-
Some(Projection { kind: ProjectionKind::Deref, .. })
1908-
))
1906+
// This is just inlined `place.deref_tys()` but truncated to just
1907+
// the child projections. Namely, look for a `&T` deref, since we
1908+
// can always extend `&'short mut &'long T` to `&'long T`.
1909+
&& !child_capture
1910+
.place
1911+
.projections
1912+
.iter()
1913+
.enumerate()
1914+
.skip(parent_capture.place.projections.len())
1915+
.any(|(idx, proj)| {
1916+
matches!(proj.kind, ProjectionKind::Deref)
1917+
&& matches!(
1918+
child_capture.place.ty_before_projection(idx).kind(),
1919+
ty::Ref(.., ty::Mutability::Not)
1920+
)
1921+
}))
19091922
// (2.)
19101923
|| matches!(child_capture.info.capture_kind, UpvarCapture::ByRef(ty::BorrowKind::Mutable))
19111924
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
//@ edition: 2021
2+
//@ check-pass
3+
4+
#![feature(impl_trait_in_bindings)]
5+
6+
struct FooS {
7+
precise: i32,
8+
}
9+
10+
fn ref_inside_mut(f: &mut &FooS) {
11+
let x: impl AsyncFn() = async move || {
12+
let y = &f.precise;
13+
};
14+
}
15+
16+
fn mut_inside_ref(f: &&mut FooS) {
17+
let x: impl AsyncFn() = async move || {
18+
let y = &f.precise;
19+
};
20+
}
21+
22+
fn mut_ref_inside_mut(f: &mut &mut FooS) {
23+
let x: impl AsyncFn() = async move || {
24+
let y = &f.precise;
25+
};
26+
}
27+
28+
fn ref_inside_box(f: Box<&FooS>) {
29+
let x: impl AsyncFn() = async move || {
30+
let y = &f.precise;
31+
};
32+
}
33+
34+
fn box_inside_ref(f: &Box<FooS>) {
35+
let x: impl AsyncFn() = async move || {
36+
let y = &f.precise;
37+
};
38+
}
39+
40+
fn box_inside_box(f: Box<Box<FooS>>) {
41+
let x: impl AsyncFn() = async move || {
42+
let y = &f.precise;
43+
};
44+
}
45+
46+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
//@ edition: 2021
2+
3+
#![feature(impl_trait_in_bindings)]
4+
5+
struct FooS {
6+
precise: i32,
7+
}
8+
9+
fn ref_inside_mut(f: &mut &FooS) {
10+
let x: impl Fn() -> _ = async move || {
11+
let y = &f.precise;
12+
};
13+
}
14+
15+
fn mut_inside_ref(f: &&mut FooS) {
16+
let x: impl Fn() -> _ = async move || {
17+
let y = &f.precise;
18+
};
19+
}
20+
21+
// Expected to fail, no immutable reference here.
22+
fn mut_ref_inside_mut(f: &mut &mut FooS) {
23+
let x: impl Fn() -> _ = async move || {
24+
//~^ ERROR async closure does not implement `Fn`
25+
let y = &f.precise;
26+
};
27+
}
28+
29+
fn ref_inside_box(f: Box<&FooS>) {
30+
let x: impl Fn() -> _ = async move || {
31+
let y = &f.precise;
32+
};
33+
}
34+
35+
fn box_inside_ref(f: &Box<FooS>) {
36+
let x: impl Fn() -> _ = async move || {
37+
let y = &f.precise;
38+
};
39+
}
40+
41+
// Expected to fail, no immutable reference here.
42+
fn box_inside_box(f: Box<Box<FooS>>) {
43+
let x: impl Fn() -> _ = async move || {
44+
//~^ ERROR async closure does not implement `Fn`
45+
let y = &f.precise;
46+
};
47+
}
48+
49+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
error: async closure does not implement `Fn` because it captures state from its environment
2+
--> $DIR/imm-deref-not-lending.rs:23:29
3+
|
4+
LL | let x: impl Fn() -> _ = async move || {
5+
| ^^^^^^^^^^^^^
6+
7+
error: async closure does not implement `Fn` because it captures state from its environment
8+
--> $DIR/imm-deref-not-lending.rs:43:29
9+
|
10+
LL | let x: impl Fn() -> _ = async move || {
11+
| ^^^^^^^^^^^^^
12+
13+
error: aborting due to 2 previous errors
14+

0 commit comments

Comments
 (0)
Failed to load comments.