Skip to content

Commit 9a36f25

Browse files
authored
Unrolled build for rust-lang#123311
Rollup merge of rust-lang#123311 - Jules-Bertholet:andpat-everywhere, r=Nadrieril Match ergonomics: implement "`&`pat everywhere" Implements the eat-two-layers (feature gate `and_pat_everywhere`, all editions) ~and the eat-one-layer (feature gate `and_eat_one_layer_2024`, edition 2024 only, takes priority on that edition when both feature gates are active)~ (EDIT: will be done in later PR) semantics. cc rust-lang#123076 r? ``@Nadrieril`` ``@rustbot`` label A-patterns A-edition-2024
2 parents 5958f5e + e5376f3 commit 9a36f25

File tree

11 files changed

+281
-28
lines changed

11 files changed

+281
-28
lines changed

compiler/rustc_feature/src/unstable.rs

+2
Original file line numberDiff line numberDiff line change
@@ -567,6 +567,8 @@ declare_features! (
567567
(unstable, proc_macro_hygiene, "1.30.0", Some(54727)),
568568
/// Allows `&raw const $place_expr` and `&raw mut $place_expr` expressions.
569569
(unstable, raw_ref_op, "1.41.0", Some(64490)),
570+
/// Allows `&` and `&mut` patterns to consume match-ergonomics-inserted references.
571+
(incomplete, ref_pat_everywhere, "CURRENT_RUSTC_VERSION", Some(123076)),
570572
/// Allows using the `#[register_tool]` attribute.
571573
(unstable, register_tool, "1.41.0", Some(66079)),
572574
/// Allows the `#[repr(i128)]` attribute for enums.

compiler/rustc_hir_typeck/src/pat.rs

+64-27
Original file line numberDiff line numberDiff line change
@@ -130,7 +130,14 @@ enum AdjustMode {
130130
/// Peel off all immediate reference types.
131131
Peel,
132132
/// Reset binding mode to the initial mode.
133+
/// Used for destructuring assignment, where we don't want any match ergonomics.
133134
Reset,
135+
/// Produced by ref patterns.
136+
/// Reset the binding mode to the initial mode,
137+
/// and if the old biding mode was by-reference
138+
/// with mutability matching the pattern,
139+
/// mark the pattern as having consumed this reference.
140+
ResetAndConsumeRef(Mutability),
134141
/// Pass on the input binding mode and expected type.
135142
Pass,
136143
}
@@ -174,7 +181,8 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
174181
_ => None,
175182
};
176183
let adjust_mode = self.calc_adjust_mode(pat, path_res.map(|(res, ..)| res));
177-
let (expected, def_bm) = self.calc_default_binding_mode(pat, expected, def_bm, adjust_mode);
184+
let (expected, def_bm, ref_pattern_already_consumed) =
185+
self.calc_default_binding_mode(pat, expected, def_bm, adjust_mode);
178186
let pat_info = PatInfo {
179187
binding_mode: def_bm,
180188
top_info: ti,
@@ -211,7 +219,14 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
211219
}
212220
PatKind::Box(inner) => self.check_pat_box(pat.span, inner, expected, pat_info),
213221
PatKind::Deref(inner) => self.check_pat_deref(pat.span, inner, expected, pat_info),
214-
PatKind::Ref(inner, mutbl) => self.check_pat_ref(pat, inner, mutbl, expected, pat_info),
222+
PatKind::Ref(inner, mutbl) => self.check_pat_ref(
223+
pat,
224+
inner,
225+
mutbl,
226+
expected,
227+
pat_info,
228+
ref_pattern_already_consumed,
229+
),
215230
PatKind::Slice(before, slice, after) => {
216231
self.check_pat_slice(pat.span, before, slice, after, expected, pat_info)
217232
}
@@ -264,17 +279,27 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
264279

265280
/// Compute the new expected type and default binding mode from the old ones
266281
/// as well as the pattern form we are currently checking.
282+
///
283+
/// Last entry is only relevant for ref patterns (`&` and `&mut`);
284+
/// if `true`, then the ref pattern consumed a match ergonomics inserted reference
285+
/// and so does no need to match against a reference in the scrutinee type.
267286
fn calc_default_binding_mode(
268287
&self,
269288
pat: &'tcx Pat<'tcx>,
270289
expected: Ty<'tcx>,
271290
def_bm: BindingAnnotation,
272291
adjust_mode: AdjustMode,
273-
) -> (Ty<'tcx>, BindingAnnotation) {
292+
) -> (Ty<'tcx>, BindingAnnotation, bool) {
274293
match adjust_mode {
275-
AdjustMode::Pass => (expected, def_bm),
276-
AdjustMode::Reset => (expected, INITIAL_BM),
277-
AdjustMode::Peel => self.peel_off_references(pat, expected, def_bm),
294+
AdjustMode::Pass => (expected, def_bm, false),
295+
AdjustMode::Reset => (expected, INITIAL_BM, false),
296+
AdjustMode::ResetAndConsumeRef(mutbl) => {
297+
(expected, INITIAL_BM, def_bm.0 == ByRef::Yes(mutbl))
298+
}
299+
AdjustMode::Peel => {
300+
let peeled = self.peel_off_references(pat, expected, def_bm);
301+
(peeled.0, peeled.1, false)
302+
}
278303
}
279304
}
280305

@@ -329,7 +354,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
329354
// ```
330355
//
331356
// See issue #46688.
332-
PatKind::Ref(..) => AdjustMode::Reset,
357+
PatKind::Ref(_, mutbl) => AdjustMode::ResetAndConsumeRef(*mutbl),
333358
// A `_` pattern works with any expected type, so there's no need to do anything.
334359
PatKind::Wild
335360
// A malformed pattern doesn't have an expected type, so let's just accept any type.
@@ -840,8 +865,8 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
840865
&& let Some(mt) = self.shallow_resolve(expected).builtin_deref(true)
841866
&& let ty::Dynamic(..) = mt.ty.kind()
842867
{
843-
// This is "x = SomeTrait" being reduced from
844-
// "let &x = &SomeTrait" or "let box x = Box<SomeTrait>", an error.
868+
// This is "x = dyn SomeTrait" being reduced from
869+
// "let &x = &dyn SomeTrait" or "let box x = Box<dyn SomeTrait>", an error.
845870
let type_str = self.ty_to_string(expected);
846871
let mut err = struct_span_code_err!(
847872
self.dcx(),
@@ -2036,6 +2061,7 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
20362061
mutbl: Mutability,
20372062
expected: Ty<'tcx>,
20382063
pat_info: PatInfo<'tcx, '_>,
2064+
consumed_inherited_ref: bool,
20392065
) -> Ty<'tcx> {
20402066
let tcx = self.tcx;
20412067
let expected = self.shallow_resolve(expected);
@@ -2051,26 +2077,37 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
20512077
match *expected.kind() {
20522078
ty::Ref(_, r_ty, r_mutbl) if r_mutbl == mutbl => (expected, r_ty),
20532079
_ => {
2054-
let inner_ty = self.next_ty_var(TypeVariableOrigin {
2055-
kind: TypeVariableOriginKind::TypeInference,
2056-
span: inner.span,
2057-
});
2058-
let ref_ty = self.new_ref_ty(pat.span, mutbl, inner_ty);
2059-
debug!("check_pat_ref: demanding {:?} = {:?}", expected, ref_ty);
2060-
let err = self.demand_eqtype_pat_diag(
2061-
pat.span,
2062-
expected,
2063-
ref_ty,
2064-
pat_info.top_info,
2065-
);
2080+
if consumed_inherited_ref && self.tcx.features().ref_pat_everywhere {
2081+
// We already matched against a match-ergonmics inserted reference,
2082+
// so we don't need to match against a reference from the original type.
2083+
// Save this infor for use in lowering later
2084+
self.typeck_results
2085+
.borrow_mut()
2086+
.skipped_ref_pats_mut()
2087+
.insert(pat.hir_id);
2088+
(expected, expected)
2089+
} else {
2090+
let inner_ty = self.next_ty_var(TypeVariableOrigin {
2091+
kind: TypeVariableOriginKind::TypeInference,
2092+
span: inner.span,
2093+
});
2094+
let ref_ty = self.new_ref_ty(pat.span, mutbl, inner_ty);
2095+
debug!("check_pat_ref: demanding {:?} = {:?}", expected, ref_ty);
2096+
let err = self.demand_eqtype_pat_diag(
2097+
pat.span,
2098+
expected,
2099+
ref_ty,
2100+
pat_info.top_info,
2101+
);
20662102

2067-
// Look for a case like `fn foo(&foo: u32)` and suggest
2068-
// `fn foo(foo: &u32)`
2069-
if let Some(mut err) = err {
2070-
self.borrow_pat_suggestion(&mut err, pat);
2071-
err.emit();
2103+
// Look for a case like `fn foo(&foo: u32)` and suggest
2104+
// `fn foo(foo: &u32)`
2105+
if let Some(mut err) = err {
2106+
self.borrow_pat_suggestion(&mut err, pat);
2107+
err.emit();
2108+
}
2109+
(ref_ty, inner_ty)
20722110
}
2073-
(ref_ty, inner_ty)
20742111
}
20752112
}
20762113
}

compiler/rustc_hir_typeck/src/writeback.rs

+9
Original file line numberDiff line numberDiff line change
@@ -345,6 +345,7 @@ impl<'cx, 'tcx> Visitor<'tcx> for WritebackCx<'cx, 'tcx> {
345345
_ => {}
346346
};
347347

348+
self.visit_skipped_ref_pats(p.hir_id);
348349
self.visit_pat_adjustments(p.span, p.hir_id);
349350

350351
self.visit_node_id(p.span, p.hir_id);
@@ -674,6 +675,14 @@ impl<'cx, 'tcx> WritebackCx<'cx, 'tcx> {
674675
}
675676
}
676677

678+
#[instrument(skip(self), level = "debug")]
679+
fn visit_skipped_ref_pats(&mut self, hir_id: hir::HirId) {
680+
if self.fcx.typeck_results.borrow_mut().skipped_ref_pats_mut().remove(hir_id) {
681+
debug!("node is a skipped ref pat");
682+
self.typeck_results.skipped_ref_pats_mut().insert(hir_id);
683+
}
684+
}
685+
677686
fn visit_liberated_fn_sigs(&mut self) {
678687
let fcx_typeck_results = self.fcx.typeck_results.borrow();
679688
assert_eq!(fcx_typeck_results.hir_owner, self.typeck_results.hir_owner);

compiler/rustc_middle/src/ty/typeck_results.rs

+56
Original file line numberDiff line numberDiff line change
@@ -96,6 +96,10 @@ pub struct TypeckResults<'tcx> {
9696
/// <https://github.com/rust-lang/rfcs/blob/master/text/2005-match-ergonomics.md#definitions>
9797
pat_adjustments: ItemLocalMap<Vec<Ty<'tcx>>>,
9898

99+
/// Set of reference patterns that match against a match-ergonomics inserted reference
100+
/// (as opposed to against a reference in the scrutinee type).
101+
skipped_ref_pats: ItemLocalSet,
102+
99103
/// Records the reasons that we picked the kind of each closure;
100104
/// not all closures are present in the map.
101105
closure_kind_origins: ItemLocalMap<(Span, HirPlace<'tcx>)>,
@@ -228,6 +232,7 @@ impl<'tcx> TypeckResults<'tcx> {
228232
adjustments: Default::default(),
229233
pat_binding_modes: Default::default(),
230234
pat_adjustments: Default::default(),
235+
skipped_ref_pats: Default::default(),
231236
closure_kind_origins: Default::default(),
232237
liberated_fn_sigs: Default::default(),
233238
fru_field_types: Default::default(),
@@ -435,6 +440,14 @@ impl<'tcx> TypeckResults<'tcx> {
435440
LocalTableInContextMut { hir_owner: self.hir_owner, data: &mut self.pat_adjustments }
436441
}
437442

443+
pub fn skipped_ref_pats(&self) -> LocalSetInContext<'_> {
444+
LocalSetInContext { hir_owner: self.hir_owner, data: &self.skipped_ref_pats }
445+
}
446+
447+
pub fn skipped_ref_pats_mut(&mut self) -> LocalSetInContextMut<'_> {
448+
LocalSetInContextMut { hir_owner: self.hir_owner, data: &mut self.skipped_ref_pats }
449+
}
450+
438451
/// Does the pattern recursively contain a `ref mut` binding in it?
439452
///
440453
/// This is used to determined whether a `deref` pattern should emit a `Deref`
@@ -629,6 +642,49 @@ impl<'a, V> LocalTableInContextMut<'a, V> {
629642
}
630643
}
631644

645+
#[derive(Clone, Copy, Debug)]
646+
pub struct LocalSetInContext<'a> {
647+
hir_owner: OwnerId,
648+
data: &'a ItemLocalSet,
649+
}
650+
651+
impl<'a> LocalSetInContext<'a> {
652+
pub fn is_empty(&self) -> bool {
653+
self.data.is_empty()
654+
}
655+
656+
pub fn contains(&self, id: hir::HirId) -> bool {
657+
validate_hir_id_for_typeck_results(self.hir_owner, id);
658+
self.data.contains(&id.local_id)
659+
}
660+
}
661+
662+
#[derive(Debug)]
663+
pub struct LocalSetInContextMut<'a> {
664+
hir_owner: OwnerId,
665+
data: &'a mut ItemLocalSet,
666+
}
667+
668+
impl<'a> LocalSetInContextMut<'a> {
669+
pub fn is_empty(&self) -> bool {
670+
self.data.is_empty()
671+
}
672+
673+
pub fn contains(&self, id: hir::HirId) -> bool {
674+
validate_hir_id_for_typeck_results(self.hir_owner, id);
675+
self.data.contains(&id.local_id)
676+
}
677+
pub fn insert(&mut self, id: hir::HirId) -> bool {
678+
validate_hir_id_for_typeck_results(self.hir_owner, id);
679+
self.data.insert(id.local_id)
680+
}
681+
682+
pub fn remove(&mut self, id: hir::HirId) -> bool {
683+
validate_hir_id_for_typeck_results(self.hir_owner, id);
684+
self.data.remove(&id.local_id)
685+
}
686+
}
687+
632688
rustc_index::newtype_index! {
633689
#[derive(HashStable)]
634690
#[encodable]

compiler/rustc_mir_build/src/thir/pattern/mod.rs

+8-1
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,14 @@ impl<'a, 'tcx> PatCtxt<'a, 'tcx> {
6565
// we wrap the unadjusted pattern in `PatKind::Deref` repeatedly, consuming the
6666
// adjustments in *reverse order* (last-in-first-out, so that the last `Deref` inserted
6767
// gets the least-dereferenced type).
68-
let unadjusted_pat = self.lower_pattern_unadjusted(pat);
68+
let unadjusted_pat = match pat.kind {
69+
hir::PatKind::Ref(inner, _)
70+
if self.typeck_results.skipped_ref_pats().contains(pat.hir_id) =>
71+
{
72+
self.lower_pattern_unadjusted(inner)
73+
}
74+
_ => self.lower_pattern_unadjusted(pat),
75+
};
6976
self.typeck_results.pat_adjustments().get(pat.hir_id).unwrap_or(&vec![]).iter().rev().fold(
7077
unadjusted_pat,
7178
|pat: Box<_>, ref_ty| {

compiler/rustc_span/src/symbol.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1456,6 +1456,7 @@ symbols! {
14561456
receiver,
14571457
recursion_limit,
14581458
reexport_test_harness_main,
1459+
ref_pat_everywhere,
14591460
ref_unwind_safe_trait,
14601461
reference,
14611462
reflect,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
pub fn main() {
2+
if let Some(Some(&x)) = &Some(&Some(0)) {
3+
//~^ ERROR: mismatched types [E0308]
4+
let _: u32 = x;
5+
}
6+
if let Some(&Some(x)) = &Some(Some(0)) {
7+
//~^ ERROR: mismatched types [E0308]
8+
let _: u32 = x;
9+
}
10+
if let Some(Some(&mut x)) = &mut Some(&mut Some(0)) {
11+
//~^ ERROR: mismatched types [E0308]
12+
let _: u32 = x;
13+
}
14+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
error[E0308]: mismatched types
2+
--> $DIR/feature-gate-ref_pat_everywhere.rs:2:22
3+
|
4+
LL | if let Some(Some(&x)) = &Some(&Some(0)) {
5+
| ^^ --------------- this expression has type `&Option<&Option<{integer}>>`
6+
| |
7+
| expected integer, found `&_`
8+
|
9+
= note: expected type `{integer}`
10+
found reference `&_`
11+
help: consider removing `&` from the pattern
12+
|
13+
LL | if let Some(Some(x)) = &Some(&Some(0)) {
14+
| ~
15+
16+
error[E0308]: mismatched types
17+
--> $DIR/feature-gate-ref_pat_everywhere.rs:6:17
18+
|
19+
LL | if let Some(&Some(x)) = &Some(Some(0)) {
20+
| ^^^^^^^^ -------------- this expression has type `&Option<Option<{integer}>>`
21+
| |
22+
| expected `Option<{integer}>`, found `&_`
23+
|
24+
= note: expected enum `Option<{integer}>`
25+
found reference `&_`
26+
27+
error[E0308]: mismatched types
28+
--> $DIR/feature-gate-ref_pat_everywhere.rs:10:22
29+
|
30+
LL | if let Some(Some(&mut x)) = &mut Some(&mut Some(0)) {
31+
| ^^^^^^ ----------------------- this expression has type `&mut Option<&mut Option<{integer}>>`
32+
| |
33+
| expected integer, found `&mut _`
34+
|
35+
= note: expected type `{integer}`
36+
found mutable reference `&mut _`
37+
note: to declare a mutable binding use: `mut x`
38+
--> $DIR/feature-gate-ref_pat_everywhere.rs:10:22
39+
|
40+
LL | if let Some(Some(&mut x)) = &mut Some(&mut Some(0)) {
41+
| ^^^^^^
42+
help: consider removing `&mut` from the pattern
43+
|
44+
LL | if let Some(Some(x)) = &mut Some(&mut Some(0)) {
45+
| ~
46+
47+
error: aborting due to 3 previous errors
48+
49+
For more information about this error, try `rustc --explain E0308`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
#![allow(incomplete_features)]
2+
#![feature(ref_pat_everywhere)]
3+
pub fn main() {
4+
if let Some(&x) = Some(0) {
5+
//~^ ERROR: mismatched types [E0308]
6+
let _: u32 = x;
7+
}
8+
if let &Some(x) = &mut Some(0) {
9+
//~^ ERROR: mismatched types [E0308]
10+
let _: u32 = x;
11+
}
12+
if let Some(&x) = &mut Some(0) {
13+
//~^ ERROR: mismatched types [E0308]
14+
let _: u32 = x;
15+
}
16+
}

0 commit comments

Comments
 (0)