Skip to content

Commit d4cdd73

Browse files
Jules-Bertholetlcnr
authored andcommittedJun 10, 2024
Gate implicit mutable by-reference bindings behind mut ref
1 parent 2c89cb1 commit d4cdd73

11 files changed

+61
-65
lines changed
 

‎compiler/rustc_feature/src/removed.rs

+2
Original file line numberDiff line numberDiff line change
@@ -128,6 +128,8 @@ declare_features! (
128128
/// Allows the use of type alias impl trait in function return positions
129129
(removed, min_type_alias_impl_trait, "1.56.0", Some(63063),
130130
Some("removed in favor of full type_alias_impl_trait")),
131+
/// Make `mut` not reset the binding mode on edition >= 2024.
132+
(removed, mut_preserve_binding_mode_2024, "1.79.0", Some(123076), Some("superseded by `ref_pat_eat_one_layer_2024`")),
131133
(removed, needs_allocator, "1.4.0", Some(27389),
132134
Some("subsumed by `#![feature(allocator_internals)]`")),
133135
/// Allows use of unary negate on unsigned integers, e.g., -e for e: u8

‎compiler/rustc_feature/src/unstable.rs

-2
Original file line numberDiff line numberDiff line change
@@ -529,8 +529,6 @@ declare_features! (
529529
(unstable, more_qualified_paths, "1.54.0", Some(86935)),
530530
/// Allows the `#[must_not_suspend]` attribute.
531531
(unstable, must_not_suspend, "1.57.0", Some(83310)),
532-
/// Make `mut` not reset the binding mode on edition >= 2024.
533-
(incomplete, mut_preserve_binding_mode_2024, "1.79.0", Some(123076)),
534532
/// Allows `mut ref` and `mut ref mut` identifier patterns.
535533
(incomplete, mut_ref, "1.79.0", Some(123076)),
536534
/// Allows using `#[naked]` on functions.

‎compiler/rustc_hir_typeck/src/pat.rs

+22-12
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ use rustc_infer::infer;
1212
use rustc_middle::mir::interpret::ErrorHandled;
1313
use rustc_middle::ty::{self, Ty, TypeVisitableExt};
1414
use rustc_middle::{bug, span_bug};
15-
use rustc_session::lint::builtin::NON_EXHAUSTIVE_OMITTED_PATTERNS;
15+
use rustc_session::{lint::builtin::NON_EXHAUSTIVE_OMITTED_PATTERNS, parse::feature_err};
1616
use rustc_span::edit_distance::find_best_match_for_name;
1717
use rustc_span::hygiene::DesugaringKind;
1818
use rustc_span::source_map::Spanned;
@@ -669,17 +669,27 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
669669

670670
// Determine the binding mode...
671671
let bm = match user_bind_annot {
672-
// `mut` resets binding mode on edition <= 2021
673-
BindingMode(ByRef::No, Mutability::Mut)
674-
if !(pat.span.at_least_rust_2024()
675-
&& self.tcx.features().mut_preserve_binding_mode_2024)
676-
&& matches!(def_br, ByRef::Yes(_)) =>
677-
{
678-
self.typeck_results
679-
.borrow_mut()
680-
.rust_2024_migration_desugared_pats_mut()
681-
.insert(pat_info.top_info.hir_id);
682-
BindingMode(ByRef::No, Mutability::Mut)
672+
BindingMode(ByRef::No, Mutability::Mut) if matches!(def_br, ByRef::Yes(_)) => {
673+
if pat.span.at_least_rust_2024() && self.tcx.features().ref_pat_eat_one_layer_2024 {
674+
if !self.tcx.features().mut_ref {
675+
feature_err(
676+
&self.tcx.sess,
677+
sym::mut_ref,
678+
pat.span.until(ident.span),
679+
"binding cannot be both mutable and by-reference",
680+
)
681+
.emit();
682+
}
683+
684+
BindingMode(def_br, Mutability::Mut)
685+
} else {
686+
// `mut` resets binding mode on edition <= 2021
687+
self.typeck_results
688+
.borrow_mut()
689+
.rust_2024_migration_desugared_pats_mut()
690+
.insert(pat_info.top_info.hir_id);
691+
BindingMode(ByRef::No, Mutability::Mut)
692+
}
683693
}
684694
BindingMode(ByRef::No, mutbl) => BindingMode(def_br, mutbl),
685695
BindingMode(ByRef::Yes(_), _) => user_bind_annot,

‎tests/ui/match/ref_pat_eat_one_layer_2024/ref_pat_eat_one_layer_2024_fail.rs

+10
Original file line numberDiff line numberDiff line change
@@ -49,4 +49,14 @@ pub fn main() {
4949

5050
let &mut &mut &mut &mut _ = &mut &&&&mut &&&mut &mut 0;
5151
//~^ ERROR: mismatched types
52+
53+
struct Foo(u8);
54+
55+
let Foo(mut a) = &Foo(0);
56+
//~^ ERROR: binding cannot be both mutable and by-reference
57+
a = &42;
58+
59+
let Foo(mut a) = &mut Foo(0);
60+
//~^ ERROR: binding cannot be both mutable and by-reference
61+
a = &mut 42;
5262
}

‎tests/ui/match/ref_pat_eat_one_layer_2024/ref_pat_eat_one_layer_2024_fail.stderr

+23-2
Original file line numberDiff line numberDiff line change
@@ -152,6 +152,27 @@ LL | let &mut &mut &mut &mut _ = &mut &&&&mut &&&mut &mut 0;
152152
= note: expected reference `&&&&mut &&&mut &mut {integer}`
153153
found mutable reference `&mut _`
154154

155-
error: aborting due to 14 previous errors
155+
error[E0658]: binding cannot be both mutable and by-reference
156+
--> $DIR/ref_pat_eat_one_layer_2024_fail.rs:55:13
157+
|
158+
LL | let Foo(mut a) = &Foo(0);
159+
| ^^^^
160+
|
161+
= note: see issue #123076 <https://github.com/rust-lang/rust/issues/123076> for more information
162+
= help: add `#![feature(mut_ref)]` to the crate attributes to enable
163+
= note: this compiler was built on YYYY-MM-DD; consider upgrading it if it is out of date
164+
165+
error[E0658]: binding cannot be both mutable and by-reference
166+
--> $DIR/ref_pat_eat_one_layer_2024_fail.rs:59:13
167+
|
168+
LL | let Foo(mut a) = &mut Foo(0);
169+
| ^^^^
170+
|
171+
= note: see issue #123076 <https://github.com/rust-lang/rust/issues/123076> for more information
172+
= help: add `#![feature(mut_ref)]` to the crate attributes to enable
173+
= note: this compiler was built on YYYY-MM-DD; consider upgrading it if it is out of date
174+
175+
error: aborting due to 16 previous errors
156176

157-
For more information about this error, try `rustc --explain E0308`.
177+
Some errors have detailed explanations: E0308, E0658.
178+
For more information about an error, try `rustc --explain E0308`.

‎tests/ui/pattern/feature-gate-mut_preserve_binding_mode_2024.rs

-14
This file was deleted.

‎tests/ui/pattern/feature-gate-mut_preserve_binding_mode_2024.stderr

-31
This file was deleted.

‎tests/ui/pattern/match_ergonomics_2024.fixed

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
//@ run-rustfix
33
//@ rustfix-only-machine-applicable
44
//@ aux-build:match_ergonomics_2024_macros.rs
5-
#![feature(mut_preserve_binding_mode_2024, ref_pat_eat_one_layer_2024)]
5+
#![feature(mut_ref, ref_pat_eat_one_layer_2024)]
66
#![allow(incomplete_features, unused)]
77
#![deny(rust_2024_incompatible_pat)]
88

‎tests/ui/pattern/match_ergonomics_2024.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
//@ run-rustfix
33
//@ rustfix-only-machine-applicable
44
//@ aux-build:match_ergonomics_2024_macros.rs
5-
#![feature(mut_preserve_binding_mode_2024, ref_pat_eat_one_layer_2024)]
5+
#![feature(mut_ref, ref_pat_eat_one_layer_2024)]
66
#![allow(incomplete_features, unused)]
77
#![deny(rust_2024_incompatible_pat)]
88

‎tests/ui/pattern/mut_preserve_binding_mode_2021.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
//@ edition: 2021
22
//@ compile-flags: -Zunstable-options
3-
#![feature(mut_preserve_binding_mode_2024)]
3+
#![feature(ref_pat_eat_one_layer_2024)]
44
#![allow(incomplete_features)]
55

66
struct Foo(u8);

‎tests/ui/pattern/mut_preserve_binding_mode_2024.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
//@ run-pass
22
//@ edition: 2024
33
//@ compile-flags: -Zunstable-options
4-
#![feature(mut_preserve_binding_mode_2024)]
4+
#![feature(mut_ref, ref_pat_eat_one_layer_2024)]
55
#![allow(incomplete_features, unused)]
66

77
struct Foo(u8);

0 commit comments

Comments
 (0)