Skip to content

Commit 48c5e0c

Browse files
authored
Rollup merge of #103034 - nathanwhit:let-chains-rhs-temporaries, r=wesleywiser
Let expressions on RHS shouldn't be terminating scopes Fixes #100276. Before this PR, we were unconditionally marking the RHS of short-circuiting binary expressions as a terminating scope. In the case of a let chain where the `let` expression was on the RHS, this meant that temporaries within the `let` expr would only live until the end of the expression. Since this only affected the RHS, this led to surprising behavior ([example](https://play.rust-lang.org/?version=nightly&mode=debug&edition=2021&gist=d1b0a5d1f01882f9c89c2194a75eb19f)). After this PR, we only mark the RHS as a terminating scope if it is not a `let` expression.
2 parents 5c2c476 + 3041bc9 commit 48c5e0c

File tree

4 files changed

+24
-8
lines changed

4 files changed

+24
-8
lines changed

compiler/rustc_hir_analysis/src/check/region.rs

+6-2
Original file line numberDiff line numberDiff line change
@@ -252,9 +252,13 @@ fn resolve_expr<'tcx>(visitor: &mut RegionResolutionVisitor<'tcx>, expr: &'tcx h
252252
) => {
253253
// For shortcircuiting operators, mark the RHS as a terminating
254254
// scope since it only executes conditionally.
255-
terminating(r.hir_id.local_id);
256-
}
257255

256+
// `Let` expressions (in a let-chain) shouldn't be terminating, as their temporaries
257+
// should live beyond the immediate expression
258+
if !matches!(r.kind, hir::ExprKind::Let(_)) {
259+
terminating(r.hir_id.local_id);
260+
}
261+
}
258262
hir::ExprKind::If(_, ref then, Some(ref otherwise)) => {
259263
terminating(then.hir_id.local_id);
260264
terminating(otherwise.hir_id.local_id);

src/test/ui/drop/drop_order.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -129,10 +129,10 @@ impl DropOrderCollector {
129129
// take the "else" branch
130130
if self.option_loud_drop(6).is_some() // 2
131131
&& self.option_loud_drop(5).is_some() // 1
132-
&& let None = self.option_loud_drop(7) { // 3
132+
&& let None = self.option_loud_drop(8) { // 4
133133
unreachable!();
134134
} else {
135-
self.print(8); // 4
135+
self.print(7); // 3
136136
}
137137

138138
// let exprs interspersed

src/test/ui/drop/issue-100276.rs

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
// check-pass
2+
// compile-flags: -Z validate-mir
3+
#![feature(let_chains)]
4+
5+
fn let_chains(entry: std::io::Result<std::fs::DirEntry>) {
6+
if let Ok(entry) = entry
7+
&& let Some(s) = entry.file_name().to_str()
8+
&& s.contains("")
9+
{}
10+
}
11+
12+
fn main() {}

src/test/ui/mir/mir_let_chains_drop_order.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ use std::panic;
1212
pub struct DropLogger<'a, T> {
1313
extra: T,
1414
id: usize,
15-
log: &'a panic::AssertUnwindSafe<RefCell<Vec<usize>>>
15+
log: &'a panic::AssertUnwindSafe<RefCell<Vec<usize>>>,
1616
}
1717

1818
impl<'a, T> Drop for DropLogger<'a, T> {
@@ -55,9 +55,9 @@ fn main() {
5555
else {
5656
// 10 is not constructed
5757
d(10, None)
58-
}
58+
},
5959
);
60-
assert_eq!(get(), vec![3, 8, 7, 1, 2]);
60+
assert_eq!(get(), vec![8, 7, 1, 3, 2]);
6161
}
6262
assert_eq!(get(), vec![0, 4, 6, 9, 5]);
6363

@@ -89,5 +89,5 @@ fn main() {
8989
panic::panic_any(InjectedFailure)
9090
);
9191
});
92-
assert_eq!(get(), vec![14, 19, 20, 17, 15, 11, 18, 16, 12, 13]);
92+
assert_eq!(get(), vec![20, 17, 15, 11, 19, 18, 16, 12, 14, 13]);
9393
}

0 commit comments

Comments
 (0)