@@ -1716,24 +1716,28 @@ impl<'hir> LoweringContext<'_, 'hir> {
1716
1716
// `mut iter => { ... }`
1717
1717
let iter_arm = self . arm ( iter_pat, loop_expr) ;
1718
1718
1719
- let into_iter_expr = match loop_kind {
1719
+ let match_expr = match loop_kind {
1720
1720
ForLoopKind :: For => {
1721
1721
// `::std::iter::IntoIterator::into_iter(<head>)`
1722
- self . expr_call_lang_item_fn (
1722
+ let into_iter_expr = self . expr_call_lang_item_fn (
1723
1723
head_span,
1724
1724
hir:: LangItem :: IntoIterIntoIter ,
1725
1725
arena_vec ! [ self ; head] ,
1726
- )
1726
+ ) ;
1727
+
1728
+ self . arena . alloc ( self . expr_match (
1729
+ for_span,
1730
+ into_iter_expr,
1731
+ arena_vec ! [ self ; iter_arm] ,
1732
+ hir:: MatchSource :: ForLoopDesugar ,
1733
+ ) )
1727
1734
}
1728
- // ` unsafe { Pin::new_unchecked(&mut into_async_iter(<head>)) }`
1735
+ // `match into_async_iter(<head>) { ref mut iter => match unsafe { Pin::new_unchecked(iter) } { ... } }`
1729
1736
ForLoopKind :: ForAwait => {
1730
- // `::core::async_iter::IntoAsyncIterator::into_async_iter(<head>)`
1731
- let iter = self . expr_call_lang_item_fn (
1732
- head_span,
1733
- hir:: LangItem :: IntoAsyncIterIntoIter ,
1734
- arena_vec ! [ self ; head] ,
1735
- ) ;
1736
- let iter = self . expr_mut_addr_of ( head_span, iter) ;
1737
+ let iter_ident = iter;
1738
+ let ( async_iter_pat, async_iter_pat_id) =
1739
+ self . pat_ident_binding_mode ( head_span, iter_ident, hir:: BindingMode :: REF_MUT ) ;
1740
+ let iter = self . expr_ident_mut ( head_span, iter_ident, async_iter_pat_id) ;
1737
1741
// `Pin::new_unchecked(...)`
1738
1742
let iter = self . arena . alloc ( self . expr_call_lang_item_fn_mut (
1739
1743
head_span,
@@ -1742,17 +1746,29 @@ impl<'hir> LoweringContext<'_, 'hir> {
1742
1746
) ) ;
1743
1747
// `unsafe { ... }`
1744
1748
let iter = self . arena . alloc ( self . expr_unsafe ( iter) ) ;
1745
- iter
1749
+ let inner_match_expr = self . arena . alloc ( self . expr_match (
1750
+ for_span,
1751
+ iter,
1752
+ arena_vec ! [ self ; iter_arm] ,
1753
+ hir:: MatchSource :: ForLoopDesugar ,
1754
+ ) ) ;
1755
+
1756
+ // `::core::async_iter::IntoAsyncIterator::into_async_iter(<head>)`
1757
+ let iter = self . expr_call_lang_item_fn (
1758
+ head_span,
1759
+ hir:: LangItem :: IntoAsyncIterIntoIter ,
1760
+ arena_vec ! [ self ; head] ,
1761
+ ) ;
1762
+ let iter_arm = self . arm ( async_iter_pat, inner_match_expr) ;
1763
+ self . arena . alloc ( self . expr_match (
1764
+ for_span,
1765
+ iter,
1766
+ arena_vec ! [ self ; iter_arm] ,
1767
+ hir:: MatchSource :: ForLoopDesugar ,
1768
+ ) )
1746
1769
}
1747
1770
} ;
1748
1771
1749
- let match_expr = self . arena . alloc ( self . expr_match (
1750
- for_span,
1751
- into_iter_expr,
1752
- arena_vec ! [ self ; iter_arm] ,
1753
- hir:: MatchSource :: ForLoopDesugar ,
1754
- ) ) ;
1755
-
1756
1772
// This is effectively `{ let _result = ...; _result }`.
1757
1773
// The construct was introduced in #21984 and is necessary to make sure that
1758
1774
// temporaries in the `head` expression are dropped and do not leak to the
0 commit comments