Lower let-else in MIR by dingxiangfei2009 · Pull Request #98574 · rust-lang/rust (original) (raw)
Conversation
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.Learn more about bidirectional Unicode characters
[ Show hidden characters]({{ revealButtonHref }})
rustbot added the T-compiler
Relevant to the compiler team, which will review and decide on the PR/issue.
label
bors added S-waiting-on-bors
Status: Waiting on bors to run and complete tests. Bors will change the label on completion.
and removed S-waiting-on-review
Status: Awaiting review from the assignee but also interested parties.
labels
Dylan-DPC added a commit to Dylan-DPC/rust that referenced this pull request
…li-obk
Lower let-else in MIR
This MR will switch to lower let-else statements in MIR building instead.
To lower let-else in MIR, we build a mini-switch two branches. One branch leads to the matching case, and the other leads to the else
block. This arrangement will allow temporary lifetime analysis running as-is so that the temporaries are properly extended according to the same rule applied to regular let
statements.
Fix rust-lang#98672
Dylan-DPC added a commit to Dylan-DPC/rust that referenced this pull request
…li-obk
Lower let-else in MIR
This MR will switch to lower let-else statements in MIR building instead.
To lower let-else in MIR, we build a mini-switch two branches. One branch leads to the matching case, and the other leads to the else
block. This arrangement will allow temporary lifetime analysis running as-is so that the temporaries are properly extended according to the same rule applied to regular let
statements.
Fix rust-lang#98672
This was referenced
Jul 13, 2022
est31 mentioned this pull request
bors added a commit to rust-lang-ci/rust that referenced this pull request
Rollup of 5 pull requests
Successful merges:
- rust-lang#98574 (Lower let-else in MIR)
- rust-lang#99011 (
UnsafeCell
blocks niches inside its nested type from being available outside) - rust-lang#99030 (diagnostics: error messages when struct literals fail to parse)
- rust-lang#99155 (Keep unstable target features for asm feature checking)
- rust-lang#99199 (Refactor: remove an unnecessary
span_to_snippet
)
Failed merges:
r? @ghost
@rustbot
modify labels: rollup
This was referenced
Jul 14, 2022
flip1995 pushed a commit to flip1995/rust that referenced this pull request
…li-obk
Lower let-else in MIR
This MR will switch to lower let-else statements in MIR building instead.
To lower let-else in MIR, we build a mini-switch two branches. One branch leads to the matching case, and the other leads to the else
block. This arrangement will allow temporary lifetime analysis running as-is so that the temporaries are properly extended according to the same rule applied to regular let
statements.
Fix rust-lang#98672
flip1995 pushed a commit to flip1995/rust that referenced this pull request
Rollup of 5 pull requests
Successful merges:
- rust-lang#98574 (Lower let-else in MIR)
- rust-lang#99011 (
UnsafeCell
blocks niches inside its nested type from being available outside) - rust-lang#99030 (diagnostics: error messages when struct literals fail to parse)
- rust-lang#99155 (Keep unstable target features for asm feature checking)
- rust-lang#99199 (Refactor: remove an unnecessary
span_to_snippet
)
Failed merges:
r? @ghost
@rustbot
modify labels: rollup
JohnTitor pushed a commit to JohnTitor/rust that referenced this pull request
Rollup of 5 pull requests
Successful merges:
- rust-lang#98574 (Lower let-else in MIR)
- rust-lang#99011 (
UnsafeCell
blocks niches inside its nested type from being available outside) - rust-lang#99030 (diagnostics: error messages when struct literals fail to parse)
- rust-lang#99155 (Keep unstable target features for asm feature checking)
- rust-lang#99199 (Refactor: remove an unnecessary
span_to_snippet
)
Failed merges:
r? @ghost
@rustbot
modify labels: rollup
est31 mentioned this pull request
Dylan-DPC added a commit to Dylan-DPC/rust that referenced this pull request
est31 mentioned this pull request
GuillaumeGomez added a commit to GuillaumeGomez/rust that referenced this pull request
…plett
Stabilize let else
🎉 **Stabilizes the let else
feature, added by RFC 3137 🎉
Reference PR: rust-lang/reference#1156
closes rust-lang#87335 (let else
tracking issue)
FCP: rust-lang#93628 (comment)
Stabilization report
Summary
The feature allows refutable patterns in let
statements if the expression is
followed by a diverging else
:
fn get_count_item(s: &str) -> (u64, &str) {
let mut it = s.split(' ');
let (Some(count_str), Some(item)) = (it.next(), it.next()) else {
panic!("Can't segment count item pair: '{s}'");
};
let Ok(count) = u64::from_str(count_str) else {
panic!("Can't parse integer: '{count_str}'");
};
(count, item)
}
assert_eq!(get_count_item("3 chairs"), (3, "chairs"));
Differences from the RFC / Desugaring
Outside of desugaring I'm not aware of any differences between the implementation and the RFC. The chosen desugaring has been changed from the RFC's original. You can read a detailed discussion of the implementation history of it in @cormacrelf
's [summary](rust-lang#93628 (comment)) in this thread, as well as the [followup](rust-lang#93628 (comment)). Since that followup, further changes have happened to the desugaring, in rust-lang#98574, rust-lang#99518, rust-lang#99954. The later changes were mostly about the drop order: On match, temporaries drop in the same order as they would for a let
declaration. On mismatch, temporaries drop before the else
block.
Test cases
In chronological order as they were merged.
Added by df9a2e0 (rust-lang#87688):
ui/pattern/usefulness/top-level-alternation.rs
to ensure the unreachable pattern lint visits patterns insidelet else
.
Added by 5b95df4 (rust-lang#87688):
ui/let-else/let-else-bool-binop-init.rs
to ensure that no lazy boolean expressions (using&&
or||
) are allowed in the expression, as the RFC mandates.ui/let-else/let-else-brace-before-else.rs
to ensure that no}
directly preceding theelse
is allowed in the expression, as the RFC mandates.ui/let-else/let-else-check.rs
to ensure that#[allow(...)]
attributes added to the entirelet
statement apply for theelse
block.ui/let-else/let-else-irrefutable.rs
to ensure that theirrefutable_let_patterns
lint fires.ui/let-else/let-else-missing-semicolon.rs
to ensure the presence of semicolons at the end of thelet
statement.ui/let-else/let-else-non-diverging.rs
to ensure theelse
block diverges.ui/let-else/let-else-run-pass.rs
to ensure the feature works in some simple test case settings.ui/let-else/let-else-scope.rs
to ensure the bindings created by the outerlet
expression are not available in theelse
block of it.
Added by bf7c32a (rust-lang#89965):
ui/let-else/issue-89960.rs
as a regression test for the ICE-on-error bug rust-lang#89960 . Later in 102b912 this got removed in favour of more comprehensive tests.
Added by 8565419 (rust-lang#89974):
ui/let-else/let-else-if.rs
to test for the improved error message that points out thatlet else if
is not possible.
Added by 9b45713:
ui/let-else/let-else-allow-unused.rs
as a regression test for rust-lang#89807, to ensure that#[allow(...)]
attributes added to the entirelet
statement apply for bindings created by thelet else
pattern.
Added by 61bcd8d (rust-lang#89841):
ui/let-else/let-else-non-copy.rs
to ensure that a copy is performed out of non-copy wrapper types. This mirrorsif let
behaviour. The test case bases on rustc internal changes originally meant for rust-lang#89933 but then removed from the PR due to the error prior to the improvements of rust-lang#89841.ui/let-else/let-else-source-expr-nomove-pass.rs
to ensure that while there is a move of the binding in the successful case, theelse
case can still access the non-matching value. This mirrorsif let
behaviour.
Added by 102b912 (rust-lang#89841):
ui/let-else/let-else-ref-bindings.rs
andui/let-else/let-else-ref-bindings-pass.rs
to checkref
andref mut
keywords in the pattern work correctly and error when needed.
Added by 2715c5f (rust-lang#89841):
- Match ergonomic tests adapted from the
rfc2005
test suite.
Added by fec8a50 (rust-lang#89841):
ui/let-else/let-else-deref-coercion-annotated.rs
andui/let-else/let-else-deref-coercion.rs
to check deref coercions.
Added since this stabilization report was originally written (2022-02-09)
Added by 76ea566 (rust-lang#94211):
ui/let-else/let-else-destructuring.rs
to give a nice error message if an user tries to do an assignment with a (possibly refutable) pattern and anelse
block, like asked for in rust-lang#93995.
Added by e7730dc (rust-lang#94208):
ui/let-else/let-else-allow-in-expr.rs
to test whether#[allow(unused_variables)]
works in the expr, as well as its non presence, as well as putting it on the entirelet else
affects the expr, too. This was adding a missing test as pointed out by the stabilization report.- Expansion of
ui/let-else/let-else-allow-unused.rs
andui/let-else/let-else-check.rs
to ensure that non-presence of#[allow(unused)]
does issue the unused lint. This was adding a missing test case as pointed out by the stabilization report.
Added by 5bd7106 (rust-lang#94208):
ui/let-else/let-else-slicing-error.rs
, a regression test for rust-lang#92069, which got fixed without addition of a regression test. This resolves a missing test as pointed out by the stabilization report.
Added by 5374688 (rust-lang#98574):
src/test/ui/async-await/async-await-let-else.rs
to test the interaction of async/await withlet else
Added by 6c529de (rust-lang#98574):
src/test/ui/let-else/let-else-temporary-lifetime.rs
as a (partial) regression test for rust-lang#98672
Added by 9b56640 (rust-lang#99518):
src/test/ui/let-else/let-else-temp-borrowck.rs
as a regression test for rust-lang#93951- Extension of
src/test/ui/let-else/let-else-temporary-lifetime.rs
to include a partial regression test for rust-lang#98672 (especially regardingelse
drop order)
Added by baf9a7c (rust-lang#99518):
- Extension of
src/test/ui/let-else/let-else-temporary-lifetime.rs
to include a partial regression test for rust-lang#93951, similar tolet-else-temp-borrowck.rs
Added by 60be2de (rust-lang#99518):
- Extension of
src/test/ui/let-else/let-else-temporary-lifetime.rs
to include a program that can now be compiled thanks to borrow checker implications of rust-lang#99518
Added by 47a7a91 (rust-lang#100132):
src/test/ui/let-else/issue-100103.rs
, as a regression test for rust-lang#100103, to ensure that there is no ICE when doingErr(...)?
inside else blocks.
Added by e3c5bd6 (rust-lang#100443):
src/test/ui/let-else/let-else-then-diverge.rs
, to verify that there is no unreachable code error with the current desugaring.
Added by 9818526 (rust-lang#100443):
src/test/ui/let-else/issue-94176.rs
, to make sure that a correct span is emitted for a missing trailing expression error. Regression test for rust-lang#94176.
Added by e182d12 (rust-lang#100434):
- src/test/ui/unpretty/pretty-let-else.rs, as a regression test to ensure pretty printing works for
let else
(this bug surfaced in many different ways)
Added by e262856 (rust-lang#99954):
src/test/ui/let-else/let-else-temporary-lifetime.rs
extended to contain & borrows as well, as this was identified as an earlier issue with the desugaring: rust-lang#98672 (comment)
Added by 2d8460e (rust-lang#99291):
src/test/ui/let-else/let-else-drop-order.rs
a matrix based test for various drop order behaviour oflet else
. Especially, it verifies equality oflet
andlet else
drop orders, [resolving](rust-lang#93628 (comment)) a [stabilization blocker](rust-lang#93628 (comment)).
Added by 1b87ce0 (rust-lang#101410):
- Edit to
src/test/ui/let-else/let-else-temporary-lifetime.rs
to add the-Zvalidate-mir
flag, as a regression test for rust-lang#99228
Added by af591eb (rust-lang#101410):
src/test/ui/let-else/issue-99975.rs
as a regression test for the ICE rust-lang#99975.
Added by this PR:
ui/let-else/let-else.rs
, a simple run-pass check, similar toui/let-else/let-else-run-pass.rs
.
Things not currently tested
The→ test added by e7730dc#[allow(...)]
tests check whether allow works, but they don't check whether the non-presence of allow causes a lint to fire.There is no→ test added by e7730dc#[allow(...)]
test for the expression, as there are tests for the pattern and the else block.→ test added by e7730dclet-else-brace-before-else.rs
forbids thelet ... = {} else {}
pattern and there is a rustfix to obtainlet ... = ({}) else {}
. I'm not sure whether the.fixed
files are checked by the tooling that they compile. But if there is no such check, it would be neat to make sure thatlet ... = ({}) else {}
compiles.rust-lang#92069 got closed as fixed, but no regression test was added. Not sure it's worth to add one.→ test added by 5bd7106- ~~consistency between
let else
andif let
regarding lifetimes and drop order: rust-lang#93628 (comment) → test added by 2d8460e
Edit: they are all tested now.
Possible future work / Refutable destructuring assignments
RFC 2909 specifies destructuring assignment, allowing statements like FooBar { a, b, c } = foo();
.
As it was stabilized, destructuring assignment only allows irrefutable patterns, which before the advent of let else
were the only patterns that let
supported.
So the combination of let else
and destructuring assignments gives reason to think about extensions of the destructuring assignments feature that allow refutable patterns, discussed in rust-lang#93995.
A naive mapping of let else
to destructuring assignments in the form of Some(v) = foo() else { ... };
might not be the ideal way. let else
needs a diverging else
clause as it introduces new bindings, while assignments have a default behaviour to fall back to if the pattern does not match, in the form of not performing the assignment. Thus, there is no good case to require divergence, or even an else
clause at all, beyond the need for having some introducer syntax so that it is clear to readers that the assignment is not a given (enums and structs look similar). There are better candidates for introducer syntax however than an empty else {}
clause, like maybe
which could be added as a keyword on an edition boundary:
let mut v = 0;
maybe Some(v) = foo(&v);
maybe Some(v) = foo(&v) else { bar() };
Further design discussion is left to an RFC, or the linked issue.
Dylan-DPC added a commit to Dylan-DPC/rust that referenced this pull request
…plett
Stabilize let else
🎉 **Stabilizes the let else
feature, added by RFC 3137 🎉
Reference PR: rust-lang/reference#1156
closes rust-lang#87335 (let else
tracking issue)
FCP: rust-lang#93628 (comment)
Stabilization report
Summary
The feature allows refutable patterns in let
statements if the expression is
followed by a diverging else
:
fn get_count_item(s: &str) -> (u64, &str) {
let mut it = s.split(' ');
let (Some(count_str), Some(item)) = (it.next(), it.next()) else {
panic!("Can't segment count item pair: '{s}'");
};
let Ok(count) = u64::from_str(count_str) else {
panic!("Can't parse integer: '{count_str}'");
};
(count, item)
}
assert_eq!(get_count_item("3 chairs"), (3, "chairs"));
Differences from the RFC / Desugaring
Outside of desugaring I'm not aware of any differences between the implementation and the RFC. The chosen desugaring has been changed from the RFC's original. You can read a detailed discussion of the implementation history of it in @cormacrelf
's [summary](rust-lang#93628 (comment)) in this thread, as well as the [followup](rust-lang#93628 (comment)). Since that followup, further changes have happened to the desugaring, in rust-lang#98574, rust-lang#99518, rust-lang#99954. The later changes were mostly about the drop order: On match, temporaries drop in the same order as they would for a let
declaration. On mismatch, temporaries drop before the else
block.
Test cases
In chronological order as they were merged.
Added by df9a2e0 (rust-lang#87688):
ui/pattern/usefulness/top-level-alternation.rs
to ensure the unreachable pattern lint visits patterns insidelet else
.
Added by 5b95df4 (rust-lang#87688):
ui/let-else/let-else-bool-binop-init.rs
to ensure that no lazy boolean expressions (using&&
or||
) are allowed in the expression, as the RFC mandates.ui/let-else/let-else-brace-before-else.rs
to ensure that no}
directly preceding theelse
is allowed in the expression, as the RFC mandates.ui/let-else/let-else-check.rs
to ensure that#[allow(...)]
attributes added to the entirelet
statement apply for theelse
block.ui/let-else/let-else-irrefutable.rs
to ensure that theirrefutable_let_patterns
lint fires.ui/let-else/let-else-missing-semicolon.rs
to ensure the presence of semicolons at the end of thelet
statement.ui/let-else/let-else-non-diverging.rs
to ensure theelse
block diverges.ui/let-else/let-else-run-pass.rs
to ensure the feature works in some simple test case settings.ui/let-else/let-else-scope.rs
to ensure the bindings created by the outerlet
expression are not available in theelse
block of it.
Added by bf7c32a (rust-lang#89965):
ui/let-else/issue-89960.rs
as a regression test for the ICE-on-error bug rust-lang#89960 . Later in 102b912 this got removed in favour of more comprehensive tests.
Added by 8565419 (rust-lang#89974):
ui/let-else/let-else-if.rs
to test for the improved error message that points out thatlet else if
is not possible.
Added by 9b45713:
ui/let-else/let-else-allow-unused.rs
as a regression test for rust-lang#89807, to ensure that#[allow(...)]
attributes added to the entirelet
statement apply for bindings created by thelet else
pattern.
Added by 61bcd8d (rust-lang#89841):
ui/let-else/let-else-non-copy.rs
to ensure that a copy is performed out of non-copy wrapper types. This mirrorsif let
behaviour. The test case bases on rustc internal changes originally meant for rust-lang#89933 but then removed from the PR due to the error prior to the improvements of rust-lang#89841.ui/let-else/let-else-source-expr-nomove-pass.rs
to ensure that while there is a move of the binding in the successful case, theelse
case can still access the non-matching value. This mirrorsif let
behaviour.
Added by 102b912 (rust-lang#89841):
ui/let-else/let-else-ref-bindings.rs
andui/let-else/let-else-ref-bindings-pass.rs
to checkref
andref mut
keywords in the pattern work correctly and error when needed.
Added by 2715c5f (rust-lang#89841):
- Match ergonomic tests adapted from the
rfc2005
test suite.
Added by fec8a50 (rust-lang#89841):
ui/let-else/let-else-deref-coercion-annotated.rs
andui/let-else/let-else-deref-coercion.rs
to check deref coercions.
Added since this stabilization report was originally written (2022-02-09)
Added by 76ea566 (rust-lang#94211):
ui/let-else/let-else-destructuring.rs
to give a nice error message if an user tries to do an assignment with a (possibly refutable) pattern and anelse
block, like asked for in rust-lang#93995.
Added by e7730dc (rust-lang#94208):
ui/let-else/let-else-allow-in-expr.rs
to test whether#[allow(unused_variables)]
works in the expr, as well as its non presence, as well as putting it on the entirelet else
affects the expr, too. This was adding a missing test as pointed out by the stabilization report.- Expansion of
ui/let-else/let-else-allow-unused.rs
andui/let-else/let-else-check.rs
to ensure that non-presence of#[allow(unused)]
does issue the unused lint. This was adding a missing test case as pointed out by the stabilization report.
Added by 5bd7106 (rust-lang#94208):
ui/let-else/let-else-slicing-error.rs
, a regression test for rust-lang#92069, which got fixed without addition of a regression test. This resolves a missing test as pointed out by the stabilization report.
Added by 5374688 (rust-lang#98574):
src/test/ui/async-await/async-await-let-else.rs
to test the interaction of async/await withlet else
Added by 6c529de (rust-lang#98574):
src/test/ui/let-else/let-else-temporary-lifetime.rs
as a (partial) regression test for rust-lang#98672
Added by 9b56640 (rust-lang#99518):
src/test/ui/let-else/let-else-temp-borrowck.rs
as a regression test for rust-lang#93951- Extension of
src/test/ui/let-else/let-else-temporary-lifetime.rs
to include a partial regression test for rust-lang#98672 (especially regardingelse
drop order)
Added by baf9a7c (rust-lang#99518):
- Extension of
src/test/ui/let-else/let-else-temporary-lifetime.rs
to include a partial regression test for rust-lang#93951, similar tolet-else-temp-borrowck.rs
Added by 60be2de (rust-lang#99518):
- Extension of
src/test/ui/let-else/let-else-temporary-lifetime.rs
to include a program that can now be compiled thanks to borrow checker implications of rust-lang#99518
Added by 47a7a91 (rust-lang#100132):
src/test/ui/let-else/issue-100103.rs
, as a regression test for rust-lang#100103, to ensure that there is no ICE when doingErr(...)?
inside else blocks.
Added by e3c5bd6 (rust-lang#100443):
src/test/ui/let-else/let-else-then-diverge.rs
, to verify that there is no unreachable code error with the current desugaring.
Added by 9818526 (rust-lang#100443):
src/test/ui/let-else/issue-94176.rs
, to make sure that a correct span is emitted for a missing trailing expression error. Regression test for rust-lang#94176.
Added by e182d12 (rust-lang#100434):
- src/test/ui/unpretty/pretty-let-else.rs, as a regression test to ensure pretty printing works for
let else
(this bug surfaced in many different ways)
Added by e262856 (rust-lang#99954):
src/test/ui/let-else/let-else-temporary-lifetime.rs
extended to contain & borrows as well, as this was identified as an earlier issue with the desugaring: rust-lang#98672 (comment)
Added by 2d8460e (rust-lang#99291):
src/test/ui/let-else/let-else-drop-order.rs
a matrix based test for various drop order behaviour oflet else
. Especially, it verifies equality oflet
andlet else
drop orders, [resolving](rust-lang#93628 (comment)) a [stabilization blocker](rust-lang#93628 (comment)).
Added by 1b87ce0 (rust-lang#101410):
- Edit to
src/test/ui/let-else/let-else-temporary-lifetime.rs
to add the-Zvalidate-mir
flag, as a regression test for rust-lang#99228
Added by af591eb (rust-lang#101410):
src/test/ui/let-else/issue-99975.rs
as a regression test for the ICE rust-lang#99975.
Added by this PR:
ui/let-else/let-else.rs
, a simple run-pass check, similar toui/let-else/let-else-run-pass.rs
.
Things not currently tested
The→ test added by e7730dc#[allow(...)]
tests check whether allow works, but they don't check whether the non-presence of allow causes a lint to fire.There is no→ test added by e7730dc#[allow(...)]
test for the expression, as there are tests for the pattern and the else block.→ test added by e7730dclet-else-brace-before-else.rs
forbids thelet ... = {} else {}
pattern and there is a rustfix to obtainlet ... = ({}) else {}
. I'm not sure whether the.fixed
files are checked by the tooling that they compile. But if there is no such check, it would be neat to make sure thatlet ... = ({}) else {}
compiles.rust-lang#92069 got closed as fixed, but no regression test was added. Not sure it's worth to add one.→ test added by 5bd7106- ~~consistency between
let else
andif let
regarding lifetimes and drop order: rust-lang#93628 (comment) → test added by 2d8460e
Edit: they are all tested now.
Possible future work / Refutable destructuring assignments
RFC 2909 specifies destructuring assignment, allowing statements like FooBar { a, b, c } = foo();
.
As it was stabilized, destructuring assignment only allows irrefutable patterns, which before the advent of let else
were the only patterns that let
supported.
So the combination of let else
and destructuring assignments gives reason to think about extensions of the destructuring assignments feature that allow refutable patterns, discussed in rust-lang#93995.
A naive mapping of let else
to destructuring assignments in the form of Some(v) = foo() else { ... };
might not be the ideal way. let else
needs a diverging else
clause as it introduces new bindings, while assignments have a default behaviour to fall back to if the pattern does not match, in the form of not performing the assignment. Thus, there is no good case to require divergence, or even an else
clause at all, beyond the need for having some introducer syntax so that it is clear to readers that the assignment is not a given (enums and structs look similar). There are better candidates for introducer syntax however than an empty else {}
clause, like maybe
which could be added as a keyword on an edition boundary:
let mut v = 0;
maybe Some(v) = foo(&v);
maybe Some(v) = foo(&v) else { bar() };
Further design discussion is left to an RFC, or the linked issue.
matthiaskrgr added a commit to matthiaskrgr/rust that referenced this pull request
…low, r=oli-obk
Restore control flow on error in EUV
cc @Nilstrieb
Fix rust-lang#104649
Since rust-lang#98574 refactored a piece of scrutinee memory categorization out as a subroutine, there is a subtle change in handling match arms especially when the categorization process faults and bails. In the correct case, it is not supposed to continue to process the arms any more. This PR restores the original control flow in EUV.
I promise to add a compile-fail test to demonstrate that this indeed fixes the issue after coming back from a nap.