1use std::collections::hash_map::Entry::{Occupied, Vacant};
2use std::{assert_matches, cmp};
34use rustc_abi::FieldIdx;
5use rustc_astas ast;
6use rustc_data_structures::fx::FxHashMap;
7use rustc_errors::codes::*;
8use rustc_errors::{
9Applicability, Diag, DiagCtxtHandle, Diagnostic, ErrorGuaranteed, Level, MultiSpan, pluralize,
10struct_span_code_err,
11};
12use rustc_hir::def::{CtorKind, DefKind, Res};
13use rustc_hir::def_id::DefId;
14use rustc_hir::pat_util::EnumerateAndAdjustIterator;
15use rustc_hir::{
16selfas hir, BindingMode, ByRef, ExprKind, HirId, LangItem, Mutability, Pat, PatExpr,
17PatExprKind, PatKind, expr_needs_parens,
18};
19use rustc_hir_analysis::autoderef::report_autoderef_recursion_limit_error;
20use rustc_infer::infer::RegionVariableOrigin;
21use rustc_middle::traits::PatternOriginExpr;
22use rustc_middle::ty::{self, Pinnedness, Ty, TypeVisitableExt, Unnormalized};
23use rustc_middle::{bug, span_bug};
24use rustc_session::errors::feature_err;
25use rustc_session::lint::builtin::NON_EXHAUSTIVE_OMITTED_PATTERNS;
26use rustc_span::edit_distance::find_best_match_for_name;
27use rustc_span::edition::Edition;
28use rustc_span::{BytePos, DUMMY_SP, Ident, Span, kw, sym};
29use rustc_trait_selection::infer::InferCtxtExt;
30use rustc_trait_selection::traits::{ObligationCause, ObligationCauseCode};
31use tracing::{debug, instrument, trace};
32use ty::VariantDef;
33use ty::adjustment::{PatAdjust, PatAdjustment};
3435use super::report_unexpected_variant_res;
36use crate::expectation::Expectation;
37use crate::gather_locals::DeclOrigin;
38use crate::{FnCtxt, errors};
3940const CANNOT_IMPLICITLY_DEREF_POINTER_TRAIT_OBJ: &str = "\
41This error indicates that a pointer to a trait type cannot be implicitly dereferenced by a \
42pattern. Every trait defines a type, but because the size of trait implementors isn't fixed, \
43this type has no compile-time size. Therefore, all accesses to trait types must be through \
44pointers. If you encounter this error you should try to avoid dereferencing the pointer.
4546You can read more about trait objects in the Trait Objects section of the Reference: \
47https://doc.rust-lang.org/reference/types.html#trait-objects";
4849fn is_number(text: &str) -> bool {
50text.chars().all(|c: char| c.is_ascii_digit())
51}
5253/// Information about the expected type at the top level of type checking a pattern.
54///
55/// **NOTE:** This is only for use by diagnostics. Do NOT use for type checking logic!
56#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for TopInfo<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for TopInfo<'tcx> {
#[inline]
fn clone(&self) -> TopInfo<'tcx> {
let _: ::core::clone::AssertParamIsClone<Ty<'tcx>>;
let _:
::core::clone::AssertParamIsClone<Option<&'tcx hir::Expr<'tcx>>>;
let _: ::core::clone::AssertParamIsClone<Option<Span>>;
let _: ::core::clone::AssertParamIsClone<HirId>;
*self
}
}Clone)]
57struct TopInfo<'tcx> {
58/// The `expected` type at the top level of type checking a pattern.
59expected: Ty<'tcx>,
60/// Was the origin of the `span` from a scrutinee expression?
61 ///
62 /// Otherwise there is no scrutinee and it could be e.g. from the type of a formal parameter.
63origin_expr: Option<&'tcx hir::Expr<'tcx>>,
64/// The span giving rise to the `expected` type, if one could be provided.
65 ///
66 /// If `origin_expr` is `true`, then this is the span of the scrutinee as in:
67 ///
68 /// - `match scrutinee { ... }`
69 /// - `let _ = scrutinee;`
70 ///
71 /// This is used to point to add context in type errors.
72 /// In the following example, `span` corresponds to the `a + b` expression:
73 ///
74 /// ```text
75 /// error[E0308]: mismatched types
76 /// --> src/main.rs:L:C
77 /// |
78 /// L | let temp: usize = match a + b {
79 /// | ----- this expression has type `usize`
80 /// L | Ok(num) => num,
81 /// | ^^^^^^^ expected `usize`, found enum `std::result::Result`
82 /// |
83 /// = note: expected type `usize`
84 /// found type `std::result::Result<_, _>`
85 /// ```
86span: Option<Span>,
87/// The [`HirId`] of the top-level pattern.
88hir_id: HirId,
89}
9091#[derive(#[automatically_derived]
impl<'tcx> ::core::marker::Copy for PatInfo<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::clone::Clone for PatInfo<'tcx> {
#[inline]
fn clone(&self) -> PatInfo<'tcx> {
let _: ::core::clone::AssertParamIsClone<ByRef>;
let _: ::core::clone::AssertParamIsClone<PinnednessCap>;
let _: ::core::clone::AssertParamIsClone<MutblCap>;
let _: ::core::clone::AssertParamIsClone<TopInfo<'tcx>>;
let _: ::core::clone::AssertParamIsClone<Option<DeclOrigin<'tcx>>>;
let _: ::core::clone::AssertParamIsClone<u32>;
*self
}
}Clone)]
92struct PatInfo<'tcx> {
93 binding_mode: ByRef,
94 max_pinnedness: PinnednessCap,
95 max_ref_mutbl: MutblCap,
96 top_info: TopInfo<'tcx>,
97 decl_origin: Option<DeclOrigin<'tcx>>,
9899/// The depth of current pattern
100current_depth: u32,
101}
102103impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
104fn pattern_cause(&self, ti: &TopInfo<'tcx>, cause_span: Span) -> ObligationCause<'tcx> {
105// If origin_expr exists, then expected represents the type of origin_expr.
106 // If span also exists, then span == origin_expr.span (although it doesn't need to exist).
107 // In that case, we can peel away references from both and treat them
108 // as the same.
109let origin_expr_info = ti.origin_expr.map(|mut cur_expr| {
110let mut count = 0;
111112// cur_ty may have more layers of references than cur_expr.
113 // We can only make suggestions about cur_expr, however, so we'll
114 // use that as our condition for stopping.
115while let ExprKind::AddrOf(.., inner) = &cur_expr.kind {
116 cur_expr = inner;
117 count += 1;
118 }
119120PatternOriginExpr {
121 peeled_span: cur_expr.span,
122 peeled_count: count,
123 peeled_prefix_suggestion_parentheses: expr_needs_parens(cur_expr),
124 }
125 });
126127let code = ObligationCauseCode::Pattern {
128 span: ti.span,
129 root_ty: ti.expected,
130 origin_expr: origin_expr_info,
131 };
132self.cause(cause_span, code)
133 }
134135fn demand_eqtype_pat_diag(
136&'a self,
137 cause_span: Span,
138 expected: Ty<'tcx>,
139 actual: Ty<'tcx>,
140 ti: &TopInfo<'tcx>,
141 ) -> Result<(), Diag<'a>> {
142self.demand_eqtype_with_origin(&self.pattern_cause(ti, cause_span), expected, actual)
143 .map_err(|mut diag| {
144if let Some(expr) = ti.origin_expr {
145self.suggest_fn_call(&mut diag, expr, expected, |output| {
146self.can_eq(self.param_env, output, actual)
147 });
148 }
149diag150 })
151 }
152153fn demand_eqtype_pat(
154&self,
155 cause_span: Span,
156 expected: Ty<'tcx>,
157 actual: Ty<'tcx>,
158 ti: &TopInfo<'tcx>,
159 ) -> Result<(), ErrorGuaranteed> {
160self.demand_eqtype_pat_diag(cause_span, expected, actual, ti).map_err(|err| err.emit())
161 }
162}
163164/// Mode for adjusting the expected type and binding mode.
165#[derive(#[automatically_derived]
impl ::core::clone::Clone for AdjustMode {
#[inline]
fn clone(&self) -> AdjustMode {
let _: ::core::clone::AssertParamIsClone<PeelKind>;
*self
}
}Clone, #[automatically_derived]
impl ::core::marker::Copy for AdjustMode { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for AdjustMode {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
AdjustMode::Peel { kind: __self_0 } =>
::core::fmt::Formatter::debug_struct_field1_finish(f, "Peel",
"kind", &__self_0),
AdjustMode::Pass => ::core::fmt::Formatter::write_str(f, "Pass"),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for AdjustMode {
#[inline]
fn eq(&self, other: &AdjustMode) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(AdjustMode::Peel { kind: __self_0 }, AdjustMode::Peel {
kind: __arg1_0 }) => __self_0 == __arg1_0,
_ => true,
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for AdjustMode {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<PeelKind>;
}
}Eq)]
166enum AdjustMode {
167/// Peel off all immediate reference types. If the `deref_patterns` feature is enabled, this
168 /// also peels smart pointer ADTs.
169Peel { kind: PeelKind },
170/// Pass on the input binding mode and expected type.
171Pass,
172}
173174/// Restrictions on what types to peel when adjusting the expected type and binding mode.
175#[derive(#[automatically_derived]
impl ::core::clone::Clone for PeelKind {
#[inline]
fn clone(&self) -> PeelKind {
let _: ::core::clone::AssertParamIsClone<Option<DefId>>;
let _: ::core::clone::AssertParamIsClone<usize>;
*self
}
}Clone, #[automatically_derived]
impl ::core::marker::Copy for PeelKind { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for PeelKind {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
PeelKind::ExplicitDerefPat =>
::core::fmt::Formatter::write_str(f, "ExplicitDerefPat"),
PeelKind::Implicit { until_adt: __self_0, pat_ref_layers: __self_1
} =>
::core::fmt::Formatter::debug_struct_field2_finish(f,
"Implicit", "until_adt", __self_0, "pat_ref_layers",
&__self_1),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for PeelKind {
#[inline]
fn eq(&self, other: &PeelKind) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(PeelKind::Implicit {
until_adt: __self_0, pat_ref_layers: __self_1 },
PeelKind::Implicit {
until_adt: __arg1_0, pat_ref_layers: __arg1_1 }) =>
__self_0 == __arg1_0 && __self_1 == __arg1_1,
_ => true,
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for PeelKind {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<Option<DefId>>;
let _: ::core::cmp::AssertParamIsEq<usize>;
}
}Eq)]
176enum PeelKind {
177/// Only peel reference types. This is used for explicit `deref!(_)` patterns, which dereference
178 /// any number of `&`/`&mut` references, plus a single smart pointer.
179ExplicitDerefPat,
180/// Implicitly peel references, and if `deref_patterns` is enabled, smart pointer ADTs.
181Implicit {
182/// The ADT the pattern is a constructor for, if applicable, so that we don't peel it. See
183 /// [`ResolvedPat`] for more information.
184until_adt: Option<DefId>,
185/// The number of references at the head of the pattern's type, so we can leave that many
186 /// untouched. This is `1` for string literals, and `0` for most patterns.
187pat_ref_layers: usize,
188 },
189}
190191impl AdjustMode {
192const fn peel_until_adt(opt_adt_def: Option<DefId>) -> AdjustMode {
193 AdjustMode::Peel { kind: PeelKind::Implicit { until_adt: opt_adt_def, pat_ref_layers: 0 } }
194 }
195const fn peel_all() -> AdjustMode {
196AdjustMode::peel_until_adt(None)
197 }
198}
199200/// `ref mut` bindings (explicit or match-ergonomics) are not allowed behind an `&` reference.
201/// Normally, the borrow checker enforces this, but for (currently experimental) match ergonomics,
202/// we track this when typing patterns for two purposes:
203///
204/// - For RFC 3627's Rule 3, when this would prevent us from binding with `ref mut`, we limit the
205/// default binding mode to be by shared `ref` when it would otherwise be `ref mut`.
206///
207/// - For RFC 3627's Rule 5, we allow `&` patterns to match against `&mut` references, treating them
208/// as if they were shared references. Since the scrutinee is mutable in this case, the borrow
209/// checker won't catch if we bind with `ref mut`, so we need to throw an error ourselves.
210#[derive(#[automatically_derived]
impl ::core::clone::Clone for MutblCap {
#[inline]
fn clone(&self) -> MutblCap {
let _: ::core::clone::AssertParamIsClone<Option<Span>>;
*self
}
}Clone, #[automatically_derived]
impl ::core::marker::Copy for MutblCap { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for MutblCap {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
MutblCap::Not => ::core::fmt::Formatter::write_str(f, "Not"),
MutblCap::WeaklyNot(__self_0) =>
::core::fmt::Formatter::debug_tuple_field1_finish(f,
"WeaklyNot", &__self_0),
MutblCap::Mut => ::core::fmt::Formatter::write_str(f, "Mut"),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for MutblCap {
#[inline]
fn eq(&self, other: &MutblCap) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(MutblCap::WeaklyNot(__self_0), MutblCap::WeaklyNot(__arg1_0))
=> __self_0 == __arg1_0,
_ => true,
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for MutblCap {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<Option<Span>>;
}
}Eq)]
211enum MutblCap {
212/// Mutability restricted to immutable.
213Not,
214215/// Mutability restricted to immutable, but only because of the pattern
216 /// (not the scrutinee type).
217 ///
218 /// The contained span, if present, points to an `&` pattern
219 /// that is the reason for the restriction,
220 /// and which will be reported in a diagnostic.
221WeaklyNot(Option<Span>),
222223/// No restriction on mutability
224Mut,
225}
226227impl MutblCap {
228#[must_use]
229fn cap_to_weakly_not(self, span: Option<Span>) -> Self {
230match self {
231 MutblCap::Not => MutblCap::Not,
232_ => MutblCap::WeaklyNot(span),
233 }
234 }
235236#[must_use]
237fn as_mutbl(self) -> Mutability {
238match self {
239 MutblCap::Not | MutblCap::WeaklyNot(_) => Mutability::Not,
240 MutblCap::Mut => Mutability::Mut,
241 }
242 }
243}
244245/// `ref` or `ref mut` bindings (not pinned, explicitly or match-ergonomics) are only allowed behind
246/// an `&pin` reference if the binding's type is `Unpin`.
247///
248/// Normally, the borrow checker enforces this (not implemented yet), but we track it here for better
249/// diagnostics.
250#[derive(#[automatically_derived]
impl ::core::clone::Clone for PinnednessCap {
#[inline]
fn clone(&self) -> PinnednessCap { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for PinnednessCap { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for PinnednessCap {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
PinnednessCap::Not => "Not",
PinnednessCap::Pinned => "Pinned",
})
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for PinnednessCap {
#[inline]
fn eq(&self, other: &PinnednessCap) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for PinnednessCap {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {}
}Eq)]
251enum PinnednessCap {
252/// No restriction on pinnedness.
253Not,
254/// Pinnedness restricted to pinned.
255Pinned,
256}
257258/// Variations on RFC 3627's Rule 4: when do reference patterns match against inherited references?
259///
260/// "Inherited reference" designates the `&`/`&mut` types that arise from using match ergonomics, i.e.
261/// from matching a reference type with a non-reference pattern. E.g. when `Some(x)` matches on
262/// `&mut Option<&T>`, `x` gets type `&mut &T` and the outer `&mut` is considered "inherited".
263#[derive(#[automatically_derived]
impl ::core::clone::Clone for InheritedRefMatchRule {
#[inline]
fn clone(&self) -> InheritedRefMatchRule {
let _: ::core::clone::AssertParamIsClone<bool>;
*self
}
}Clone, #[automatically_derived]
impl ::core::marker::Copy for InheritedRefMatchRule { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for InheritedRefMatchRule {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
InheritedRefMatchRule::EatOuter =>
::core::fmt::Formatter::write_str(f, "EatOuter"),
InheritedRefMatchRule::EatInner =>
::core::fmt::Formatter::write_str(f, "EatInner"),
InheritedRefMatchRule::EatBoth { consider_inherited_ref: __self_0
} =>
::core::fmt::Formatter::debug_struct_field1_finish(f,
"EatBoth", "consider_inherited_ref", &__self_0),
}
}
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for InheritedRefMatchRule {
#[inline]
fn eq(&self, other: &InheritedRefMatchRule) -> bool {
let __self_discr = ::core::intrinsics::discriminant_value(self);
let __arg1_discr = ::core::intrinsics::discriminant_value(other);
__self_discr == __arg1_discr &&
match (self, other) {
(InheritedRefMatchRule::EatBoth {
consider_inherited_ref: __self_0 },
InheritedRefMatchRule::EatBoth {
consider_inherited_ref: __arg1_0 }) => __self_0 == __arg1_0,
_ => true,
}
}
}PartialEq, #[automatically_derived]
impl ::core::cmp::Eq for InheritedRefMatchRule {
#[inline]
#[doc(hidden)]
#[coverage(off)]
fn assert_fields_are_eq(&self) {
let _: ::core::cmp::AssertParamIsEq<bool>;
}
}Eq)]
264enum InheritedRefMatchRule {
265/// Reference patterns consume only the inherited reference if possible, regardless of whether
266 /// the underlying type being matched against is a reference type. If there is no inherited
267 /// reference, a reference will be consumed from the underlying type.
268EatOuter,
269/// Reference patterns consume only a reference from the underlying type if possible. If the
270 /// underlying type is not a reference type, the inherited reference will be consumed.
271EatInner,
272/// When the underlying type is a reference type, reference patterns consume both layers of
273 /// reference, i.e. they both reset the binding mode and consume the reference type.
274EatBoth {
275/// If `true`, an inherited reference will be considered when determining whether a reference
276 /// pattern matches a given type:
277 /// - If the underlying type is not a reference, a reference pattern may eat the inherited reference;
278 /// - If the underlying type is a reference, a reference pattern matches if it can eat either one
279 /// of the underlying and inherited references. E.g. a `&mut` pattern is allowed if either the
280 /// underlying type is `&mut` or the inherited reference is `&mut`.
281 ///
282 /// If `false`, a reference pattern is only matched against the underlying type.
283 /// This is `false` for stable Rust and `true` for both the `ref_pat_eat_one_layer_2024` and
284 /// `ref_pat_eat_one_layer_2024_structural` feature gates.
285consider_inherited_ref: bool,
286 },
287}
288289/// When checking patterns containing paths, we need to know the path's resolution to determine
290/// whether to apply match ergonomics and implicitly dereference the scrutinee. For instance, when
291/// the `deref_patterns` feature is enabled and we're matching against a scrutinee of type
292/// `Cow<'a, Option<u8>>`, we insert an implicit dereference to allow the pattern `Some(_)` to type,
293/// but we must not dereference it when checking the pattern `Cow::Borrowed(_)`.
294///
295/// `ResolvedPat` contains the information from resolution needed to determine match ergonomics
296/// adjustments, and to finish checking the pattern once we know its adjusted type.
297#[derive(#[automatically_derived]
impl<'tcx> ::core::clone::Clone for ResolvedPat<'tcx> {
#[inline]
fn clone(&self) -> ResolvedPat<'tcx> {
let _: ::core::clone::AssertParamIsClone<Ty<'tcx>>;
let _: ::core::clone::AssertParamIsClone<ResolvedPatKind<'tcx>>;
*self
}
}Clone, #[automatically_derived]
impl<'tcx> ::core::marker::Copy for ResolvedPat<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for ResolvedPat<'tcx> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_struct_field2_finish(f, "ResolvedPat",
"ty", &self.ty, "kind", &&self.kind)
}
}Debug)]
298struct ResolvedPat<'tcx> {
299/// The type of the pattern, to be checked against the type of the scrutinee after peeling. This
300 /// is also used to avoid peeling the scrutinee's constructors (see the `Cow` example above).
301ty: Ty<'tcx>,
302 kind: ResolvedPatKind<'tcx>,
303}
304305#[derive(#[automatically_derived]
impl<'tcx> ::core::clone::Clone for ResolvedPatKind<'tcx> {
#[inline]
fn clone(&self) -> ResolvedPatKind<'tcx> {
let _: ::core::clone::AssertParamIsClone<Res>;
let _:
::core::clone::AssertParamIsClone<&'tcx [hir::PathSegment<'tcx>]>;
let _: ::core::clone::AssertParamIsClone<&'tcx VariantDef>;
let _: ::core::clone::AssertParamIsClone<&'tcx VariantDef>;
*self
}
}Clone, #[automatically_derived]
impl<'tcx> ::core::marker::Copy for ResolvedPatKind<'tcx> { }Copy, #[automatically_derived]
impl<'tcx> ::core::fmt::Debug for ResolvedPatKind<'tcx> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
match self {
ResolvedPatKind::Path {
res: __self_0, pat_res: __self_1, segments: __self_2 } =>
::core::fmt::Formatter::debug_struct_field3_finish(f, "Path",
"res", __self_0, "pat_res", __self_1, "segments",
&__self_2),
ResolvedPatKind::Struct { variant: __self_0 } =>
::core::fmt::Formatter::debug_struct_field1_finish(f,
"Struct", "variant", &__self_0),
ResolvedPatKind::TupleStruct { res: __self_0, variant: __self_1 }
=>
::core::fmt::Formatter::debug_struct_field2_finish(f,
"TupleStruct", "res", __self_0, "variant", &__self_1),
}
}
}Debug)]
306enum ResolvedPatKind<'tcx> {
307 Path { res: Res, pat_res: Res, segments: &'tcx [hir::PathSegment<'tcx>] },
308 Struct { variant: &'tcx VariantDef },
309 TupleStruct { res: Res, variant: &'tcx VariantDef },
310}
311312impl<'tcx> ResolvedPat<'tcx> {
313fn adjust_mode(&self) -> AdjustMode {
314if let ResolvedPatKind::Path { res, .. } = self.kind
315 && #[allow(non_exhaustive_omitted_patterns)] match res {
Res::Def(DefKind::Const { .. } | DefKind::AssocConst { .. }, _) => true,
_ => false,
}matches!(res, Res::Def(DefKind::Const { .. } | DefKind::AssocConst { .. }, _))316 {
317// These constants can be of a reference type, e.g. `const X: &u8 = &0;`.
318 // Peeling the reference types too early will cause type checking failures.
319 // Although it would be possible to *also* peel the types of the constants too.
320AdjustMode::Pass321 } else {
322// The remaining possible resolutions for path, struct, and tuple struct patterns are
323 // ADT constructors. As such, we may peel references freely, but we must not peel the
324 // ADT itself from the scrutinee if it's a smart pointer.
325AdjustMode::peel_until_adt(self.ty.ty_adt_def().map(|adt| adt.did()))
326 }
327 }
328}
329330impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
331/// Experimental pattern feature: after matching against a shared reference, do we limit the
332 /// default binding mode in subpatterns to be `ref` when it would otherwise be `ref mut`?
333 /// This corresponds to Rule 3 of RFC 3627.
334fn downgrade_mut_inside_shared(&self) -> bool {
335// NB: RFC 3627 proposes stabilizing Rule 3 in all editions. If we adopt the same behavior
336 // across all editions, this may be removed.
337self.tcx.features().ref_pat_eat_one_layer_2024_structural()
338 }
339340/// Experimental pattern feature: when do reference patterns match against inherited references?
341 /// This corresponds to variations on Rule 4 of RFC 3627.
342fn ref_pat_matches_inherited_ref(&self, edition: Edition) -> InheritedRefMatchRule {
343// NB: The particular rule used here is likely to differ across editions, so calls to this
344 // may need to become edition checks after match ergonomics stabilize.
345if edition.at_least_rust_2024() {
346if self.tcx.features().ref_pat_eat_one_layer_2024() {
347 InheritedRefMatchRule::EatOuter348 } else if self.tcx.features().ref_pat_eat_one_layer_2024_structural() {
349 InheritedRefMatchRule::EatInner350 } else {
351// Currently, matching against an inherited ref on edition 2024 is an error.
352 // Use `EatBoth` as a fallback to be similar to stable Rust.
353InheritedRefMatchRule::EatBoth { consider_inherited_ref: false }
354 }
355 } else {
356 InheritedRefMatchRule::EatBoth {
357 consider_inherited_ref: self.tcx.features().ref_pat_eat_one_layer_2024()
358 || self.tcx.features().ref_pat_eat_one_layer_2024_structural(),
359 }
360 }
361 }
362363/// Experimental pattern feature: do `&` patterns match against `&mut` references, treating them
364 /// as if they were shared references? This corresponds to Rule 5 of RFC 3627.
365fn ref_pat_matches_mut_ref(&self) -> bool {
366// NB: RFC 3627 proposes stabilizing Rule 5 in all editions. If we adopt the same behavior
367 // across all editions, this may be removed.
368self.tcx.features().ref_pat_eat_one_layer_2024()
369 || self.tcx.features().ref_pat_eat_one_layer_2024_structural()
370 }
371372/// Type check the given top level pattern against the `expected` type.
373 ///
374 /// If a `Some(span)` is provided and `origin_expr` holds,
375 /// then the `span` represents the scrutinee's span.
376 /// The scrutinee is found in e.g. `match scrutinee { ... }` and `let pat = scrutinee;`.
377 ///
378 /// Otherwise, `Some(span)` represents the span of a type expression
379 /// which originated the `expected` type.
380pub(crate) fn check_pat_top(
381&self,
382 pat: &'tcx Pat<'tcx>,
383 expected: Ty<'tcx>,
384 span: Option<Span>,
385 origin_expr: Option<&'tcx hir::Expr<'tcx>>,
386 decl_origin: Option<DeclOrigin<'tcx>>,
387 ) {
388let top_info = TopInfo { expected, origin_expr, span, hir_id: pat.hir_id };
389let pat_info = PatInfo {
390 binding_mode: ByRef::No,
391 max_pinnedness: PinnednessCap::Not,
392 max_ref_mutbl: MutblCap::Mut,
393top_info,
394decl_origin,
395 current_depth: 0,
396 };
397self.check_pat(pat, expected, pat_info);
398 }
399400/// Type check the given `pat` against the `expected` type
401 /// with the provided `binding_mode` (default binding mode).
402 ///
403 /// Outside of this module, `check_pat_top` should always be used.
404 /// Conversely, inside this module, `check_pat_top` should never be used.
405#[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("check_pat",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(405u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["pat", "expected"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&pat)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&expected)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: () = loop {};
return __tracing_attr_fake_return;
}
{
let opt_path_res =
match pat.kind {
PatKind::Expr(PatExpr {
kind: PatExprKind::Path(qpath), hir_id, span }) => {
Some(self.resolve_pat_path(*hir_id, *span, qpath))
}
PatKind::Struct(ref qpath, ..) =>
Some(self.resolve_pat_struct(pat, qpath)),
PatKind::TupleStruct(ref qpath, ..) =>
Some(self.resolve_pat_tuple_struct(pat, qpath)),
_ => None,
};
let adjust_mode = self.calc_adjust_mode(pat, opt_path_res);
let ty =
self.check_pat_inner(pat, opt_path_res, adjust_mode, expected,
pat_info);
self.write_ty(pat.hir_id, ty);
if let Some(derefed_tys) =
self.typeck_results.borrow().pat_adjustments().get(pat.hir_id)
&&
derefed_tys.iter().any(|adjust|
adjust.kind == PatAdjust::OverloadedDeref) {
self.register_deref_mut_bounds_if_needed(pat.span, pat,
derefed_tys.iter().filter_map(|adjust|
match adjust.kind {
PatAdjust::OverloadedDeref => Some(adjust.source),
PatAdjust::BuiltinDeref | PatAdjust::PinDeref => None,
}));
}
}
}
}#[instrument(level = "debug", skip(self, pat_info))]406fn check_pat(&self, pat: &'tcx Pat<'tcx>, expected: Ty<'tcx>, pat_info: PatInfo<'tcx>) {
407// For patterns containing paths, we need the path's resolution to determine whether to
408 // implicitly dereference the scrutinee before matching.
409let opt_path_res = match pat.kind {
410 PatKind::Expr(PatExpr { kind: PatExprKind::Path(qpath), hir_id, span }) => {
411Some(self.resolve_pat_path(*hir_id, *span, qpath))
412 }
413 PatKind::Struct(ref qpath, ..) => Some(self.resolve_pat_struct(pat, qpath)),
414 PatKind::TupleStruct(ref qpath, ..) => Some(self.resolve_pat_tuple_struct(pat, qpath)),
415_ => None,
416 };
417let adjust_mode = self.calc_adjust_mode(pat, opt_path_res);
418let ty = self.check_pat_inner(pat, opt_path_res, adjust_mode, expected, pat_info);
419self.write_ty(pat.hir_id, ty);
420421// If we implicitly inserted overloaded dereferences before matching check the pattern to
422 // see if the dereferenced types need `DerefMut` bounds.
423if let Some(derefed_tys) = self.typeck_results.borrow().pat_adjustments().get(pat.hir_id)
424 && derefed_tys.iter().any(|adjust| adjust.kind == PatAdjust::OverloadedDeref)
425 {
426self.register_deref_mut_bounds_if_needed(
427 pat.span,
428 pat,
429 derefed_tys.iter().filter_map(|adjust| match adjust.kind {
430 PatAdjust::OverloadedDeref => Some(adjust.source),
431 PatAdjust::BuiltinDeref | PatAdjust::PinDeref => None,
432 }),
433 );
434 }
435436// (note_1): In most of the cases where (note_1) is referenced
437 // (literals and constants being the exception), we relate types
438 // using strict equality, even though subtyping would be sufficient.
439 // There are a few reasons for this, some of which are fairly subtle
440 // and which cost me (nmatsakis) an hour or two debugging to remember,
441 // so I thought I'd write them down this time.
442 //
443 // 1. There is no loss of expressiveness here, though it does
444 // cause some inconvenience. What we are saying is that the type
445 // of `x` becomes *exactly* what is expected. This can cause unnecessary
446 // errors in some cases, such as this one:
447 //
448 // ```
449 // fn foo<'x>(x: &'x i32) {
450 // let a = 1;
451 // let mut z = x;
452 // z = &a;
453 // }
454 // ```
455 //
456 // The reason we might get an error is that `z` might be
457 // assigned a type like `&'x i32`, and then we would have
458 // a problem when we try to assign `&a` to `z`, because
459 // the lifetime of `&a` (i.e., the enclosing block) is
460 // shorter than `'x`.
461 //
462 // HOWEVER, this code works fine. The reason is that the
463 // expected type here is whatever type the user wrote, not
464 // the initializer's type. In this case the user wrote
465 // nothing, so we are going to create a type variable `Z`.
466 // Then we will assign the type of the initializer (`&'x i32`)
467 // as a subtype of `Z`: `&'x i32 <: Z`. And hence we
468 // will instantiate `Z` as a type `&'0 i32` where `'0` is
469 // a fresh region variable, with the constraint that `'x : '0`.
470 // So basically we're all set.
471 //
472 // Note that there are two tests to check that this remains true
473 // (`regions-reassign-{match,let}-bound-pointer.rs`).
474 //
475 // 2. An outdated issue related to the old HIR borrowck. See the test
476 // `regions-relate-bound-regions-on-closures-to-inference-variables.rs`,
477}
478479// Helper to avoid resolving the same path pattern several times.
480fn check_pat_inner(
481&self,
482 pat: &'tcx Pat<'tcx>,
483 opt_path_res: Option<Result<ResolvedPat<'tcx>, ErrorGuaranteed>>,
484 adjust_mode: AdjustMode,
485 expected: Ty<'tcx>,
486 pat_info: PatInfo<'tcx>,
487 ) -> Ty<'tcx> {
488#[cfg(debug_assertions)]
489if #[allow(non_exhaustive_omitted_patterns)] match pat_info.binding_mode {
ByRef::Yes(_, Mutability::Mut) => true,
_ => false,
}matches!(pat_info.binding_mode, ByRef::Yes(_, Mutability::Mut))490 && pat_info.max_ref_mutbl != MutblCap::Mut491 && self.downgrade_mut_inside_shared()
492 {
493::rustc_middle::util::bug::span_bug_fmt(pat.span,
format_args!("Pattern mutability cap violated!"));span_bug!(pat.span, "Pattern mutability cap violated!");
494 }
495496// Resolve type if needed.
497let expected = if let AdjustMode::Peel { .. } = adjust_mode498 && pat.default_binding_modes
499 {
500self.resolve_vars_with_obligations(expected)
501 } else {
502expected503 };
504let old_pat_info = pat_info;
505let pat_info = PatInfo { current_depth: old_pat_info.current_depth + 1, ..old_pat_info };
506507match pat.kind {
508// Peel off a `&` or `&mut`from the scrutinee type. See the examples in
509 // `tests/ui/rfcs/rfc-2005-default-binding-mode`.
510_ if let AdjustMode::Peel { kind: peel_kind } = adjust_mode511 && pat.default_binding_modes
512 && let &ty::Ref(_, inner_ty, inner_mutability) = expected.kind()
513 && self.should_peel_ref(peel_kind, expected) =>
514 {
515{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:515",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(515u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("inspecting {0:?}",
expected) as &dyn Value))])
});
} else { ; }
};debug!("inspecting {:?}", expected);
516517{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:517",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(517u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("current discriminant is Ref, inserting implicit deref")
as &dyn Value))])
});
} else { ; }
};debug!("current discriminant is Ref, inserting implicit deref");
518// Preserve the reference type. We'll need it later during THIR lowering.
519self.typeck_results
520 .borrow_mut()
521 .pat_adjustments_mut()
522 .entry(pat.hir_id)
523 .or_default()
524 .push(PatAdjustment { kind: PatAdjust::BuiltinDeref, source: expected });
525526// Use the old pat info to keep `current_depth` to its old value.
527let new_pat_info =
528self.adjust_pat_info(Pinnedness::Not, inner_mutability, old_pat_info);
529530// Recurse with the new expected type.
531self.check_pat_inner(pat, opt_path_res, adjust_mode, inner_ty, new_pat_info)
532 }
533// If `pin_ergonomics` is enabled, peel the `&pin` from the pinned reference type. See the
534 // examples in `tests/ui/async-await/pin-ergonomics/`.
535_ if self.tcx.features().pin_ergonomics()
536 && let AdjustMode::Peel { kind: peel_kind } = adjust_mode537 && pat.default_binding_modes
538 && self.should_peel_smart_pointer(peel_kind, expected)
539 && let Some(pinned_ty) = expected.pinned_ty()
540// Currently, only pinned reference is specially handled, leaving other
541 // pinned types (e.g. `Pin<Box<T>>` to deref patterns) handled as a
542 // deref pattern.
543&& let &ty::Ref(_, inner_ty, inner_mutability) = pinned_ty.kind() =>
544 {
545{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:545",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(545u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("scrutinee ty {0:?} is a pinned reference, inserting pin deref",
expected) as &dyn Value))])
});
} else { ; }
};debug!("scrutinee ty {expected:?} is a pinned reference, inserting pin deref");
546547// if the inner_ty is an ADT, make sure that it can be structurally pinned
548 // (i.e., it is `#[pin_v2]`).
549if let Some(adt) = inner_ty.ty_adt_def()
550 && !adt.is_pin_project()
551 && !adt.is_pin()
552 {
553let def_span: Option<Span> = self.tcx.hir_span_if_local(adt.did());
554let sugg_span = def_span.map(|span| span.shrink_to_lo());
555self.dcx().emit_err(crate::errors::ProjectOnNonPinProjectType {
556 span: pat.span,
557def_span,
558sugg_span,
559 });
560 }
561562// Use the old pat info to keep `current_depth` to its old value.
563let new_pat_info =
564self.adjust_pat_info(Pinnedness::Pinned, inner_mutability, old_pat_info);
565566self.check_deref_pattern(
567pat,
568opt_path_res,
569adjust_mode,
570expected,
571inner_ty,
572 PatAdjust::PinDeref,
573new_pat_info,
574 )
575 }
576// If `deref_patterns` is enabled, peel a smart pointer from the scrutinee type. See the
577 // examples in `tests/ui/pattern/deref_patterns/`.
578_ if self.tcx.features().deref_patterns()
579 && let AdjustMode::Peel { kind: peel_kind } = adjust_mode580 && pat.default_binding_modes
581 && self.should_peel_smart_pointer(peel_kind, expected) =>
582 {
583{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:583",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(583u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("scrutinee ty {0:?} is a smart pointer, inserting pin deref",
expected) as &dyn Value))])
});
} else { ; }
};debug!("scrutinee ty {expected:?} is a smart pointer, inserting pin deref");
584585// The scrutinee is a smart pointer; implicitly dereference it. This adds a
586 // requirement that `expected: DerefPure`.
587let inner_ty = self.deref_pat_target(pat.span, expected);
588// Once we've checked `pat`, we'll add a `DerefMut` bound if it contains any
589 // `ref mut` bindings. See `Self::register_deref_mut_bounds_if_needed`.
590591self.check_deref_pattern(
592pat,
593opt_path_res,
594adjust_mode,
595expected,
596inner_ty,
597 PatAdjust::OverloadedDeref,
598old_pat_info,
599 )
600 }
601 PatKind::Missing | PatKind::Wild | PatKind::Err(_) => expected,
602// We allow any type here; we ensure that the type is uninhabited during match checking.
603PatKind::Never => expected,
604 PatKind::Expr(PatExpr { kind: PatExprKind::Path(_), hir_id, .. }) => {
605let ty = match opt_path_res.unwrap() {
606Ok(ref pr) => {
607self.check_pat_path(pat.hir_id, pat.span, pr, expected, &pat_info.top_info)
608 }
609Err(guar) => Ty::new_error(self.tcx, guar),
610 };
611self.write_ty(*hir_id, ty);
612ty613 }
614 PatKind::Expr(expr @ PatExpr { kind: PatExprKind::Lit { lit, .. }, .. }) => {
615self.check_pat_lit(pat.span, expr, &lit.node, expected, &pat_info.top_info)
616 }
617 PatKind::Range(lhs, rhs, _) => {
618self.check_pat_range(pat.span, lhs, rhs, expected, &pat_info.top_info)
619 }
620 PatKind::Binding(ba, var_id, ident, sub) => {
621self.check_pat_ident(pat, ba, var_id, ident, sub, expected, pat_info)
622 }
623 PatKind::TupleStruct(ref qpath, subpats, ddpos) => match opt_path_res.unwrap() {
624Ok(ResolvedPat { ty, kind: ResolvedPatKind::TupleStruct { res, variant } }) => self625 .check_pat_tuple_struct(
626pat, qpath, subpats, ddpos, res, ty, variant, expected, pat_info,
627 ),
628Err(guar) => {
629let ty_err = Ty::new_error(self.tcx, guar);
630for subpat in subpats {
631self.check_pat(subpat, ty_err, pat_info);
632 }
633ty_err634 }
635Ok(pr) => ::rustc_middle::util::bug::span_bug_fmt(pat.span,
format_args!("tuple struct pattern resolved to {0:?}", pr))span_bug!(pat.span, "tuple struct pattern resolved to {pr:?}"),
636 },
637 PatKind::Struct(_, fields, has_rest_pat) => match opt_path_res.unwrap() {
638Ok(ResolvedPat { ty, kind: ResolvedPatKind::Struct { variant } }) => self639 .check_pat_struct(
640pat,
641fields,
642has_rest_pat.is_some(),
643ty,
644variant,
645expected,
646pat_info,
647 ),
648Err(guar) => {
649let ty_err = Ty::new_error(self.tcx, guar);
650for field in fields {
651self.check_pat(field.pat, ty_err, pat_info);
652 }
653ty_err654 }
655Ok(pr) => ::rustc_middle::util::bug::span_bug_fmt(pat.span,
format_args!("struct pattern resolved to {0:?}", pr))span_bug!(pat.span, "struct pattern resolved to {pr:?}"),
656 },
657 PatKind::Guard(pat, cond) => {
658self.check_pat(pat, expected, pat_info);
659self.check_expr_has_type_or_error(cond, self.tcx.types.bool, |_| {});
660expected661 }
662 PatKind::Or(pats) => {
663for pat in pats {
664self.check_pat(pat, expected, pat_info);
665 }
666expected667 }
668 PatKind::Tuple(elements, ddpos) => {
669self.check_pat_tuple(pat.span, elements, ddpos, expected, pat_info)
670 }
671 PatKind::Box(inner) => self.check_pat_box(pat.span, inner, expected, pat_info),
672 PatKind::Deref(inner) => self.check_pat_deref(pat.span, inner, expected, pat_info),
673 PatKind::Ref(inner, pinned, mutbl) => {
674self.check_pat_ref(pat, inner, pinned, mutbl, expected, pat_info)
675 }
676 PatKind::Slice(before, slice, after) => {
677self.check_pat_slice(pat.span, before, slice, after, expected, pat_info)
678 }
679 }
680 }
681682fn adjust_pat_info(
683&self,
684 inner_pinnedness: Pinnedness,
685 inner_mutability: Mutability,
686 pat_info: PatInfo<'tcx>,
687 ) -> PatInfo<'tcx> {
688let mut binding_mode = match pat_info.binding_mode {
689// If default binding mode is by value, make it `ref`, `ref mut`, `ref pin const`
690 // or `ref pin mut` (depending on whether we observe `&`, `&mut`, `&pin const` or
691 // `&pin mut`).
692ByRef::No => ByRef::Yes(inner_pinnedness, inner_mutability),
693 ByRef::Yes(pinnedness, mutability) => {
694let pinnedness = match pinnedness {
695// When `ref`, stay a `ref` (on `&`) or downgrade to `ref pin` (on `&pin`).
696Pinnedness::Not => inner_pinnedness,
697// When `ref pin`, stay a `ref pin`.
698 // This is because we cannot get an `&mut T` from `&mut &pin mut T` unless `T: Unpin`.
699 // Note that `&T` and `&mut T` are `Unpin`, which implies
700 // `& &pin const T` <-> `&pin const &T` and `&mut &pin mut T` <-> `&pin mut &mut T`
701 // (i.e. mutually coercible).
702Pinnedness::Pinned => Pinnedness::Pinned,
703 };
704705let mutability = match mutability {
706// When `ref mut`, stay a `ref mut` (on `&mut`) or downgrade to `ref` (on `&`).
707Mutability::Mut => inner_mutability,
708// Once a `ref`, always a `ref`.
709 // This is because a `& &mut` cannot mutate the underlying value.
710Mutability::Not => Mutability::Not,
711 };
712 ByRef::Yes(pinnedness, mutability)
713 }
714 };
715716let PatInfo { mut max_ref_mutbl, mut max_pinnedness, .. } = pat_info;
717if self.downgrade_mut_inside_shared() {
718binding_mode = binding_mode.cap_ref_mutability(max_ref_mutbl.as_mutbl());
719 }
720match binding_mode {
721 ByRef::Yes(_, Mutability::Not) => max_ref_mutbl = MutblCap::Not,
722 ByRef::Yes(Pinnedness::Pinned, _) => max_pinnedness = PinnednessCap::Pinned,
723_ => {}
724 }
725{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:725",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(725u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("default binding mode is now {0:?}",
binding_mode) as &dyn Value))])
});
} else { ; }
};debug!("default binding mode is now {:?}", binding_mode);
726PatInfo { binding_mode, max_pinnedness, max_ref_mutbl, ..pat_info }
727 }
728729fn check_deref_pattern(
730&self,
731 pat: &'tcx Pat<'tcx>,
732 opt_path_res: Option<Result<ResolvedPat<'tcx>, ErrorGuaranteed>>,
733 adjust_mode: AdjustMode,
734 expected: Ty<'tcx>,
735mut inner_ty: Ty<'tcx>,
736 pat_adjust_kind: PatAdjust,
737 pat_info: PatInfo<'tcx>,
738 ) -> Ty<'tcx> {
739if true {
if !!#[allow(non_exhaustive_omitted_patterns)] match pat_adjust_kind {
PatAdjust::BuiltinDeref => true,
_ => false,
} {
{
::core::panicking::panic_fmt(format_args!("unexpected deref pattern for builtin reference type {0:?}",
expected));
}
};
};debug_assert!(
740 !matches!(pat_adjust_kind, PatAdjust::BuiltinDeref),
741"unexpected deref pattern for builtin reference type {expected:?}",
742 );
743744let mut typeck_results = self.typeck_results.borrow_mut();
745let mut pat_adjustments_table = typeck_results.pat_adjustments_mut();
746let pat_adjustments = pat_adjustments_table.entry(pat.hir_id).or_default();
747// We may reach the recursion limit if a user matches on a type `T` satisfying
748 // `T: Deref<Target = T>`; error gracefully in this case.
749 // FIXME(deref_patterns): If `deref_patterns` stabilizes, it may make sense to move
750 // this check out of this branch. Alternatively, this loop could be implemented with
751 // autoderef and this check removed. For now though, don't break code compiling on
752 // stable with lots of `&`s and a low recursion limit, if anyone's done that.
753if self.tcx.recursion_limit().value_within_limit(pat_adjustments.len()) {
754// Preserve the smart pointer type for THIR lowering and closure upvar analysis.
755pat_adjustments.push(PatAdjustment { kind: pat_adjust_kind, source: expected });
756 } else {
757let guar = report_autoderef_recursion_limit_error(self.tcx, pat.span, expected);
758inner_ty = Ty::new_error(self.tcx, guar);
759 }
760drop(typeck_results);
761762// Recurse, using the old pat info to keep `current_depth` to its old value.
763 // Peeling smart pointers does not update the default binding mode.
764self.check_pat_inner(pat, opt_path_res, adjust_mode, inner_ty, pat_info)
765 }
766767/// How should the binding mode and expected type be adjusted?
768 ///
769 /// When the pattern contains a path, `opt_path_res` must be `Some(path_res)`.
770fn calc_adjust_mode(
771&self,
772 pat: &'tcx Pat<'tcx>,
773 opt_path_res: Option<Result<ResolvedPat<'tcx>, ErrorGuaranteed>>,
774 ) -> AdjustMode {
775match &pat.kind {
776// Type checking these product-like types successfully always require
777 // that the expected type be of those types and not reference types.
778PatKind::Tuple(..) | PatKind::Range(..) | PatKind::Slice(..) => AdjustMode::peel_all(),
779// When checking an explicit deref pattern, only peel reference types.
780 // FIXME(deref_patterns): If box patterns and deref patterns need to coexist, box
781 // patterns may want `PeelKind::Implicit`, stopping on encountering a box.
782PatKind::Box(_) | PatKind::Deref(_) => {
783 AdjustMode::Peel { kind: PeelKind::ExplicitDerefPat }
784 }
785// A never pattern behaves somewhat like a literal or unit variant.
786PatKind::Never => AdjustMode::peel_all(),
787// For patterns with paths, how we peel the scrutinee depends on the path's resolution.
788PatKind::Struct(..)
789 | PatKind::TupleStruct(..)
790 | PatKind::Expr(PatExpr { kind: PatExprKind::Path(_), .. }) => {
791// If there was an error resolving the path, default to peeling everything.
792opt_path_res.unwrap().map_or(AdjustMode::peel_all(), |pr| pr.adjust_mode())
793 }
794795// String and byte-string literals result in types `&str` and `&[u8]` respectively.
796 // All other literals result in non-reference types.
797 // As a result, we allow `if let 0 = &&0 {}` but not `if let "foo" = &&"foo" {}` unless
798 // `deref_patterns` is enabled.
799PatKind::Expr(lt) => {
800// Path patterns have already been handled, and inline const blocks currently
801 // aren't possible to write, so any handling for them would be untested.
802if truecfg!(debug_assertions)803 && self.tcx.features().deref_patterns()
804 && !#[allow(non_exhaustive_omitted_patterns)] match lt.kind {
PatExprKind::Lit { .. } => true,
_ => false,
}matches!(lt.kind, PatExprKind::Lit { .. })805 {
806::rustc_middle::util::bug::span_bug_fmt(lt.span,
format_args!("FIXME(deref_patterns): adjust mode unimplemented for {0:?}",
lt.kind));span_bug!(
807 lt.span,
808"FIXME(deref_patterns): adjust mode unimplemented for {:?}",
809 lt.kind
810 );
811 }
812// Call `resolve_vars_if_possible` here for inline const blocks.
813let lit_ty = self.resolve_vars_if_possible(self.check_pat_expr_unadjusted(lt));
814// If `deref_patterns` is enabled, allow `if let "foo" = &&"foo" {}`.
815if self.tcx.features().deref_patterns() {
816let mut peeled_ty = lit_ty;
817let mut pat_ref_layers = 0;
818while let ty::Ref(_, inner_ty, mutbl) =
819*self.resolve_vars_with_obligations(peeled_ty).kind()
820 {
821// We rely on references at the head of constants being immutable.
822if true {
if !mutbl.is_not() {
::core::panicking::panic("assertion failed: mutbl.is_not()")
};
};debug_assert!(mutbl.is_not());
823 pat_ref_layers += 1;
824 peeled_ty = inner_ty;
825 }
826 AdjustMode::Peel {
827 kind: PeelKind::Implicit { until_adt: None, pat_ref_layers },
828 }
829 } else {
830if lit_ty.is_ref() { AdjustMode::Pass } else { AdjustMode::peel_all() }
831 }
832 }
833834// Ref patterns are complicated, we handle them in `check_pat_ref`.
835PatKind::Ref(..)
836// No need to do anything on a missing pattern.
837| PatKind::Missing838// A `_` pattern works with any expected type, so there's no need to do anything.
839| PatKind::Wild840// A malformed pattern doesn't have an expected type, so let's just accept any type.
841| PatKind::Err(_)
842// Bindings also work with whatever the expected type is,
843 // and moreover if we peel references off, that will give us the wrong binding type.
844 // Also, we can have a subpattern `binding @ pat`.
845 // Each side of the `@` should be treated independently (like with OR-patterns).
846| PatKind::Binding(..)
847// An OR-pattern just propagates to each individual alternative.
848 // This is maximally flexible, allowing e.g., `Some(mut x) | &Some(mut x)`.
849 // In that example, `Some(mut x)` results in `Peel` whereas `&Some(mut x)` in `Reset`.
850| PatKind::Or(_)
851// Like or-patterns, guard patterns just propagate to their subpatterns.
852| PatKind::Guard(..) => AdjustMode::Pass,
853 }
854 }
855856/// Assuming `expected` is a reference type, determine whether to peel it before matching.
857fn should_peel_ref(&self, peel_kind: PeelKind, mut expected: Ty<'tcx>) -> bool {
858if true {
if !expected.is_ref() {
::core::panicking::panic("assertion failed: expected.is_ref()")
};
};debug_assert!(expected.is_ref());
859let pat_ref_layers = match peel_kind {
860 PeelKind::ExplicitDerefPat => 0,
861 PeelKind::Implicit { pat_ref_layers, .. } => pat_ref_layers,
862 };
863864// Most patterns don't have reference types, so we'll want to peel all references from the
865 // scrutinee before matching. To optimize for the common case, return early.
866if pat_ref_layers == 0 {
867return true;
868 }
869if true {
if !self.tcx.features().deref_patterns() {
{
::core::panicking::panic_fmt(format_args!("Peeling for patterns with reference types is gated by `deref_patterns`."));
}
};
};debug_assert!(
870self.tcx.features().deref_patterns(),
871"Peeling for patterns with reference types is gated by `deref_patterns`."
872);
873874// If the pattern has as many or more layers of reference as the expected type, we can match
875 // without peeling more, unless we find a smart pointer or `&mut` that we also need to peel.
876 // We don't treat `&` and `&mut` as interchangeable, but by peeling `&mut`s before matching,
877 // we can still, e.g., match on a `&mut str` with a string literal pattern. This is because
878 // string literal patterns may be used where `str` is expected.
879let mut expected_ref_layers = 0;
880while let ty::Ref(_, inner_ty, mutbl) = *expected.kind() {
881if mutbl.is_mut() {
882// Mutable references can't be in the final value of constants, thus they can't be
883 // at the head of their types, thus we should always peel `&mut`.
884return true;
885 }
886 expected_ref_layers += 1;
887 expected = inner_ty;
888 }
889pat_ref_layers < expected_ref_layers || self.should_peel_smart_pointer(peel_kind, expected)
890 }
891892/// Determine whether `expected` is a smart pointer type that should be peeled before matching.
893fn should_peel_smart_pointer(&self, peel_kind: PeelKind, expected: Ty<'tcx>) -> bool {
894// Explicit `deref!(_)` patterns match against smart pointers; don't peel in that case.
895if let PeelKind::Implicit { until_adt, .. } = peel_kind896// For simplicity, only apply overloaded derefs if `expected` is a known ADT.
897 // FIXME(deref_patterns): we'll get better diagnostics for users trying to
898 // implicitly deref generics if we allow them here, but primitives, tuples, and
899 // inference vars definitely should be stopped. Figure out what makes most sense.
900&& let ty::Adt(scrutinee_adt, _) = *expected.kind()
901// Don't peel if the pattern type already matches the scrutinee. E.g., stop here if
902 // matching on a `Cow<'a, T>` scrutinee with a `Cow::Owned(_)` pattern.
903&& until_adt != Some(scrutinee_adt.did())
904// At this point, the pattern isn't able to match `expected` without peeling. Check
905 // that it implements `Deref` before assuming it's a smart pointer, to get a normal
906 // type error instead of a missing impl error if not. This only checks for `Deref`,
907 // not `DerefPure`: we require that too, but we want a trait error if it's missing.
908&& let Some(deref_trait) = self.tcx.lang_items().deref_trait()
909 && self.type_implements_trait(deref_trait, [expected], self.param_env).may_apply()
910 {
911true
912} else {
913false
914}
915 }
916917fn check_pat_expr_unadjusted(&self, lt: &'tcx hir::PatExpr<'tcx>) -> Ty<'tcx> {
918let ty = match <.kind {
919 rustc_hir::PatExprKind::Lit { lit, negated } => {
920let ty = self.check_expr_lit(lit, lt.hir_id, Expectation::NoExpectation);
921if *negated {
922self.register_bound(
923ty,
924self.tcx.require_lang_item(LangItem::Neg, lt.span),
925ObligationCause::dummy_with_span(lt.span),
926 );
927 }
928ty929 }
930 rustc_hir::PatExprKind::Path(qpath) => {
931let (res, opt_ty, segments) =
932self.resolve_ty_and_res_fully_qualified_call(qpath, lt.hir_id, lt.span);
933self.instantiate_value_path(segments, opt_ty, res, lt.span, lt.span, lt.hir_id).0
934}
935 };
936self.write_ty(lt.hir_id, ty);
937ty938 }
939940fn check_pat_lit(
941&self,
942 span: Span,
943 expr: &hir::PatExpr<'tcx>,
944 lit_kind: &ast::LitKind,
945 expected: Ty<'tcx>,
946 ti: &TopInfo<'tcx>,
947 ) -> Ty<'tcx> {
948{
match expr.kind {
hir::PatExprKind::Lit { .. } => {}
ref left_val => {
::core::panicking::assert_matches_failed(left_val,
"hir::PatExprKind::Lit { .. }", ::core::option::Option::None);
}
}
};assert_matches!(expr.kind, hir::PatExprKind::Lit { .. });
949950// We've already computed the type above (when checking for a non-ref pat),
951 // so avoid computing it again.
952let ty = self.node_ty(expr.hir_id);
953954// Byte string patterns behave the same way as array patterns
955 // They can denote both statically and dynamically-sized byte arrays.
956 // Additionally, when `deref_patterns` is enabled, byte string literal patterns may have
957 // types `[u8]` or `[u8; N]`, in order to type, e.g., `deref!(b"..."): Vec<u8>`.
958let mut pat_ty = ty;
959if #[allow(non_exhaustive_omitted_patterns)] match lit_kind {
ast::LitKind::ByteStr(..) => true,
_ => false,
}matches!(lit_kind, ast::LitKind::ByteStr(..)) {
960let tcx = self.tcx;
961let expected = self.structurally_resolve_type(span, expected);
962match *expected.kind() {
963// Allow `b"...": &[u8]`
964ty::Ref(_, inner_ty, _)
965if self.resolve_vars_with_obligations(inner_ty).is_slice() =>
966 {
967{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:967",
"rustc_hir_typeck::pat", ::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(967u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message",
"expr.hir_id.local_id"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::TRACE <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("polymorphic byte string lit")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&expr.hir_id.local_id)
as &dyn Value))])
});
} else { ; }
};trace!(?expr.hir_id.local_id, "polymorphic byte string lit");
968pat_ty = Ty::new_imm_ref(
969tcx,
970tcx.lifetimes.re_static,
971Ty::new_slice(tcx, tcx.types.u8),
972 );
973 }
974// Allow `b"...": [u8; 3]` for `deref_patterns`
975ty::Array(..) if tcx.features().deref_patterns() => {
976pat_ty = match *ty.kind() {
977 ty::Ref(_, inner_ty, _) => inner_ty,
978_ => ::rustc_middle::util::bug::span_bug_fmt(span,
format_args!("found byte string literal with non-ref type {0:?}", ty))span_bug!(span, "found byte string literal with non-ref type {ty:?}"),
979 }
980 }
981// Allow `b"...": [u8]` for `deref_patterns`
982ty::Slice(..) if tcx.features().deref_patterns() => {
983pat_ty = Ty::new_slice(tcx, tcx.types.u8);
984 }
985// Otherwise, `b"...": &[u8; 3]`
986_ => {}
987 }
988 }
989990// When `deref_patterns` is enabled, in order to allow `deref!("..."): String`, we allow
991 // string literal patterns to have type `str`. This is accounted for when lowering to MIR.
992if self.tcx.features().deref_patterns()
993 && #[allow(non_exhaustive_omitted_patterns)] match lit_kind {
ast::LitKind::Str(..) => true,
_ => false,
}matches!(lit_kind, ast::LitKind::Str(..))994 && self.resolve_vars_with_obligations(expected).is_str()
995 {
996pat_ty = self.tcx.types.str_;
997 }
998999// Somewhat surprising: in this case, the subtyping relation goes the
1000 // opposite way as the other cases. Actually what we really want is not
1001 // a subtyping relation at all but rather that there exists a LUB
1002 // (so that they can be compared). However, in practice, constants are
1003 // always scalars or strings. For scalars subtyping is irrelevant,
1004 // and for strings `ty` is type is `&'static str`, so if we say that
1005 //
1006 // &'static str <: expected
1007 //
1008 // then that's equivalent to there existing a LUB.
1009let cause = self.pattern_cause(ti, span);
1010if let Err(mut err) = self.demand_suptype_with_origin(&cause, expected, pat_ty) {
1011// If scrutinee is String and pattern is &str, suggest .as_str()
1012let expected = self.resolve_vars_with_obligations(expected);
1013if let ty::Adt(adt, _) = expected.kind()
1014 && self.tcx.is_lang_item(adt.did(), LangItem::String)
1015 && pat_ty.is_ref()
1016 && pat_ty.peel_refs().is_str()
1017 && let Some(origin_expr) = ti.origin_expr
1018 {
1019err.span_suggestion_verbose(
1020origin_expr.span.shrink_to_hi(),
1021"consider converting the `String` to a `&str` using `.as_str()`",
1022".as_str()",
1023 Applicability::MachineApplicable,
1024 );
1025 }
1026err.emit();
1027 }
10281029pat_ty1030 }
10311032fn check_pat_range(
1033&self,
1034 span: Span,
1035 lhs: Option<&'tcx hir::PatExpr<'tcx>>,
1036 rhs: Option<&'tcx hir::PatExpr<'tcx>>,
1037 expected: Ty<'tcx>,
1038 ti: &TopInfo<'tcx>,
1039 ) -> Ty<'tcx> {
1040let calc_side = |opt_expr: Option<&'tcx hir::PatExpr<'tcx>>| match opt_expr {
1041None => None,
1042Some(expr) => {
1043let ty = self.check_pat_expr_unadjusted(expr);
1044// Check that the end-point is possibly of numeric or char type.
1045 // The early check here is not for correctness, but rather better
1046 // diagnostics (e.g. when `&str` is being matched, `expected` will
1047 // be peeled to `str` while ty here is still `&str`, if we don't
1048 // err early here, a rather confusing unification error will be
1049 // emitted instead).
1050let ty = self.resolve_vars_with_obligations(ty);
1051let fail =
1052 !(ty.is_numeric() || ty.is_char() || ty.is_ty_var() || ty.references_error());
1053Some((fail, ty, expr.span))
1054 }
1055 };
1056let mut lhs = calc_side(lhs);
1057let mut rhs = calc_side(rhs);
10581059if let (Some((true, ..)), _) | (_, Some((true, ..))) = (lhs, rhs) {
1060// There exists a side that didn't meet our criteria that the end-point
1061 // be of a numeric or char type, as checked in `calc_side` above.
1062let guar = self.emit_err_pat_range(span, lhs, rhs);
1063return Ty::new_error(self.tcx, guar);
1064 }
10651066// Unify each side with `expected`.
1067 // Subtyping doesn't matter here, as the value is some kind of scalar.
1068let demand_eqtype = |x: &mut _, y| {
1069if let Some((ref mut fail, x_ty, x_span)) = *x1070 && let Err(mut err) = self.demand_eqtype_pat_diag(x_span, expected, x_ty, ti)
1071 {
1072if let Some((_, y_ty, y_span)) = y {
1073self.endpoint_has_type(&mut err, y_span, y_ty);
1074 }
1075err.emit();
1076*fail = true;
1077 }
1078 };
1079demand_eqtype(&mut lhs, rhs);
1080demand_eqtype(&mut rhs, lhs);
10811082if let (Some((true, ..)), _) | (_, Some((true, ..))) = (lhs, rhs) {
1083return Ty::new_misc_error(self.tcx);
1084 }
10851086// Find the unified type and check if it's of numeric or char type again.
1087 // This check is needed if both sides are inference variables.
1088 // We require types to be resolved here so that we emit inference failure
1089 // rather than "_ is not a char or numeric".
1090let ty = self.structurally_resolve_type(span, expected);
1091if !(ty.is_numeric() || ty.is_char() || ty.references_error()) {
1092if let Some((ref mut fail, _, _)) = lhs {
1093*fail = true;
1094 }
1095if let Some((ref mut fail, _, _)) = rhs {
1096*fail = true;
1097 }
1098let guar = self.emit_err_pat_range(span, lhs, rhs);
1099return Ty::new_error(self.tcx, guar);
1100 }
1101ty1102 }
11031104fn endpoint_has_type(&self, err: &mut Diag<'_>, span: Span, ty: Ty<'_>) {
1105if !ty.references_error() {
1106err.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this is of type `{0}`", ty))
})format!("this is of type `{ty}`"));
1107 }
1108 }
11091110fn emit_err_pat_range(
1111&self,
1112 span: Span,
1113 lhs: Option<(bool, Ty<'tcx>, Span)>,
1114 rhs: Option<(bool, Ty<'tcx>, Span)>,
1115 ) -> ErrorGuaranteed {
1116let span = match (lhs, rhs) {
1117 (Some((true, ..)), Some((true, ..))) => span,
1118 (Some((true, _, sp)), _) => sp,
1119 (_, Some((true, _, sp))) => sp,
1120_ => ::rustc_middle::util::bug::span_bug_fmt(span,
format_args!("emit_err_pat_range: no side failed or exists but still error?"))span_bug!(span, "emit_err_pat_range: no side failed or exists but still error?"),
1121 };
1122let mut err = {
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("only `char` and numeric types are allowed in range patterns"))
})).with_code(E0029)
}struct_span_code_err!(
1123self.dcx(),
1124 span,
1125 E0029,
1126"only `char` and numeric types are allowed in range patterns"
1127);
1128let msg = |ty| {
1129let ty = self.resolve_vars_if_possible(ty);
1130::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this is of type `{0}` but it should be `char` or numeric",
ty))
})format!("this is of type `{ty}` but it should be `char` or numeric")1131 };
1132let mut one_side_err = |first_span, first_ty, second: Option<(bool, Ty<'tcx>, Span)>| {
1133err.span_label(first_span, msg(first_ty));
1134if let Some((_, ty, sp)) = second {
1135let ty = self.resolve_vars_if_possible(ty);
1136self.endpoint_has_type(&mut err, sp, ty);
1137 }
1138 };
1139match (lhs, rhs) {
1140 (Some((true, lhs_ty, lhs_sp)), Some((true, rhs_ty, rhs_sp))) => {
1141err.span_label(lhs_sp, msg(lhs_ty));
1142err.span_label(rhs_sp, msg(rhs_ty));
1143 }
1144 (Some((true, lhs_ty, lhs_sp)), rhs) => one_side_err(lhs_sp, lhs_ty, rhs),
1145 (lhs, Some((true, rhs_ty, rhs_sp))) => one_side_err(rhs_sp, rhs_ty, lhs),
1146_ => ::rustc_middle::util::bug::span_bug_fmt(span,
format_args!("Impossible, verified above."))span_bug!(span, "Impossible, verified above."),
1147 }
1148if (lhs, rhs).references_error() {
1149err.downgrade_to_delayed_bug();
1150 }
1151if self.tcx.sess.teach(err.code.unwrap()) {
1152err.note(
1153"In a match expression, only numbers and characters can be matched \
1154 against a range. This is because the compiler checks that the range \
1155 is non-empty at compile-time, and is unable to evaluate arbitrary \
1156 comparison functions. If you want to capture values of an orderable \
1157 type between two end-points, you can use a guard.",
1158 );
1159 }
1160err.emit()
1161 }
11621163fn check_pat_ident(
1164&self,
1165 pat: &'tcx Pat<'tcx>,
1166 user_bind_annot: BindingMode,
1167 var_id: HirId,
1168 ident: Ident,
1169 sub: Option<&'tcx Pat<'tcx>>,
1170 expected: Ty<'tcx>,
1171 pat_info: PatInfo<'tcx>,
1172 ) -> Ty<'tcx> {
1173let PatInfo { binding_mode: def_br, top_info: ti, .. } = pat_info;
11741175// Determine the binding mode...
1176let bm = match user_bind_annot {
1177BindingMode(ByRef::No, Mutability::Mut) if let ByRef::Yes(_, def_br_mutbl) = def_br => {
1178// Only mention the experimental `mut_ref` feature if if we're in edition 2024 and
1179 // using other experimental matching features compatible with it.
1180if pat.span.at_least_rust_2024()
1181 && (self.tcx.features().ref_pat_eat_one_layer_2024()
1182 || self.tcx.features().ref_pat_eat_one_layer_2024_structural())
1183 {
1184if !self.tcx.features().mut_ref() {
1185feature_err(
1186self.tcx.sess,
1187 sym::mut_ref,
1188pat.span.until(ident.span),
1189"binding cannot be both mutable and by-reference",
1190 )
1191 .emit();
1192 }
11931194BindingMode(def_br, Mutability::Mut)
1195 } else {
1196// `mut` resets the binding mode on edition <= 2021
1197self.add_rust_2024_migration_desugared_pat(
1198pat_info.top_info.hir_id,
1199pat,
1200't', // last char of `mut`
1201def_br_mutbl,
1202 );
1203BindingMode(ByRef::No, Mutability::Mut)
1204 }
1205 }
1206BindingMode(ByRef::No, mutbl) => BindingMode(def_br, mutbl),
1207BindingMode(ByRef::Yes(_, user_br_mutbl), _) => {
1208if let ByRef::Yes(_, def_br_mutbl) = def_br {
1209// `ref`/`ref mut` overrides the binding mode on edition <= 2021
1210self.add_rust_2024_migration_desugared_pat(
1211pat_info.top_info.hir_id,
1212pat,
1213match user_br_mutbl {
1214 Mutability::Not => 'f', // last char of `ref`
1215Mutability::Mut => 't', // last char of `ref mut`
1216},
1217def_br_mutbl,
1218 );
1219 }
1220user_bind_annot1221 }
1222 };
12231224// If there exists a pinned reference in the pattern but the binding is not pinned,
1225 // it means the binding is unpinned and thus requires an `Unpin` bound.
1226if pat_info.max_pinnedness == PinnednessCap::Pinned1227 && #[allow(non_exhaustive_omitted_patterns)] match bm.0 {
ByRef::Yes(Pinnedness::Not, _) => true,
_ => false,
}matches!(bm.0, ByRef::Yes(Pinnedness::Not, _))1228 {
1229self.register_bound(
1230expected,
1231self.tcx.require_lang_item(hir::LangItem::Unpin, pat.span),
1232self.misc(pat.span),
1233 )
1234 }
12351236if #[allow(non_exhaustive_omitted_patterns)] match bm.0 {
ByRef::Yes(_, Mutability::Mut) => true,
_ => false,
}matches!(bm.0, ByRef::Yes(_, Mutability::Mut))1237 && let MutblCap::WeaklyNot(and_pat_span) = pat_info.max_ref_mutbl
1238 {
1239let mut err = {
self.dcx().struct_span_err(ident.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("cannot borrow as mutable inside an `&` pattern"))
})).with_code(E0596)
}struct_span_code_err!(
1240self.dcx(),
1241 ident.span,
1242 E0596,
1243"cannot borrow as mutable inside an `&` pattern"
1244);
12451246if let Some(span) = and_pat_span {
1247err.span_suggestion(
1248span,
1249"replace this `&` with `&mut`",
1250"&mut ",
1251 Applicability::MachineApplicable,
1252 );
1253 }
1254err.emit();
1255 }
12561257// ...and store it in a side table:
1258self.typeck_results.borrow_mut().pat_binding_modes_mut().insert(pat.hir_id, bm);
12591260{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:1260",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(1260u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("check_pat_ident: pat.hir_id={0:?} bm={1:?}",
pat.hir_id, bm) as &dyn Value))])
});
} else { ; }
};debug!("check_pat_ident: pat.hir_id={:?} bm={:?}", pat.hir_id, bm);
12611262let local_ty = self.local_ty(pat.span, pat.hir_id);
1263let eq_ty = match bm.0 {
1264 ByRef::Yes(pinnedness, mutbl) => {
1265// If the binding is like `ref x | ref mut x`,
1266 // then `x` is assigned a value of type `&M T` where M is the
1267 // mutability and T is the expected type.
1268 //
1269 // Under pin ergonomics, if the binding is like `ref pin const|mut x`,
1270 // then `x` is assigned a value of type `&pin M T` where M is the
1271 // mutability and T is the expected type.
1272 //
1273 // `x` is assigned a value of type `&M T`, hence `&M T <: typeof(x)`
1274 // is required. However, we use equality, which is stronger.
1275 // See (note_1) for an explanation.
1276self.new_ref_ty(pat.span, pinnedness, mutbl, expected)
1277 }
1278// Otherwise, the type of x is the expected type `T`.
1279ByRef::No => expected, // As above, `T <: typeof(x)` is required, but we use equality, see (note_1).
1280};
12811282// We have a concrete type for the local, so we do not need to taint it and hide follow up errors *using* the local.
1283let _ = self.demand_eqtype_pat(pat.span, eq_ty, local_ty, &ti);
12841285// If there are multiple arms, make sure they all agree on
1286 // what the type of the binding `x` ought to be.
1287if var_id != pat.hir_id {
1288self.check_binding_alt_eq_ty(user_bind_annot, pat.span, var_id, local_ty, &ti);
1289 }
12901291if let Some(p) = sub {
1292self.check_pat(p, expected, pat_info);
1293 }
12941295local_ty1296 }
12971298/// When a variable is bound several times in a `PatKind::Or`, it'll resolve all of the
1299 /// subsequent bindings of the same name to the first usage. Verify that all of these
1300 /// bindings have the same type by comparing them all against the type of that first pat.
1301fn check_binding_alt_eq_ty(
1302&self,
1303 ba: BindingMode,
1304 span: Span,
1305 var_id: HirId,
1306 ty: Ty<'tcx>,
1307 ti: &TopInfo<'tcx>,
1308 ) {
1309let var_ty = self.local_ty(span, var_id);
1310if let Err(mut err) = self.demand_eqtype_pat_diag(span, var_ty, ty, ti) {
1311let var_ty = self.resolve_vars_if_possible(var_ty);
1312let msg = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("first introduced with type `{0}` here",
var_ty))
})format!("first introduced with type `{var_ty}` here");
1313err.span_label(self.tcx.hir_span(var_id), msg);
1314let in_match = self.tcx.hir_parent_iter(var_id).any(|(_, n)| {
1315#[allow(non_exhaustive_omitted_patterns)] match n {
hir::Node::Expr(hir::Expr {
kind: hir::ExprKind::Match(.., hir::MatchSource::Normal), .. }) =>
true,
_ => false,
}matches!(
1316 n,
1317 hir::Node::Expr(hir::Expr {
1318 kind: hir::ExprKind::Match(.., hir::MatchSource::Normal),
1319 ..
1320 })
1321 )1322 });
1323let pre = if in_match { "in the same arm, " } else { "" };
1324err.note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}a binding must have the same type in all alternatives",
pre))
})format!("{pre}a binding must have the same type in all alternatives"));
1325self.suggest_adding_missing_ref_or_removing_ref(
1326&mut err,
1327span,
1328var_ty,
1329self.resolve_vars_if_possible(ty),
1330ba,
1331 );
1332err.emit();
1333 }
1334 }
13351336fn suggest_adding_missing_ref_or_removing_ref(
1337&self,
1338 err: &mut Diag<'_>,
1339 span: Span,
1340 expected: Ty<'tcx>,
1341 actual: Ty<'tcx>,
1342 ba: BindingMode,
1343 ) {
1344match (expected.kind(), actual.kind(), ba) {
1345 (ty::Ref(_, inner_ty, _), _, BindingMode::NONE)
1346if self.can_eq(self.param_env, *inner_ty, actual) =>
1347 {
1348err.span_suggestion_verbose(
1349span.shrink_to_lo(),
1350"consider adding `ref`",
1351"ref ",
1352 Applicability::MaybeIncorrect,
1353 );
1354 }
1355 (_, ty::Ref(_, inner_ty, _), BindingMode::REF)
1356if self.can_eq(self.param_env, expected, *inner_ty) =>
1357 {
1358err.span_suggestion_verbose(
1359span.with_hi(span.lo() + BytePos(4)),
1360"consider removing `ref`",
1361"",
1362 Applicability::MaybeIncorrect,
1363 );
1364 }
1365_ => (),
1366 }
1367 }
13681369/// Precondition: pat is a `Ref(_)` pattern
1370fn borrow_pat_suggestion(&self, err: &mut Diag<'_>, pat: &Pat<'_>) {
1371let tcx = self.tcx;
1372if let PatKind::Ref(inner, pinned, mutbl) = pat.kind
1373 && let PatKind::Binding(_, _, binding, ..) = inner.kind
1374 {
1375let binding_parent = tcx.parent_hir_node(pat.hir_id);
1376{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:1376",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(1376u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["inner", "pat",
"binding_parent"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&inner) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&pat) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&binding_parent)
as &dyn Value))])
});
} else { ; }
};debug!(?inner, ?pat, ?binding_parent);
13771378let pin_and_mut = pinned.prefix_str(mutbl).trim_end();
13791380let mut_var_suggestion = 'block: {
1381if mutbl.is_not() {
1382break 'block None;
1383 }
13841385let ident_kind = match binding_parent {
1386 hir::Node::Param(_) => "parameter",
1387 hir::Node::LetStmt(_) => "variable",
1388 hir::Node::Arm(_) => "binding",
13891390// Provide diagnostics only if the parent pattern is struct-like,
1391 // i.e. where `mut binding` makes sense
1392hir::Node::Pat(Pat { kind, .. }) => match kind {
1393 PatKind::Struct(..)
1394 | PatKind::TupleStruct(..)
1395 | PatKind::Or(..)
1396 | PatKind::Guard(..)
1397 | PatKind::Tuple(..)
1398 | PatKind::Slice(..) => "binding",
13991400 PatKind::Missing1401 | PatKind::Wild1402 | PatKind::Never1403 | PatKind::Binding(..)
1404 | PatKind::Box(..)
1405 | PatKind::Deref(_)
1406 | PatKind::Ref(..)
1407 | PatKind::Expr(..)
1408 | PatKind::Range(..)
1409 | PatKind::Err(_) => break 'block None,
1410 },
14111412// Don't provide suggestions in other cases
1413_ => break 'block None,
1414 };
14151416Some((
1417pat.span,
1418::alloc::__export::must_use({
::alloc::fmt::format(format_args!("to declare a mutable {0} use",
ident_kind))
})format!("to declare a mutable {ident_kind} use"),
1419::alloc::__export::must_use({
::alloc::fmt::format(format_args!("mut {0}", binding))
})format!("mut {binding}"),
1420 ))
1421 };
14221423match binding_parent {
1424 hir::Node::Param(hir::Param { ty_span, pat, .. })
1425if pat.span != *ty_span1426 && pinned.is_pinned()
1427 && !tcx.features().pin_ergonomics() =>
1428 {
1429// FIXME(pin_ergonomics): Once `pin_ergonomics` is stabilized, remove this
1430 // gate and allow the pinned reference type-position suggestion unconditionally.
1431}
1432// Check that there is explicit type (ie this is not a closure param with inferred type)
1433 // so we don't suggest moving something to the type that does not exist
1434hir::Node::Param(hir::Param { ty_span, pat, .. }) if pat.span != *ty_span => {
1435err.multipart_suggestion(
1436::alloc::__export::must_use({
::alloc::fmt::format(format_args!("to take parameter `{0}` by reference, move `&{1}` to the type",
binding, pin_and_mut))
})format!("to take parameter `{binding}` by reference, move `&{pin_and_mut}` to the type"),
1437::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(pat.span.until(inner.span), "".to_owned()),
(ty_span.shrink_to_lo(),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("&{0}",
pinned.prefix_str(mutbl)))
}))]))vec![
1438 (pat.span.until(inner.span), "".to_owned()),
1439 (ty_span.shrink_to_lo(), format!("&{}", pinned.prefix_str(mutbl))),
1440 ],
1441 Applicability::MachineApplicable1442 );
14431444if let Some((sp, msg, sugg)) = mut_var_suggestion {
1445err.span_note(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: `{1}`", msg, sugg))
})format!("{msg}: `{sugg}`"));
1446 }
1447 }
1448 hir::Node::Pat(pt) if let PatKind::TupleStruct(_, pat_arr, _) = pt.kind => {
1449for i in pat_arr.iter() {
1450if let PatKind::Ref(the_ref, _, _) = i.kind
1451 && let PatKind::Binding(mt, _, ident, _) = the_ref.kind
1452 {
1453let BindingMode(_, mtblty) = mt;
1454 err.span_suggestion_verbose(
1455 i.span,
1456::alloc::__export::must_use({
::alloc::fmt::format(format_args!("consider removing `&{0}` from the pattern",
pin_and_mut))
})format!("consider removing `&{pin_and_mut}` from the pattern"),
1457 mtblty.prefix_str().to_string() + &ident.name.to_string(),
1458 Applicability::MaybeIncorrect,
1459 );
1460 }
1461 }
1462if let Some((sp, msg, sugg)) = mut_var_suggestion {
1463err.span_note(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: `{1}`", msg, sugg))
})format!("{msg}: `{sugg}`"));
1464 }
1465 }
1466 hir::Node::Param(_) | hir::Node::Arm(_) | hir::Node::Pat(_) => {
1467// rely on match ergonomics or it might be nested `&&pat`
1468err.span_suggestion_verbose(
1469pat.span.until(inner.span),
1470::alloc::__export::must_use({
::alloc::fmt::format(format_args!("consider removing `&{0}` from the pattern",
pin_and_mut))
})format!("consider removing `&{pin_and_mut}` from the pattern"),
1471"",
1472 Applicability::MaybeIncorrect,
1473 );
14741475if let Some((sp, msg, sugg)) = mut_var_suggestion {
1476err.span_note(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: `{1}`", msg, sugg))
})format!("{msg}: `{sugg}`"));
1477 }
1478 }
1479_ if let Some((sp, msg, sugg)) = mut_var_suggestion => {
1480err.span_suggestion(sp, msg, sugg, Applicability::MachineApplicable);
1481 }
1482_ => {} // don't provide suggestions in other cases #55175
1483}
1484 }
1485 }
14861487fn check_dereferenceable(
1488&self,
1489 span: Span,
1490 expected: Ty<'tcx>,
1491 inner: &Pat<'_>,
1492 ) -> Result<(), ErrorGuaranteed> {
1493if let PatKind::Binding(..) = inner.kind
1494 && let Some(pointee_ty) = self.shallow_resolve(expected).builtin_deref(true)
1495 && let ty::Dynamic(..) = pointee_ty.kind()
1496 {
1497// This is "x = dyn SomeTrait" being reduced from
1498 // "let &x = &dyn SomeTrait" or "let box x = Box<dyn SomeTrait>", an error.
1499let type_str = self.ty_to_string(expected);
1500let mut err = {
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("type `{0}` cannot be dereferenced",
type_str))
})).with_code(E0033)
}struct_span_code_err!(
1501self.dcx(),
1502 span,
1503 E0033,
1504"type `{}` cannot be dereferenced",
1505 type_str
1506 );
1507err.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("type `{0}` cannot be dereferenced",
type_str))
})format!("type `{type_str}` cannot be dereferenced"));
1508if self.tcx.sess.teach(err.code.unwrap()) {
1509err.note(CANNOT_IMPLICITLY_DEREF_POINTER_TRAIT_OBJ);
1510 }
1511return Err(err.emit());
1512 }
1513Ok(())
1514 }
15151516fn resolve_pat_struct(
1517&self,
1518 pat: &'tcx Pat<'tcx>,
1519 qpath: &hir::QPath<'tcx>,
1520 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1521// Resolve the path and check the definition for errors.
1522let (variant, pat_ty) = self.check_struct_path(qpath, pat.hir_id)?;
1523Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::Struct { variant } })
1524 }
15251526fn check_pat_struct(
1527&self,
1528 pat: &'tcx Pat<'tcx>,
1529 fields: &'tcx [hir::PatField<'tcx>],
1530 has_rest_pat: bool,
1531 pat_ty: Ty<'tcx>,
1532 variant: &'tcx VariantDef,
1533 expected: Ty<'tcx>,
1534 pat_info: PatInfo<'tcx>,
1535 ) -> Ty<'tcx> {
1536// Type-check the path.
1537let had_err = self.demand_eqtype_pat(pat.span, expected, pat_ty, &pat_info.top_info);
15381539// Type-check subpatterns.
1540match self.check_struct_pat_fields(pat_ty, pat, variant, fields, has_rest_pat, pat_info) {
1541Ok(()) => match had_err {
1542Ok(()) => pat_ty,
1543Err(guar) => Ty::new_error(self.tcx, guar),
1544 },
1545Err(guar) => Ty::new_error(self.tcx, guar),
1546 }
1547 }
15481549fn resolve_pat_path(
1550&self,
1551 path_id: HirId,
1552 span: Span,
1553 qpath: &'tcx hir::QPath<'_>,
1554 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1555let tcx = self.tcx;
15561557let (res, opt_ty, segments) =
1558self.resolve_ty_and_res_fully_qualified_call(qpath, path_id, span);
1559match res {
1560 Res::Err => {
1561let e =
1562self.dcx().span_delayed_bug(qpath.span(), "`Res::Err` but no error emitted");
1563self.set_tainted_by_errors(e);
1564return Err(e);
1565 }
1566 Res::Def(DefKind::AssocFn | DefKind::Ctor(_, CtorKind::Fn) | DefKind::Variant, _) => {
1567let expected = "unit struct, unit variant or constant";
1568let e = report_unexpected_variant_res(tcx, res, None, qpath, span, E0533, expected);
1569return Err(e);
1570 }
1571 Res::SelfCtor(def_id) => {
1572if let ty::Adt(adt_def, _) = *tcx.type_of(def_id).skip_binder().kind()
1573 && adt_def.is_struct()
1574 && let Some((CtorKind::Const, _)) = adt_def.non_enum_variant().ctor
1575 {
1576// Ok, we allow unit struct ctors in patterns only.
1577} else {
1578let e = report_unexpected_variant_res(
1579tcx,
1580res,
1581None,
1582qpath,
1583span,
1584E0533,
1585"unit struct",
1586 );
1587return Err(e);
1588 }
1589 }
1590 Res::Def(
1591 DefKind::Ctor(_, CtorKind::Const)
1592 | DefKind::Const { .. }
1593 | DefKind::AssocConst { .. }
1594 | DefKind::ConstParam,
1595_,
1596 ) => {} // OK
1597_ => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern resolution: {0:?}",
res))bug!("unexpected pattern resolution: {:?}", res),
1598 }
15991600// Find the type of the path pattern, for later checking.
1601let (pat_ty, pat_res) =
1602self.instantiate_value_path(segments, opt_ty, res, span, span, path_id);
1603Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::Path { res, pat_res, segments } })
1604 }
16051606fn check_pat_path(
1607&self,
1608 pat_id_for_diag: HirId,
1609 span: Span,
1610 resolved: &ResolvedPat<'tcx>,
1611 expected: Ty<'tcx>,
1612 ti: &TopInfo<'tcx>,
1613 ) -> Ty<'tcx> {
1614if let Err(err) =
1615self.demand_suptype_with_origin(&self.pattern_cause(ti, span), expected, resolved.ty)
1616 {
1617self.emit_bad_pat_path(err, pat_id_for_diag, span, resolved);
1618 }
1619resolved.ty
1620 }
16211622fn maybe_suggest_range_literal(
1623&self,
1624 e: &mut Diag<'_>,
1625 opt_def_id: Option<hir::def_id::DefId>,
1626 ident: Ident,
1627 ) -> bool {
1628if let Some(def_id) = opt_def_id1629 && let Some(hir::Node::Item(hir::Item {
1630 kind: hir::ItemKind::Const(_, _, _, ct_rhs),
1631 ..
1632 })) = self.tcx.hir_get_if_local(def_id)
1633 && let hir::Node::Expr(expr) = self.tcx.hir_node(ct_rhs.hir_id())
1634 && hir::is_range_literal(expr)
1635 {
1636let span = self.tcx.hir_span(ct_rhs.hir_id());
1637if let Ok(snip) = self.tcx.sess.source_map().span_to_snippet(span) {
1638e.span_suggestion_verbose(
1639ident.span,
1640"you may want to move the range into the match block",
1641snip,
1642 Applicability::MachineApplicable,
1643 );
1644return true;
1645 }
1646 }
1647false
1648}
16491650fn emit_bad_pat_path(
1651&self,
1652mut e: Diag<'_>,
1653 hir_id: HirId,
1654 pat_span: Span,
1655 resolved_pat: &ResolvedPat<'tcx>,
1656 ) {
1657let ResolvedPatKind::Path { res, pat_res, segments } = resolved_pat.kind else {
1658::rustc_middle::util::bug::span_bug_fmt(pat_span,
format_args!("unexpected resolution for path pattern: {0:?}",
resolved_pat));span_bug!(pat_span, "unexpected resolution for path pattern: {resolved_pat:?}");
1659 };
16601661let span = match (self.tcx.hir_res_span(pat_res), res.opt_def_id()) {
1662 (Some(span), _) => span,
1663 (None, Some(def_id)) => self.tcx.def_span(def_id),
1664 (None, None) => {
1665e.emit();
1666return;
1667 }
1668 };
1669if let [hir::PathSegment { ident, args: None, .. }] = segments1670 && e.suggestions.len() == 0
1671{
1672e.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} defined here", res.descr()))
})format!("{} defined here", res.descr()));
1673e.span_label(
1674pat_span,
1675::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` is interpreted as {1} {2}, not a new binding",
ident, res.article(), res.descr()))
})format!(
1676"`{}` is interpreted as {} {}, not a new binding",
1677 ident,
1678 res.article(),
1679 res.descr(),
1680 ),
1681 );
1682match self.tcx.parent_hir_node(hir_id) {
1683 hir::Node::PatField(..) => {
1684e.span_suggestion_verbose(
1685ident.span.shrink_to_hi(),
1686"bind the struct field to a different name instead",
1687::alloc::__export::must_use({
::alloc::fmt::format(format_args!(": other_{0}",
ident.as_str().to_lowercase()))
})format!(": other_{}", ident.as_str().to_lowercase()),
1688 Applicability::HasPlaceholders,
1689 );
1690 }
1691_ => {
1692let (type_def_id, item_def_id) = match resolved_pat.ty.kind() {
1693 ty::Adt(def, _) => match res {
1694 Res::Def(DefKind::Const { .. }, def_id) => {
1695 (Some(def.did()), Some(def_id))
1696 }
1697_ => (None, None),
1698 },
1699_ => (None, None),
1700 };
17011702let is_range = #[allow(non_exhaustive_omitted_patterns)] match type_def_id.and_then(|id|
self.tcx.as_lang_item(id)) {
Some(LangItem::Range | LangItem::RangeFrom | LangItem::RangeTo |
LangItem::RangeFull | LangItem::RangeInclusiveStruct |
LangItem::RangeToInclusive) => true,
_ => false,
}matches!(
1703 type_def_id.and_then(|id| self.tcx.as_lang_item(id)),
1704Some(
1705 LangItem::Range
1706 | LangItem::RangeFrom
1707 | LangItem::RangeTo
1708 | LangItem::RangeFull
1709 | LangItem::RangeInclusiveStruct
1710 | LangItem::RangeToInclusive,
1711 )
1712 );
1713if is_range {
1714if !self.maybe_suggest_range_literal(&mut e, item_def_id, *ident) {
1715let msg = "constants only support matching by type, \
1716 if you meant to match against a range of values, \
1717 consider using a range pattern like `min ..= max` in the match block";
1718e.note(msg);
1719 }
1720 } else {
1721let msg = "introduce a new binding instead";
1722let sugg = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("other_{0}",
ident.as_str().to_lowercase()))
})format!("other_{}", ident.as_str().to_lowercase());
1723e.span_suggestion_verbose(
1724ident.span,
1725msg,
1726sugg,
1727 Applicability::HasPlaceholders,
1728 );
1729 }
1730 }
1731 };
1732 }
1733e.emit();
1734 }
17351736fn resolve_pat_tuple_struct(
1737&self,
1738 pat: &'tcx Pat<'tcx>,
1739 qpath: &'tcx hir::QPath<'tcx>,
1740 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1741let tcx = self.tcx;
1742let report_unexpected_res = |res: Res| {
1743let expected = "tuple struct or tuple variant";
1744let e = report_unexpected_variant_res(tcx, res, None, qpath, pat.span, E0164, expected);
1745Err(e)
1746 };
17471748// Resolve the path and check the definition for errors.
1749let (res, opt_ty, segments) =
1750self.resolve_ty_and_res_fully_qualified_call(qpath, pat.hir_id, pat.span);
1751if res == Res::Err {
1752let e = self.dcx().span_delayed_bug(pat.span, "`Res::Err` but no error emitted");
1753self.set_tainted_by_errors(e);
1754return Err(e);
1755 }
17561757// Type-check the path.
1758let (pat_ty, res) =
1759self.instantiate_value_path(segments, opt_ty, res, pat.span, pat.span, pat.hir_id);
1760if !pat_ty.is_fn() {
1761return report_unexpected_res(res);
1762 }
17631764let variant = match res {
1765 Res::Err => {
1766self.dcx().span_bug(pat.span, "`Res::Err` but no error emitted");
1767 }
1768 Res::Def(DefKind::AssocConst { .. } | DefKind::AssocFn, _) => {
1769return report_unexpected_res(res);
1770 }
1771 Res::Def(DefKind::Ctor(_, CtorKind::Fn), _) => tcx.expect_variant_res(res),
1772_ => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern resolution: {0:?}",
res))bug!("unexpected pattern resolution: {:?}", res),
1773 };
17741775// Replace constructor type with constructed type for tuple struct patterns.
1776let pat_ty = pat_ty.fn_sig(tcx).output();
1777let pat_ty = pat_ty.no_bound_vars().expect("expected fn type");
17781779Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::TupleStruct { res, variant } })
1780 }
17811782fn check_pat_tuple_struct(
1783&self,
1784 pat: &'tcx Pat<'tcx>,
1785 qpath: &'tcx hir::QPath<'tcx>,
1786 subpats: &'tcx [Pat<'tcx>],
1787 ddpos: hir::DotDotPos,
1788 res: Res,
1789 pat_ty: Ty<'tcx>,
1790 variant: &'tcx VariantDef,
1791 expected: Ty<'tcx>,
1792 pat_info: PatInfo<'tcx>,
1793 ) -> Ty<'tcx> {
1794let tcx = self.tcx;
1795let on_error = |e| {
1796for pat in subpats {
1797self.check_pat(pat, Ty::new_error(tcx, e), pat_info);
1798 }
1799 };
18001801// Type-check the tuple struct pattern against the expected type.
1802let had_err = self.demand_eqtype_pat(pat.span, expected, pat_ty, &pat_info.top_info);
18031804// Type-check subpatterns.
1805if subpats.len() == variant.fields.len()
1806 || subpats.len() < variant.fields.len() && ddpos.as_opt_usize().is_some()
1807 {
1808let ty::Adt(_, args) = pat_ty.kind() else {
1809::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern type {0:?}",
pat_ty));bug!("unexpected pattern type {:?}", pat_ty);
1810 };
1811for (i, subpat) in subpats.iter().enumerate_and_adjust(variant.fields.len(), ddpos) {
1812let field = &variant.fields[FieldIdx::from_usize(i)];
1813let field_ty = self.field_ty(subpat.span, field, args);
1814self.check_pat(subpat, field_ty, pat_info);
18151816self.tcx.check_stability(
1817 variant.fields[FieldIdx::from_usize(i)].did,
1818Some(subpat.hir_id),
1819 subpat.span,
1820None,
1821 );
1822 }
1823if let Err(e) = had_err {
1824on_error(e);
1825return Ty::new_error(tcx, e);
1826 }
1827 } else {
1828let e = self.emit_err_pat_wrong_number_of_fields(
1829pat.span,
1830res,
1831qpath,
1832subpats,
1833&variant.fields.raw,
1834expected,
1835had_err,
1836 );
1837on_error(e);
1838return Ty::new_error(tcx, e);
1839 }
1840pat_ty1841 }
18421843fn emit_err_pat_wrong_number_of_fields(
1844&self,
1845 pat_span: Span,
1846 res: Res,
1847 qpath: &hir::QPath<'_>,
1848 subpats: &'tcx [Pat<'tcx>],
1849 fields: &'tcx [ty::FieldDef],
1850 expected: Ty<'tcx>,
1851 had_err: Result<(), ErrorGuaranteed>,
1852 ) -> ErrorGuaranteed {
1853let subpats_ending = if subpats.len() == 1 { "" } else { "s" }pluralize!(subpats.len());
1854let fields_ending = if fields.len() == 1 { "" } else { "s" }pluralize!(fields.len());
18551856let subpat_spans = if subpats.is_empty() {
1857::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[pat_span]))vec![pat_span]1858 } else {
1859subpats.iter().map(|p| p.span).collect()
1860 };
1861let last_subpat_span = *subpat_spans.last().unwrap();
1862let res_span = self.tcx.def_span(res.def_id());
1863let def_ident_span = self.tcx.def_ident_span(res.def_id()).unwrap_or(res_span);
1864let field_def_spans = if fields.is_empty() {
1865::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[res_span]))vec![res_span]1866 } else {
1867fields.iter().map(|f| f.ident(self.tcx).span).collect()
1868 };
1869let last_field_def_span = *field_def_spans.last().unwrap();
18701871let mut err = {
self.dcx().struct_span_err(MultiSpan::from_spans(subpat_spans),
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this pattern has {0} field{1}, but the corresponding {2} has {3} field{4}",
subpats.len(), subpats_ending, res.descr(), fields.len(),
fields_ending))
})).with_code(E0023)
}struct_span_code_err!(
1872self.dcx(),
1873 MultiSpan::from_spans(subpat_spans),
1874 E0023,
1875"this pattern has {} field{}, but the corresponding {} has {} field{}",
1876 subpats.len(),
1877 subpats_ending,
1878 res.descr(),
1879 fields.len(),
1880 fields_ending,
1881 );
1882err.span_label(
1883last_subpat_span,
1884::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected {0} field{1}, found {2}",
fields.len(), fields_ending, subpats.len()))
})format!("expected {} field{}, found {}", fields.len(), fields_ending, subpats.len()),
1885 );
1886if self.tcx.sess.source_map().is_multiline(qpath.span().between(last_subpat_span)) {
1887err.span_label(qpath.span(), "");
1888 }
1889if self.tcx.sess.source_map().is_multiline(def_ident_span.between(last_field_def_span)) {
1890err.span_label(def_ident_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} defined here", res.descr()))
})format!("{} defined here", res.descr()));
1891 }
1892for span in &field_def_spans[..field_def_spans.len() - 1] {
1893 err.span_label(*span, "");
1894 }
1895err.span_label(
1896last_field_def_span,
1897::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} has {1} field{2}", res.descr(),
fields.len(), fields_ending))
})format!("{} has {} field{}", res.descr(), fields.len(), fields_ending),
1898 );
18991900// Identify the case `Some(x, y)` where the expected type is e.g. `Option<(T, U)>`.
1901 // More generally, the expected type wants a tuple variant with one field of an
1902 // N-arity-tuple, e.g., `V_i((p_0, .., p_N))`. Meanwhile, the user supplied a pattern
1903 // with the subpatterns directly in the tuple variant pattern, e.g., `V_i(p_0, .., p_N)`.
1904let missing_parentheses = match (expected.kind(), fields, had_err) {
1905// #67037: only do this if we could successfully type-check the expected type against
1906 // the tuple struct pattern. Otherwise the args could get out of range on e.g.,
1907 // `let P() = U;` where `P != U` with `struct Box<T>(T);`.
1908(ty::Adt(_, args), [field], Ok(())) => {
1909let field_ty = self.field_ty(pat_span, field, args);
1910match field_ty.kind() {
1911 ty::Tuple(fields) => fields.len() == subpats.len(),
1912_ => false,
1913 }
1914 }
1915_ => false,
1916 };
1917if missing_parentheses {
1918let (left, right) = match subpats {
1919// This is the zero case; we aim to get the "hi" part of the `QPath`'s
1920 // span as the "lo" and then the "hi" part of the pattern's span as the "hi".
1921 // This looks like:
1922 //
1923 // help: missing parentheses
1924 // |
1925 // L | let A(()) = A(());
1926 // | ^ ^
1927[] => (qpath.span().shrink_to_hi(), pat_span),
1928// Easy case. Just take the "lo" of the first sub-pattern and the "hi" of the
1929 // last sub-pattern. In the case of `A(x)` the first and last may coincide.
1930 // This looks like:
1931 //
1932 // help: missing parentheses
1933 // |
1934 // L | let A((x, y)) = A((1, 2));
1935 // | ^ ^
1936[first, ..] => (first.span.shrink_to_lo(), subpats.last().unwrap().span),
1937 };
1938err.multipart_suggestion(
1939"missing parentheses",
1940::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[(left, "(".to_string()), (right.shrink_to_hi(), ")".to_string())]))vec![(left, "(".to_string()), (right.shrink_to_hi(), ")".to_string())],
1941 Applicability::MachineApplicable,
1942 );
1943 } else if fields.len() > subpats.len() && pat_span != DUMMY_SP {
1944let after_fields_span = pat_span.with_hi(pat_span.hi() - BytePos(1)).shrink_to_hi();
1945let all_fields_span = match subpats {
1946 [] => after_fields_span,
1947 [field] => field.span,
1948 [first, .., last] => first.span.to(last.span),
1949 };
19501951// Check if all the fields in the pattern are wildcards.
1952let all_wildcards = subpats.iter().all(|pat| #[allow(non_exhaustive_omitted_patterns)] match pat.kind {
PatKind::Wild => true,
_ => false,
}matches!(pat.kind, PatKind::Wild));
1953let first_tail_wildcard =
1954subpats.iter().enumerate().fold(None, |acc, (pos, pat)| match (acc, &pat.kind) {
1955 (None, PatKind::Wild) => Some(pos),
1956 (Some(_), PatKind::Wild) => acc,
1957_ => None,
1958 });
1959let tail_span = match first_tail_wildcard {
1960None => after_fields_span,
1961Some(0) => subpats[0].span.to(after_fields_span),
1962Some(pos) => subpats[pos - 1].span.shrink_to_hi().to(after_fields_span),
1963 };
19641965// FIXME: heuristic-based suggestion to check current types for where to add `_`.
1966let mut wildcard_sugg = ::alloc::vec::from_elem("_", fields.len() - subpats.len())vec!["_"; fields.len() - subpats.len()].join(", ");
1967if !subpats.is_empty() {
1968wildcard_sugg = String::from(", ") + &wildcard_sugg;
1969 }
19701971err.span_suggestion_verbose(
1972after_fields_span,
1973"use `_` to explicitly ignore each field",
1974wildcard_sugg,
1975 Applicability::MaybeIncorrect,
1976 );
19771978// Only suggest `..` if more than one field is missing
1979 // or the pattern consists of all wildcards.
1980if fields.len() - subpats.len() > 1 || all_wildcards {
1981if subpats.is_empty() || all_wildcards {
1982err.span_suggestion_verbose(
1983all_fields_span,
1984"use `..` to ignore all fields",
1985"..",
1986 Applicability::MaybeIncorrect,
1987 );
1988 } else {
1989err.span_suggestion_verbose(
1990tail_span,
1991"use `..` to ignore the rest of the fields",
1992", ..",
1993 Applicability::MaybeIncorrect,
1994 );
1995 }
1996 }
1997 }
19981999err.emit()
2000 }
20012002fn check_pat_tuple(
2003&self,
2004 span: Span,
2005 elements: &'tcx [Pat<'tcx>],
2006 ddpos: hir::DotDotPos,
2007 expected: Ty<'tcx>,
2008 pat_info: PatInfo<'tcx>,
2009 ) -> Ty<'tcx> {
2010let tcx = self.tcx;
2011let mut expected_len = elements.len();
2012if ddpos.as_opt_usize().is_some() {
2013// Require known type only when `..` is present.
2014if let ty::Tuple(tys) = self.structurally_resolve_type(span, expected).kind() {
2015expected_len = tys.len();
2016 }
2017 }
2018let max_len = cmp::max(expected_len, elements.len());
20192020let element_tys_iter = (0..max_len).map(|_| self.next_ty_var(span));
2021let element_tys = tcx.mk_type_list_from_iter(element_tys_iter);
2022let pat_ty = Ty::new_tup(tcx, element_tys);
2023if let Err(reported) = self.demand_eqtype_pat(span, expected, pat_ty, &pat_info.top_info) {
2024// Walk subpatterns with an expected type of `err` in this case to silence
2025 // further errors being emitted when using the bindings. #50333
2026for (_, elem) in elements.iter().enumerate_and_adjust(max_len, ddpos) {
2027self.check_pat(elem, Ty::new_error(tcx, reported), pat_info);
2028 }
2029Ty::new_error(tcx, reported)
2030 } else {
2031for (i, elem) in elements.iter().enumerate_and_adjust(max_len, ddpos) {
2032self.check_pat(elem, element_tys[i], pat_info);
2033 }
2034pat_ty2035 }
2036 }
20372038fn check_struct_pat_fields(
2039&self,
2040 adt_ty: Ty<'tcx>,
2041 pat: &'tcx Pat<'tcx>,
2042 variant: &'tcx ty::VariantDef,
2043 fields: &'tcx [hir::PatField<'tcx>],
2044 has_rest_pat: bool,
2045 pat_info: PatInfo<'tcx>,
2046 ) -> Result<(), ErrorGuaranteed> {
2047let tcx = self.tcx;
20482049let ty::Adt(adt, args) = adt_ty.kind() else {
2050::rustc_middle::util::bug::span_bug_fmt(pat.span,
format_args!("struct pattern is not an ADT"));span_bug!(pat.span, "struct pattern is not an ADT");
2051 };
20522053// Index the struct fields' types.
2054let field_map = variant2055 .fields
2056 .iter_enumerated()
2057 .map(|(i, field)| (field.ident(self.tcx).normalize_to_macros_2_0(), (i, field)))
2058 .collect::<FxHashMap<_, _>>();
20592060// Keep track of which fields have already appeared in the pattern.
2061let mut used_fields = FxHashMap::default();
2062let mut result = Ok(());
20632064let mut inexistent_fields = ::alloc::vec::Vec::new()vec![];
2065// Typecheck each field.
2066for field in fields {
2067let span = field.span;
2068let ident = tcx.adjust_ident(field.ident, variant.def_id);
2069let field_ty = match used_fields.entry(ident) {
2070 Occupied(occupied) => {
2071let guar = self.error_field_already_bound(span, field.ident, *occupied.get());
2072 result = Err(guar);
2073 Ty::new_error(tcx, guar)
2074 }
2075 Vacant(vacant) => {
2076 vacant.insert(span);
2077 field_map
2078 .get(&ident)
2079 .map(|(i, f)| {
2080self.write_field_index(field.hir_id, *i);
2081self.tcx.check_stability(f.did, Some(field.hir_id), span, None);
2082self.field_ty(span, f, args)
2083 })
2084 .unwrap_or_else(|| {
2085 inexistent_fields.push(field);
2086 Ty::new_misc_error(tcx)
2087 })
2088 }
2089 };
20902091self.check_pat(field.pat, field_ty, pat_info);
2092 }
20932094let mut unmentioned_fields = variant2095 .fields
2096 .iter()
2097 .map(|field| (field, field.ident(self.tcx).normalize_to_macros_2_0()))
2098 .filter(|(_, ident)| !used_fields.contains_key(ident))
2099 .collect::<Vec<_>>();
21002101let inexistent_fields_err = if !inexistent_fields.is_empty()
2102 && !inexistent_fields.iter().any(|field| field.ident.name == kw::Underscore)
2103 {
2104// we don't care to report errors for a struct if the struct itself is tainted
2105variant.has_errors()?;
2106Some(self.error_inexistent_fields(
2107adt.variant_descr(),
2108&inexistent_fields,
2109&mut unmentioned_fields,
2110pat,
2111variant,
2112args,
2113 ))
2114 } else {
2115None2116 };
21172118// Require `..` if struct has non_exhaustive attribute.
2119let non_exhaustive = variant.field_list_has_applicable_non_exhaustive();
2120if non_exhaustive && !has_rest_pat {
2121self.error_foreign_non_exhaustive_spat(pat, adt.variant_descr(), fields.is_empty());
2122 }
21232124let mut unmentioned_err = None;
2125// Report an error if an incorrect number of fields was specified.
2126if adt.is_union() {
2127if fields.len() != 1 {
2128self.dcx().emit_err(errors::UnionPatMultipleFields { span: pat.span });
2129 }
2130if has_rest_pat {
2131self.dcx().emit_err(errors::UnionPatDotDot { span: pat.span });
2132 }
2133 } else if !unmentioned_fields.is_empty() {
2134let accessible_unmentioned_fields: Vec<_> = unmentioned_fields2135 .iter()
2136 .copied()
2137 .filter(|(field, _)| self.is_field_suggestable(field, pat.hir_id, pat.span))
2138 .collect();
21392140if !has_rest_pat {
2141if accessible_unmentioned_fields.is_empty() {
2142unmentioned_err = Some(self.error_no_accessible_fields(pat, fields));
2143 } else {
2144unmentioned_err = Some(self.error_unmentioned_fields(
2145pat,
2146&accessible_unmentioned_fields,
2147accessible_unmentioned_fields.len() != unmentioned_fields.len(),
2148fields,
2149 ));
2150 }
2151 } else if non_exhaustive && !accessible_unmentioned_fields.is_empty() {
2152self.lint_non_exhaustive_omitted_patterns(
2153pat,
2154&accessible_unmentioned_fields,
2155adt_ty,
2156 )
2157 }
2158 }
2159match (inexistent_fields_err, unmentioned_err) {
2160 (Some(i), Some(u)) => {
2161if let Err(e) = self.error_tuple_variant_as_struct_pat(pat, fields, variant) {
2162// We don't want to show the nonexistent fields error when this was
2163 // `Foo { a, b }` when it should have been `Foo(a, b)`.
2164i.delay_as_bug();
2165u.delay_as_bug();
2166Err(e)
2167 } else {
2168i.emit();
2169Err(u.emit())
2170 }
2171 }
2172 (None, Some(u)) => {
2173if let Err(e) = self.error_tuple_variant_as_struct_pat(pat, fields, variant) {
2174u.delay_as_bug();
2175Err(e)
2176 } else {
2177Err(u.emit())
2178 }
2179 }
2180 (Some(err), None) => Err(err.emit()),
2181 (None, None) => {
2182self.error_tuple_variant_index_shorthand(variant, pat, fields)?;
2183result2184 }
2185 }
2186 }
21872188fn error_tuple_variant_index_shorthand(
2189&self,
2190 variant: &VariantDef,
2191 pat: &'_ Pat<'_>,
2192 fields: &[hir::PatField<'_>],
2193 ) -> Result<(), ErrorGuaranteed> {
2194// if this is a tuple struct, then all field names will be numbers
2195 // so if any fields in a struct pattern use shorthand syntax, they will
2196 // be invalid identifiers (for example, Foo { 0, 1 }).
2197if let (Some(CtorKind::Fn), PatKind::Struct(qpath, field_patterns, ..)) =
2198 (variant.ctor_kind(), &pat.kind)
2199 {
2200let has_shorthand_field_name = field_patterns.iter().any(|field| field.is_shorthand);
2201if has_shorthand_field_name {
2202let path = rustc_hir_pretty::qpath_to_string(&self.tcx, qpath);
2203let mut err = {
self.dcx().struct_span_err(pat.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("tuple variant `{0}` written as struct variant",
path))
})).with_code(E0769)
}struct_span_code_err!(
2204self.dcx(),
2205 pat.span,
2206 E0769,
2207"tuple variant `{path}` written as struct variant",
2208 );
2209err.span_suggestion_verbose(
2210qpath.span().shrink_to_hi().to(pat.span.shrink_to_hi()),
2211"use the tuple variant pattern syntax instead",
2212::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})",
self.get_suggested_tuple_struct_pattern(fields, variant)))
})format!("({})", self.get_suggested_tuple_struct_pattern(fields, variant)),
2213 Applicability::MaybeIncorrect,
2214 );
2215return Err(err.emit());
2216 }
2217 }
2218Ok(())
2219 }
22202221fn error_foreign_non_exhaustive_spat(&self, pat: &Pat<'_>, descr: &str, no_fields: bool) {
2222let sess = self.tcx.sess;
2223let sm = sess.source_map();
2224let sp_brace = sm.end_point(pat.span);
2225let sp_comma = sm.end_point(pat.span.with_hi(sp_brace.hi()));
2226let sugg = if no_fields || sp_brace != sp_comma { ".. }" } else { ", .. }" };
22272228{
self.dcx().struct_span_err(pat.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`..` required with {0} marked as non-exhaustive",
descr))
})).with_code(E0638)
}struct_span_code_err!(
2229self.dcx(),
2230 pat.span,
2231 E0638,
2232"`..` required with {descr} marked as non-exhaustive",
2233 )2234 .with_span_suggestion_verbose(
2235sp_comma,
2236"add `..` at the end of the field list to ignore all other fields",
2237sugg,
2238 Applicability::MachineApplicable,
2239 )
2240 .emit();
2241 }
22422243fn error_field_already_bound(
2244&self,
2245 span: Span,
2246 ident: Ident,
2247 other_field: Span,
2248 ) -> ErrorGuaranteed {
2249{
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field `{0}` bound multiple times in the pattern",
ident))
})).with_code(E0025)
}struct_span_code_err!(
2250self.dcx(),
2251 span,
2252 E0025,
2253"field `{}` bound multiple times in the pattern",
2254 ident
2255 )2256 .with_span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("multiple uses of `{0}` in pattern",
ident))
})format!("multiple uses of `{ident}` in pattern"))
2257 .with_span_label(other_field, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("first use of `{0}`", ident))
})format!("first use of `{ident}`"))
2258 .emit()
2259 }
22602261fn error_inexistent_fields(
2262&self,
2263 kind_name: &str,
2264 inexistent_fields: &[&hir::PatField<'tcx>],
2265 unmentioned_fields: &mut Vec<(&'tcx ty::FieldDef, Ident)>,
2266 pat: &'tcx Pat<'tcx>,
2267 variant: &ty::VariantDef,
2268 args: ty::GenericArgsRef<'tcx>,
2269 ) -> Diag<'a> {
2270let tcx = self.tcx;
2271let (field_names, t, plural) = if let [field] = inexistent_fields {
2272 (::alloc::__export::must_use({
::alloc::fmt::format(format_args!("a field named `{0}`", field.ident))
})format!("a field named `{}`", field.ident), "this", "")
2273 } else {
2274 (
2275::alloc::__export::must_use({
::alloc::fmt::format(format_args!("fields named {0}",
inexistent_fields.iter().map(|field|
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", field.ident))
})).collect::<Vec<String>>().join(", ")))
})format!(
2276"fields named {}",
2277 inexistent_fields
2278 .iter()
2279 .map(|field| format!("`{}`", field.ident))
2280 .collect::<Vec<String>>()
2281 .join(", ")
2282 ),
2283"these",
2284"s",
2285 )
2286 };
2287let spans = inexistent_fields.iter().map(|field| field.ident.span).collect::<Vec<_>>();
2288let mut err = {
self.dcx().struct_span_err(spans,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} `{1}` does not have {2}",
kind_name, tcx.def_path_str(variant.def_id), field_names))
})).with_code(E0026)
}struct_span_code_err!(
2289self.dcx(),
2290 spans,
2291 E0026,
2292"{} `{}` does not have {}",
2293 kind_name,
2294 tcx.def_path_str(variant.def_id),
2295 field_names
2296 );
2297if let Some(pat_field) = inexistent_fields.last() {
2298err.span_label(
2299pat_field.ident.span,
2300::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} `{1}` does not have {2} field{3}",
kind_name, tcx.def_path_str(variant.def_id), t, plural))
})format!(
2301"{} `{}` does not have {} field{}",
2302 kind_name,
2303 tcx.def_path_str(variant.def_id),
2304 t,
2305 plural
2306 ),
2307 );
23082309if let [(field_def, field)] = unmentioned_fields.as_slice()
2310 && self.is_field_suggestable(field_def, pat.hir_id, pat.span)
2311 {
2312let suggested_name =
2313find_best_match_for_name(&[field.name], pat_field.ident.name, None);
2314if let Some(suggested_name) = suggested_name {
2315err.span_suggestion_verbose(
2316pat_field.ident.span,
2317"a field with a similar name exists",
2318suggested_name,
2319 Applicability::MaybeIncorrect,
2320 );
23212322// When we have a tuple struct used with struct we don't want to suggest using
2323 // the (valid) struct syntax with numeric field names. Instead we want to
2324 // suggest the expected syntax. We infer that this is the case by parsing the
2325 // `Ident` into an unsized integer. The suggestion will be emitted elsewhere in
2326 // `smart_resolve_context_dependent_help`.
2327if suggested_name.to_ident_string().parse::<usize>().is_err() {
2328// We don't want to throw `E0027` in case we have thrown `E0026` for them.
2329unmentioned_fields.retain(|&(_, x)| x.name != suggested_name);
2330 }
2331 } else if inexistent_fields.len() == 1 {
2332match pat_field.pat.kind {
2333 PatKind::Expr(_)
2334if !self.may_coerce(
2335self.typeck_results.borrow().node_type(pat_field.pat.hir_id),
2336self.field_ty(field.span, field_def, args),
2337 ) => {}
2338_ => {
2339err.span_suggestion_short(
2340pat_field.ident.span,
2341::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` has a field named `{1}`",
tcx.def_path_str(variant.def_id), field.name))
})format!(
2342"`{}` has a field named `{}`",
2343 tcx.def_path_str(variant.def_id),
2344 field.name,
2345 ),
2346field.name,
2347 Applicability::MaybeIncorrect,
2348 );
2349 }
2350 }
2351 }
2352 }
2353 }
2354if tcx.sess.teach(err.code.unwrap()) {
2355err.note(
2356"This error indicates that a struct pattern attempted to \
2357 extract a nonexistent field from a struct. Struct fields \
2358 are identified by the name used before the colon : so struct \
2359 patterns should resemble the declaration of the struct type \
2360 being matched.\n\n\
2361 If you are using shorthand field patterns but want to refer \
2362 to the struct field by a different name, you should rename \
2363 it explicitly.",
2364 );
2365 }
2366err2367 }
23682369fn error_tuple_variant_as_struct_pat(
2370&self,
2371 pat: &Pat<'_>,
2372 fields: &'tcx [hir::PatField<'tcx>],
2373 variant: &ty::VariantDef,
2374 ) -> Result<(), ErrorGuaranteed> {
2375if let (Some(CtorKind::Fn), PatKind::Struct(qpath, pattern_fields, ..)) =
2376 (variant.ctor_kind(), &pat.kind)
2377 {
2378let is_tuple_struct_match = !pattern_fields.is_empty()
2379 && pattern_fields.iter().map(|field| field.ident.name.as_str()).all(is_number);
2380if is_tuple_struct_match {
2381return Ok(());
2382 }
23832384// we don't care to report errors for a struct if the struct itself is tainted
2385variant.has_errors()?;
23862387let path = rustc_hir_pretty::qpath_to_string(&self.tcx, qpath);
2388let mut err = {
self.dcx().struct_span_err(pat.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("tuple variant `{0}` written as struct variant",
path))
})).with_code(E0769)
}struct_span_code_err!(
2389self.dcx(),
2390 pat.span,
2391 E0769,
2392"tuple variant `{}` written as struct variant",
2393 path
2394 );
2395let (sugg, appl) = if fields.len() == variant.fields.len() {
2396 (
2397self.get_suggested_tuple_struct_pattern(fields, variant),
2398 Applicability::MachineApplicable,
2399 )
2400 } else {
2401 (
2402variant.fields.iter().map(|_| "_").collect::<Vec<&str>>().join(", "),
2403 Applicability::MaybeIncorrect,
2404 )
2405 };
2406err.span_suggestion_verbose(
2407qpath.span().shrink_to_hi().to(pat.span.shrink_to_hi()),
2408"use the tuple variant pattern syntax instead",
2409::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})", sugg))
})format!("({sugg})"),
2410appl,
2411 );
2412return Err(err.emit());
2413 }
2414Ok(())
2415 }
24162417fn get_suggested_tuple_struct_pattern(
2418&self,
2419 fields: &[hir::PatField<'_>],
2420 variant: &VariantDef,
2421 ) -> String {
2422let variant_field_idents =
2423variant.fields.iter().map(|f| f.ident(self.tcx)).collect::<Vec<Ident>>();
2424fields2425 .iter()
2426 .map(|field| {
2427match self.tcx.sess.source_map().span_to_snippet(field.pat.span) {
2428Ok(f) => {
2429// Field names are numbers, but numbers
2430 // are not valid identifiers
2431if variant_field_idents.contains(&field.ident) {
2432String::from("_")
2433 } else {
2434f2435 }
2436 }
2437Err(_) => rustc_hir_pretty::pat_to_string(&self.tcx, field.pat),
2438 }
2439 })
2440 .collect::<Vec<String>>()
2441 .join(", ")
2442 }
24432444/// Returns a diagnostic reporting a struct pattern which is missing an `..` due to
2445 /// inaccessible fields.
2446 ///
2447 /// ```text
2448 /// error: pattern requires `..` due to inaccessible fields
2449 /// --> src/main.rs:10:9
2450 /// |
2451 /// LL | let foo::Foo {} = foo::Foo::default();
2452 /// | ^^^^^^^^^^^
2453 /// |
2454 /// help: add a `..`
2455 /// |
2456 /// LL | let foo::Foo { .. } = foo::Foo::default();
2457 /// | ^^^^^^
2458 /// ```
2459fn error_no_accessible_fields(
2460&self,
2461 pat: &Pat<'_>,
2462 fields: &'tcx [hir::PatField<'tcx>],
2463 ) -> Diag<'a> {
2464let mut err = self2465 .dcx()
2466 .struct_span_err(pat.span, "pattern requires `..` due to inaccessible fields");
24672468if let Some(field) = fields.last() {
2469let tail_span = field.span.shrink_to_hi().to(pat.span.shrink_to_hi());
2470let comma_hi_offset =
2471self.tcx.sess.source_map().span_to_snippet(tail_span).ok().and_then(|snippet| {
2472let trimmed = snippet.trim_start();
2473trimmed.starts_with(',').then(|| (snippet.len() - trimmed.len() + 1) as u32)
2474 });
2475err.span_suggestion_verbose(
2476if let Some(comma_hi_offset) = comma_hi_offset {
2477tail_span.with_hi(tail_span.lo() + BytePos(comma_hi_offset)).shrink_to_hi()
2478 } else {
2479field.span.shrink_to_hi()
2480 },
2481"ignore the inaccessible and unused fields",
2482if comma_hi_offset.is_some() { " .." } else { ", .." },
2483 Applicability::MachineApplicable,
2484 );
2485 } else {
2486let qpath_span = if let PatKind::Struct(qpath, ..) = &pat.kind {
2487qpath.span()
2488 } else {
2489::rustc_middle::util::bug::bug_fmt(format_args!("`error_no_accessible_fields` called on non-struct pattern"));bug!("`error_no_accessible_fields` called on non-struct pattern");
2490 };
24912492// Shrink the span to exclude the `foo:Foo` in `foo::Foo { }`.
2493let span = pat.span.with_lo(qpath_span.shrink_to_hi().hi());
2494err.span_suggestion_verbose(
2495span,
2496"ignore the inaccessible and unused fields",
2497" { .. }",
2498 Applicability::MachineApplicable,
2499 );
2500 }
2501err2502 }
25032504/// Report that a pattern for a `#[non_exhaustive]` struct marked with `non_exhaustive_omitted_patterns`
2505 /// is not exhaustive enough.
2506 ///
2507 /// Nb: the partner lint for enums lives in `compiler/rustc_mir_build/src/thir/pattern/usefulness.rs`.
2508fn lint_non_exhaustive_omitted_patterns(
2509&self,
2510 pat: &Pat<'_>,
2511 unmentioned_fields: &[(&ty::FieldDef, Ident)],
2512 ty: Ty<'tcx>,
2513 ) {
2514struct FieldsNotListed<'a, 'b, 'tcx> {
2515 pat_span: Span,
2516 unmentioned_fields: &'a [(&'b ty::FieldDef, Ident)],
2517 joined_patterns: String,
2518 ty: Ty<'tcx>,
2519 }
25202521impl<'a, 'b, 'c, 'tcx> Diagnostic<'a, ()> for FieldsNotListed<'b, 'c, 'tcx> {
2522fn into_diag(self, dcx: DiagCtxtHandle<'a>, level: Level) -> Diag<'a, ()> {
2523let Self { pat_span, unmentioned_fields, joined_patterns, ty } = self;
2524Diag::new(dcx, level, "some fields are not explicitly listed")
2525 .with_span_label(pat_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field{0} {1} not listed",
if unmentioned_fields.len() == 1 { "" } else { "s" },
joined_patterns))
})format!("field{} {} not listed", rustc_errors::pluralize!(unmentioned_fields.len()), joined_patterns))
2526 .with_help(
2527"ensure that all fields are mentioned explicitly by adding the suggested fields",
2528 )
2529 .with_note(::alloc::__export::must_use({
::alloc::fmt::format(format_args!("the pattern is of type `{0}` and the `non_exhaustive_omitted_patterns` attribute was found",
ty))
})format!(
2530"the pattern is of type `{ty}` and the `non_exhaustive_omitted_patterns` attribute was found",
2531 ))
2532 }
2533 }
25342535fn joined_uncovered_patterns(witnesses: &[&Ident]) -> String {
2536const LIMIT: usize = 3;
2537match witnesses {
2538 [] => {
2539{
::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
format_args!("expected an uncovered pattern, otherwise why are we emitting an error?")));
}unreachable!(
2540"expected an uncovered pattern, otherwise why are we emitting an error?"
2541)2542 }
2543 [witness] => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", witness))
})format!("`{witness}`"),
2544 [head @ .., tail] if head.len() < LIMIT => {
2545let head: Vec<_> = head.iter().map(<_>::to_string).collect();
2546::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and `{1}`",
head.join("`, `"), tail))
})format!("`{}` and `{}`", head.join("`, `"), tail)2547 }
2548_ => {
2549let (head, tail) = witnesses.split_at(LIMIT);
2550let head: Vec<_> = head.iter().map(<_>::to_string).collect();
2551::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and {1} more",
head.join("`, `"), tail.len()))
})format!("`{}` and {} more", head.join("`, `"), tail.len())2552 }
2553 }
2554 }
2555let joined_patterns = joined_uncovered_patterns(
2556&unmentioned_fields.iter().map(|(_, i)| i).collect::<Vec<_>>(),
2557 );
25582559self.tcx.emit_node_span_lint(
2560NON_EXHAUSTIVE_OMITTED_PATTERNS,
2561pat.hir_id,
2562pat.span,
2563FieldsNotListed { pat_span: pat.span, unmentioned_fields, joined_patterns, ty },
2564 );
2565 }
25662567/// Returns a diagnostic reporting a struct pattern which does not mention some fields.
2568 ///
2569 /// ```text
2570 /// error[E0027]: pattern does not mention field `bar`
2571 /// --> src/main.rs:15:9
2572 /// |
2573 /// LL | let foo::Foo {} = foo::Foo::new();
2574 /// | ^^^^^^^^^^^ missing field `bar`
2575 /// ```
2576fn error_unmentioned_fields(
2577&self,
2578 pat: &Pat<'_>,
2579 unmentioned_fields: &[(&ty::FieldDef, Ident)],
2580 have_inaccessible_fields: bool,
2581 fields: &'tcx [hir::PatField<'tcx>],
2582 ) -> Diag<'a> {
2583let inaccessible = if have_inaccessible_fields { " and inaccessible fields" } else { "" };
2584let field_names = if let [(_, field)] = unmentioned_fields {
2585::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field `{0}`{1}", field,
inaccessible))
})format!("field `{field}`{inaccessible}")2586 } else {
2587let fields = unmentioned_fields2588 .iter()
2589 .map(|(_, name)| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", name))
})format!("`{name}`"))
2590 .collect::<Vec<String>>()
2591 .join(", ");
2592::alloc::__export::must_use({
::alloc::fmt::format(format_args!("fields {0}{1}", fields,
inaccessible))
})format!("fields {fields}{inaccessible}")2593 };
2594let mut err = {
self.dcx().struct_span_err(pat.span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("pattern does not mention {0}",
field_names))
})).with_code(E0027)
}struct_span_code_err!(
2595self.dcx(),
2596 pat.span,
2597 E0027,
2598"pattern does not mention {}",
2599 field_names
2600 );
2601err.span_label(pat.span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("missing {0}", field_names))
})format!("missing {field_names}"));
2602let len = unmentioned_fields.len();
2603let (prefix, postfix, sp) = match fields {
2604 [] => match &pat.kind {
2605 PatKind::Struct(path, [], None) => {
2606 (" { ", " }", path.span().shrink_to_hi().until(pat.span.shrink_to_hi()))
2607 }
2608_ => return err,
2609 },
2610 [.., field] => {
2611// Account for last field having a trailing comma or parse recovery at the tail of
2612 // the pattern to avoid invalid suggestion (#78511).
2613let tail = field.span.shrink_to_hi().with_hi(pat.span.hi());
2614match &pat.kind {
2615 PatKind::Struct(..) => (", ", " }", tail),
2616_ => return err,
2617 }
2618 }
2619 };
2620err.span_suggestion(
2621sp,
2622::alloc::__export::must_use({
::alloc::fmt::format(format_args!("include the missing field{0} in the pattern{1}",
if len == 1 { "" } else { "s" },
if have_inaccessible_fields {
" and ignore the inaccessible fields"
} else { "" }))
})format!(
2623"include the missing field{} in the pattern{}",
2624pluralize!(len),
2625if have_inaccessible_fields { " and ignore the inaccessible fields" } else { "" }
2626 ),
2627::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1}{2}{3}", prefix,
unmentioned_fields.iter().map(|(_, name)|
{
let field_name = name.to_string();
if is_number(&field_name) {
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: _", field_name))
})
} else { field_name }
}).collect::<Vec<_>>().join(", "),
if have_inaccessible_fields { ", .." } else { "" }, postfix))
})format!(
2628"{}{}{}{}",
2629 prefix,
2630 unmentioned_fields
2631 .iter()
2632 .map(|(_, name)| {
2633let field_name = name.to_string();
2634if is_number(&field_name) { format!("{field_name}: _") } else { field_name }
2635 })
2636 .collect::<Vec<_>>()
2637 .join(", "),
2638if have_inaccessible_fields { ", .." } else { "" },
2639 postfix,
2640 ),
2641 Applicability::MachineApplicable,
2642 );
2643err.span_suggestion(
2644sp,
2645::alloc::__export::must_use({
::alloc::fmt::format(format_args!("if you don\'t care about {0} missing field{1}, you can explicitly ignore {2}",
if len == 1 { "this" } else { "these" },
if len == 1 { "" } else { "s" },
if len == 1 { "it" } else { "them" }))
})format!(
2646"if you don't care about {these} missing field{s}, you can explicitly ignore {them}",
2647 these = pluralize!("this", len),
2648 s = pluralize!(len),
2649 them = if len == 1 { "it" } else { "them" },
2650 ),
2651::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}{1}{2}{3}", prefix,
unmentioned_fields.iter().map(|(_, name)|
{
let field_name = name.to_string();
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: _", field_name))
})
}).collect::<Vec<_>>().join(", "),
if have_inaccessible_fields { ", .." } else { "" }, postfix))
})format!(
2652"{}{}{}{}",
2653 prefix,
2654 unmentioned_fields
2655 .iter()
2656 .map(|(_, name)| {
2657let field_name = name.to_string();
2658format!("{field_name}: _")
2659 })
2660 .collect::<Vec<_>>()
2661 .join(", "),
2662if have_inaccessible_fields { ", .." } else { "" },
2663 postfix,
2664 ),
2665 Applicability::MachineApplicable,
2666 );
2667err.span_suggestion(
2668sp,
2669"or always ignore missing fields here",
2670::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}..{1}", prefix, postfix))
})format!("{prefix}..{postfix}"),
2671 Applicability::MachineApplicable,
2672 );
2673err2674 }
26752676fn check_pat_box(
2677&self,
2678 span: Span,
2679 inner: &'tcx Pat<'tcx>,
2680 expected: Ty<'tcx>,
2681 pat_info: PatInfo<'tcx>,
2682 ) -> Ty<'tcx> {
2683let tcx = self.tcx;
2684let (box_ty, inner_ty) = self2685 .check_dereferenceable(span, expected, inner)
2686 .and_then(|()| {
2687// Here, `demand::subtype` is good enough, but I don't
2688 // think any errors can be introduced by using `demand::eqtype`.
2689let inner_ty = self.next_ty_var(inner.span);
2690let box_ty = Ty::new_box(tcx, inner_ty);
2691self.demand_eqtype_pat(span, expected, box_ty, &pat_info.top_info)?;
2692Ok((box_ty, inner_ty))
2693 })
2694 .unwrap_or_else(|guar| {
2695let err = Ty::new_error(tcx, guar);
2696 (err, err)
2697 });
2698self.check_pat(inner, inner_ty, pat_info);
2699box_ty2700 }
27012702fn check_pat_deref(
2703&self,
2704 span: Span,
2705 inner: &'tcx Pat<'tcx>,
2706 expected: Ty<'tcx>,
2707 pat_info: PatInfo<'tcx>,
2708 ) -> Ty<'tcx> {
2709let target_ty = self.deref_pat_target(span, expected);
2710self.check_pat(inner, target_ty, pat_info);
2711self.register_deref_mut_bounds_if_needed(span, inner, [expected]);
2712expected2713 }
27142715fn deref_pat_target(&self, span: Span, source_ty: Ty<'tcx>) -> Ty<'tcx> {
2716// Register a `DerefPure` bound, which is required by all `deref!()` pats.
2717let tcx = self.tcx;
2718self.register_bound(
2719source_ty,
2720tcx.require_lang_item(hir::LangItem::DerefPure, span),
2721self.misc(span),
2722 );
2723// The expected type for the deref pat's inner pattern is `<expected as Deref>::Target`.
2724let target_ty = Ty::new_projection(
2725tcx,
2726tcx.require_lang_item(hir::LangItem::DerefTarget, span),
2727 [source_ty],
2728 );
2729let target_ty = self.normalize(span, Unnormalized::new_wip(target_ty));
2730self.resolve_vars_with_obligations(target_ty)
2731 }
27322733/// Check if the interior of a deref pattern (either explicit or implicit) has any `ref mut`
2734 /// bindings, which would require `DerefMut` to be emitted in MIR building instead of just
2735 /// `Deref`. We do this *after* checking the inner pattern, since we want to make sure to
2736 /// account for `ref mut` binding modes inherited from implicitly dereferencing `&mut` refs.
2737fn register_deref_mut_bounds_if_needed(
2738&self,
2739 span: Span,
2740 inner: &'tcx Pat<'tcx>,
2741 derefed_tys: impl IntoIterator<Item = Ty<'tcx>>,
2742 ) {
2743if self.typeck_results.borrow().pat_has_ref_mut_binding(inner) {
2744for mutably_derefed_ty in derefed_tys {
2745self.register_bound(
2746 mutably_derefed_ty,
2747self.tcx.require_lang_item(hir::LangItem::DerefMut, span),
2748self.misc(span),
2749 );
2750 }
2751 }
2752 }
27532754// Precondition: Pat is Ref(inner)
2755fn check_pat_ref(
2756&self,
2757 pat: &'tcx Pat<'tcx>,
2758 inner: &'tcx Pat<'tcx>,
2759 pat_pinned: Pinnedness,
2760 pat_mutbl: Mutability,
2761mut expected: Ty<'tcx>,
2762mut pat_info: PatInfo<'tcx>,
2763 ) -> Ty<'tcx> {
2764let tcx = self.tcx;
27652766let pat_prefix_span =
2767inner.span.find_ancestor_inside(pat.span).map(|end| pat.span.until(end));
27682769let ref_pat_matches_mut_ref = self.ref_pat_matches_mut_ref();
2770if ref_pat_matches_mut_ref && pat_mutbl == Mutability::Not {
2771// If `&` patterns can match against mutable reference types (RFC 3627, Rule 5), we need
2772 // to prevent subpatterns from binding with `ref mut`. Subpatterns of a shared reference
2773 // pattern should have read-only access to the scrutinee, and the borrow checker won't
2774 // catch it in this case.
2775pat_info.max_ref_mutbl = pat_info.max_ref_mutbl.cap_to_weakly_not(pat_prefix_span);
2776 }
27772778expected = self.resolve_vars_with_obligations(expected);
2779// Determine whether we're consuming an inherited reference and resetting the default
2780 // binding mode, based on edition and enabled experimental features.
2781if let ByRef::Yes(inh_pin, inh_mut) = pat_info.binding_mode
2782 && pat_pinned == inh_pin2783 {
2784match self.ref_pat_matches_inherited_ref(pat.span.edition()) {
2785 InheritedRefMatchRule::EatOuter => {
2786// ref pattern attempts to consume inherited reference
2787if pat_mutbl > inh_mut {
2788// Tried to match inherited `ref` with `&mut`
2789 // NB: This assumes that `&` patterns can match against mutable references
2790 // (RFC 3627, Rule 5). If we implement a pattern typing ruleset with Rule 4E
2791 // but not Rule 5, we'll need to check that here.
2792if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2793self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2794 }
27952796pat_info.binding_mode = ByRef::No;
2797self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2798self.check_pat(inner, expected, pat_info);
2799return expected;
2800 }
2801 InheritedRefMatchRule::EatInner => {
2802if let ty::Ref(_, _, r_mutbl) = *expected.kind()
2803 && pat_mutbl <= r_mutbl2804 {
2805// Match against the reference type; don't consume the inherited ref.
2806 // NB: The check for compatible pattern and ref type mutability assumes that
2807 // `&` patterns can match against mutable references (RFC 3627, Rule 5). If
2808 // we implement a pattern typing ruleset with Rule 4 (including the fallback
2809 // to matching the inherited ref when the inner ref can't match) but not
2810 // Rule 5, we'll need to check that here.
2811if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2812// NB: For RFC 3627's Rule 3, we limit the default binding mode's ref
2813 // mutability to `pat_info.max_ref_mutbl`. If we implement a pattern typing
2814 // ruleset with Rule 4 but not Rule 3, we'll need to check that here.
2815if true {
if !self.downgrade_mut_inside_shared() {
::core::panicking::panic("assertion failed: self.downgrade_mut_inside_shared()")
};
};debug_assert!(self.downgrade_mut_inside_shared());
2816let mutbl_cap = cmp::min(r_mutbl, pat_info.max_ref_mutbl.as_mutbl());
2817pat_info.binding_mode = pat_info.binding_mode.cap_ref_mutability(mutbl_cap);
2818 } else {
2819// The reference pattern can't match against the expected type, so try
2820 // matching against the inherited ref instead.
2821if pat_mutbl > inh_mut {
2822// We can't match an inherited shared reference with `&mut`.
2823 // NB: This assumes that `&` patterns can match against mutable
2824 // references (RFC 3627, Rule 5). If we implement a pattern typing
2825 // ruleset with Rule 4 but not Rule 5, we'll need to check that here.
2826 // FIXME(ref_pat_eat_one_layer_2024_structural): If we already tried
2827 // matching the real reference, the error message should explain that
2828 // falling back to the inherited reference didn't work. This should be
2829 // the same error as the old-Edition version below.
2830if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2831self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2832 }
28332834pat_info.binding_mode = ByRef::No;
2835self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2836self.check_pat(inner, expected, pat_info);
2837return expected;
2838 }
2839 }
2840 InheritedRefMatchRule::EatBoth { consider_inherited_ref: true } => {
2841// Reset binding mode on old editions
2842pat_info.binding_mode = ByRef::No;
28432844if let ty::Ref(_, inner_ty, _) = *expected.kind() {
2845// Consume both the inherited and inner references.
2846if pat_mutbl.is_mut() && inh_mut.is_mut() {
2847// As a special case, a `&mut` reference pattern will be able to match
2848 // against a reference type of any mutability if the inherited ref is
2849 // mutable. Since this allows us to match against a shared reference
2850 // type, we refer to this as "falling back" to matching the inherited
2851 // reference, though we consume the real reference as well. We handle
2852 // this here to avoid adding this case to the common logic below.
2853self.check_pat(inner, inner_ty, pat_info);
2854return expected;
2855 } else {
2856// Otherwise, use the common logic below for matching the inner
2857 // reference type.
2858 // FIXME(ref_pat_eat_one_layer_2024_structural): If this results in a
2859 // mutability mismatch, the error message should explain that falling
2860 // back to the inherited reference didn't work. This should be the same
2861 // error as the Edition 2024 version above.
2862}
2863 } else {
2864// The expected type isn't a reference type, so only match against the
2865 // inherited reference.
2866if pat_mutbl > inh_mut {
2867// We can't match a lone inherited shared reference with `&mut`.
2868self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2869 }
28702871self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2872self.check_pat(inner, expected, pat_info);
2873return expected;
2874 }
2875 }
2876 InheritedRefMatchRule::EatBoth { consider_inherited_ref: false } => {
2877// Reset binding mode on stable Rust. This will be a type error below if
2878 // `expected` is not a reference type.
2879pat_info.binding_mode = ByRef::No;
2880self.add_rust_2024_migration_desugared_pat(
2881pat_info.top_info.hir_id,
2882pat,
2883match pat_mutbl {
2884 Mutability::Not => '&', // last char of `&`
2885Mutability::Mut => 't', // last char of `&mut`
2886},
2887inh_mut,
2888 )
2889 }
2890 }
2891 }
28922893let (ref_ty, inner_ty) = match self.check_dereferenceable(pat.span, expected, inner) {
2894Ok(()) => {
2895// `demand::subtype` would be good enough, but using `eqtype` turns
2896 // out to be equally general. See (note_1) for details.
28972898 // Take region, inner-type from expected type if we can,
2899 // to avoid creating needless variables. This also helps with
2900 // the bad interactions of the given hack detailed in (note_1).
2901{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:2901",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(2901u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("check_pat_ref: expected={0:?}",
expected) as &dyn Value))])
});
} else { ; }
};debug!("check_pat_ref: expected={:?}", expected);
2902match expected.maybe_pinned_ref() {
2903Some((r_ty, r_pinned, r_mutbl, _))
2904if ((ref_pat_matches_mut_ref && r_mutbl >= pat_mutbl)
2905 || r_mutbl == pat_mutbl)
2906 && pat_pinned == r_pinned =>
2907 {
2908if r_mutbl == Mutability::Not {
2909pat_info.max_ref_mutbl = MutblCap::Not;
2910 }
2911if r_pinned == Pinnedness::Pinned {
2912pat_info.max_pinnedness = PinnednessCap::Pinned;
2913 }
29142915 (expected, r_ty)
2916 }
2917_ => {
2918let inner_ty = self.next_ty_var(inner.span);
2919let ref_ty = self.new_ref_ty(pat.span, pat_pinned, pat_mutbl, inner_ty);
2920{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:2920",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(2920u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&format_args!("check_pat_ref: demanding {0:?} = {1:?}",
expected, ref_ty) as &dyn Value))])
});
} else { ; }
};debug!("check_pat_ref: demanding {:?} = {:?}", expected, ref_ty);
2921let err = self.demand_eqtype_pat_diag(
2922pat.span,
2923expected,
2924ref_ty,
2925&pat_info.top_info,
2926 );
29272928// Look for a case like `fn foo(&foo: u32)` and suggest
2929 // `fn foo(foo: &u32)`
2930if let Err(mut err) = err {
2931self.borrow_pat_suggestion(&mut err, pat);
2932err.emit();
2933 }
2934 (ref_ty, inner_ty)
2935 }
2936 }
2937 }
2938Err(guar) => {
2939let err = Ty::new_error(tcx, guar);
2940 (err, err)
2941 }
2942 };
29432944self.check_pat(inner, inner_ty, pat_info);
2945ref_ty2946 }
29472948/// Create a reference or pinned reference type with a fresh region variable.
2949fn new_ref_ty(
2950&self,
2951 span: Span,
2952 pinnedness: Pinnedness,
2953 mutbl: Mutability,
2954 ty: Ty<'tcx>,
2955 ) -> Ty<'tcx> {
2956let region = self.next_region_var(RegionVariableOrigin::PatternRegion(span));
2957let ref_ty = Ty::new_ref(self.tcx, region, ty, mutbl);
2958if pinnedness.is_pinned() {
2959return self.new_pinned_ty(span, ref_ty);
2960 }
2961ref_ty2962 }
29632964/// Create a pinned type.
2965fn new_pinned_ty(&self, span: Span, ty: Ty<'tcx>) -> Ty<'tcx> {
2966Ty::new_adt(
2967self.tcx,
2968self.tcx.adt_def(self.tcx.require_lang_item(LangItem::Pin, span)),
2969self.tcx.mk_args(&[ty.into()]),
2970 )
2971 }
29722973fn error_inherited_ref_mutability_mismatch(
2974&self,
2975 pat: &'tcx Pat<'tcx>,
2976 pat_prefix_span: Option<Span>,
2977 ) -> ErrorGuaranteed {
2978let err_msg = "mismatched types";
2979let err = if let Some(span) = pat_prefix_span {
2980let mut err = self.dcx().struct_span_err(span, err_msg);
2981err.code(E0308);
2982err.note("cannot match inherited `&` with `&mut` pattern");
2983err.span_suggestion_verbose(
2984span,
2985"replace this `&mut` pattern with `&`",
2986"&",
2987 Applicability::MachineApplicable,
2988 );
2989err2990 } else {
2991self.dcx().struct_span_err(pat.span, err_msg)
2992 };
2993err.emit()
2994 }
29952996fn try_resolve_slice_ty_to_array_ty(
2997&self,
2998 before: &'tcx [Pat<'tcx>],
2999 slice: Option<&'tcx Pat<'tcx>>,
3000 span: Span,
3001 ) -> Option<Ty<'tcx>> {
3002if slice.is_some() {
3003return None;
3004 }
30053006let tcx = self.tcx;
3007let len = before.len();
3008let inner_ty = self.next_ty_var(span);
30093010Some(Ty::new_array(tcx, inner_ty, len.try_into().unwrap()))
3011 }
30123013/// Used to determines whether we can infer the expected type in the slice pattern to be of type array.
3014 /// This is only possible if we're in an irrefutable pattern. If we were to allow this in refutable
3015 /// patterns we wouldn't e.g. report ambiguity in the following situation:
3016 ///
3017 /// ```ignore(rust)
3018 /// struct Zeroes;
3019 /// const ARR: [usize; 2] = [0; 2];
3020 /// const ARR2: [usize; 2] = [2; 2];
3021 ///
3022 /// impl Into<&'static [usize; 2]> for Zeroes {
3023 /// fn into(self) -> &'static [usize; 2] {
3024 /// &ARR
3025 /// }
3026 /// }
3027 ///
3028 /// impl Into<&'static [usize]> for Zeroes {
3029 /// fn into(self) -> &'static [usize] {
3030 /// &ARR2
3031 /// }
3032 /// }
3033 ///
3034 /// fn main() {
3035 /// let &[a, b]: &[usize] = Zeroes.into() else {
3036 /// ..
3037 /// };
3038 /// }
3039 /// ```
3040 ///
3041 /// If we're in an irrefutable pattern we prefer the array impl candidate given that
3042 /// the slice impl candidate would be rejected anyway (if no ambiguity existed).
3043fn pat_is_irrefutable(&self, decl_origin: Option<DeclOrigin<'_>>) -> bool {
3044match decl_origin {
3045Some(DeclOrigin::LocalDecl { els: None }) => true,
3046Some(DeclOrigin::LocalDecl { els: Some(_) } | DeclOrigin::LetExpr) | None => false,
3047 }
3048 }
30493050/// Type check a slice pattern.
3051 ///
3052 /// Syntactically, these look like `[pat_0, ..., pat_n]`.
3053 /// Semantically, we are type checking a pattern with structure:
3054 /// ```ignore (not-rust)
3055 /// [before_0, ..., before_n, (slice, after_0, ... after_n)?]
3056 /// ```
3057 /// The type of `slice`, if it is present, depends on the `expected` type.
3058 /// If `slice` is missing, then so is `after_i`.
3059 /// If `slice` is present, it can still represent 0 elements.
3060fn check_pat_slice(
3061&self,
3062 span: Span,
3063 before: &'tcx [Pat<'tcx>],
3064 slice: Option<&'tcx Pat<'tcx>>,
3065 after: &'tcx [Pat<'tcx>],
3066 expected: Ty<'tcx>,
3067 pat_info: PatInfo<'tcx>,
3068 ) -> Ty<'tcx> {
3069let expected = self.resolve_vars_with_obligations(expected);
30703071// If the pattern is irrefutable and `expected` is an infer ty, we try to equate it
3072 // to an array if the given pattern allows it. See issue #76342
3073if self.pat_is_irrefutable(pat_info.decl_origin) && expected.is_ty_var() {
3074if let Some(resolved_arr_ty) =
3075self.try_resolve_slice_ty_to_array_ty(before, slice, span)
3076 {
3077{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:3077",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(3077u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["resolved_arr_ty"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&resolved_arr_ty)
as &dyn Value))])
});
} else { ; }
};debug!(?resolved_arr_ty);
3078let _ = self.demand_eqtype(span, expected, resolved_arr_ty);
3079 }
3080 }
30813082let expected = self.structurally_resolve_type(span, expected);
3083{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_hir_typeck/src/pat.rs:3083",
"rustc_hir_typeck::pat", ::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_hir_typeck/src/pat.rs"),
::tracing_core::__macro_support::Option::Some(3083u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["expected"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&expected)
as &dyn Value))])
});
} else { ; }
};debug!(?expected);
30843085let (element_ty, opt_slice_ty, inferred) = match *expected.kind() {
3086// An array, so we might have something like `let [a, b, c] = [0, 1, 2];`.
3087ty::Array(element_ty, len) => {
3088let min = before.len() as u64 + after.len() as u64;
3089let (opt_slice_ty, expected) =
3090self.check_array_pat_len(span, element_ty, expected, slice, len, min);
3091// `opt_slice_ty.is_none()` => `slice.is_none()`.
3092 // Note, though, that opt_slice_ty could be `Some(error_ty)`.
3093if !(opt_slice_ty.is_some() || slice.is_none()) {
::core::panicking::panic("assertion failed: opt_slice_ty.is_some() || slice.is_none()")
};assert!(opt_slice_ty.is_some() || slice.is_none());
3094 (element_ty, opt_slice_ty, expected)
3095 }
3096 ty::Slice(element_ty) => (element_ty, Some(expected), expected),
3097// The expected type must be an array or slice, but was neither, so error.
3098_ => {
3099let guar = expected.error_reported().err().unwrap_or_else(|| {
3100self.error_expected_array_or_slice(span, expected, pat_info)
3101 });
3102let err = Ty::new_error(self.tcx, guar);
3103 (err, Some(err), err)
3104 }
3105 };
31063107// Type check all the patterns before `slice`.
3108for elt in before {
3109self.check_pat(elt, element_ty, pat_info);
3110 }
3111// Type check the `slice`, if present, against its expected type.
3112if let Some(slice) = slice {
3113self.check_pat(slice, opt_slice_ty.unwrap(), pat_info);
3114 }
3115// Type check the elements after `slice`, if present.
3116for elt in after {
3117self.check_pat(elt, element_ty, pat_info);
3118 }
3119inferred3120 }
31213122/// Type check the length of an array pattern.
3123 ///
3124 /// Returns both the type of the variable length pattern (or `None`), and the potentially
3125 /// inferred array type. We only return `None` for the slice type if `slice.is_none()`.
3126fn check_array_pat_len(
3127&self,
3128 span: Span,
3129 element_ty: Ty<'tcx>,
3130 arr_ty: Ty<'tcx>,
3131 slice: Option<&'tcx Pat<'tcx>>,
3132 len: ty::Const<'tcx>,
3133 min_len: u64,
3134 ) -> (Option<Ty<'tcx>>, Ty<'tcx>) {
3135let len = self.try_structurally_resolve_const(span, len).try_to_target_usize(self.tcx);
31363137let guar = if let Some(len) = len {
3138// Now we know the length...
3139if slice.is_none() {
3140// ...and since there is no variable-length pattern,
3141 // we require an exact match between the number of elements
3142 // in the array pattern and as provided by the matched type.
3143if min_len == len {
3144return (None, arr_ty);
3145 }
31463147self.error_scrutinee_inconsistent_length(span, min_len, len)
3148 } else if let Some(pat_len) = len.checked_sub(min_len) {
3149// The variable-length pattern was there,
3150 // so it has an array type with the remaining elements left as its size...
3151return (Some(Ty::new_array(self.tcx, element_ty, pat_len)), arr_ty);
3152 } else {
3153// ...however, in this case, there were no remaining elements.
3154 // That is, the slice pattern requires more than the array type offers.
3155self.error_scrutinee_with_rest_inconsistent_length(span, min_len, len)
3156 }
3157 } else if slice.is_none() {
3158// We have a pattern with a fixed length,
3159 // which we can use to infer the length of the array.
3160let updated_arr_ty = Ty::new_array(self.tcx, element_ty, min_len);
3161self.demand_eqtype(span, updated_arr_ty, arr_ty);
3162return (None, updated_arr_ty);
3163 } else {
3164// We have a variable-length pattern and don't know the array length.
3165 // This happens if we have e.g.,
3166 // `let [a, b, ..] = arr` where `arr: [T; N]` where `const N: usize`.
3167self.error_scrutinee_unfixed_length(span)
3168 };
31693170// If we get here, we must have emitted an error.
3171(Some(Ty::new_error(self.tcx, guar)), arr_ty)
3172 }
31733174fn error_scrutinee_inconsistent_length(
3175&self,
3176 span: Span,
3177 min_len: u64,
3178 size: u64,
3179 ) -> ErrorGuaranteed {
3180{
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("pattern requires {0} element{1} but array has {2}",
min_len, if min_len == 1 { "" } else { "s" }, size))
})).with_code(E0527)
}struct_span_code_err!(
3181self.dcx(),
3182 span,
3183 E0527,
3184"pattern requires {} element{} but array has {}",
3185 min_len,
3186pluralize!(min_len),
3187 size,
3188 )3189 .with_span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("expected {0} element{1}", size,
if size == 1 { "" } else { "s" }))
})format!("expected {} element{}", size, pluralize!(size)))
3190 .emit()
3191 }
31923193fn error_scrutinee_with_rest_inconsistent_length(
3194&self,
3195 span: Span,
3196 min_len: u64,
3197 size: u64,
3198 ) -> ErrorGuaranteed {
3199{
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("pattern requires at least {0} element{1} but array has {2}",
min_len, if min_len == 1 { "" } else { "s" }, size))
})).with_code(E0528)
}struct_span_code_err!(
3200self.dcx(),
3201 span,
3202 E0528,
3203"pattern requires at least {} element{} but array has {}",
3204 min_len,
3205pluralize!(min_len),
3206 size,
3207 )3208 .with_span_label(
3209span,
3210::alloc::__export::must_use({
::alloc::fmt::format(format_args!("pattern cannot match array of {0} element{1}",
size, if size == 1 { "" } else { "s" }))
})format!("pattern cannot match array of {} element{}", size, pluralize!(size),),
3211 )
3212 .emit()
3213 }
32143215fn error_scrutinee_unfixed_length(&self, span: Span) -> ErrorGuaranteed {
3216{
self.dcx().struct_span_err(span,
::alloc::__export::must_use({
::alloc::fmt::format(format_args!("cannot pattern-match on an array without a fixed length"))
})).with_code(E0730)
}struct_span_code_err!(
3217self.dcx(),
3218 span,
3219 E0730,
3220"cannot pattern-match on an array without a fixed length",
3221 )3222 .emit()
3223 }
32243225fn error_expected_array_or_slice(
3226&self,
3227 span: Span,
3228 expected_ty: Ty<'tcx>,
3229 pat_info: PatInfo<'tcx>,
3230 ) -> ErrorGuaranteed {
3231let PatInfo { top_info: ti, current_depth, .. } = pat_info;
32323233let mut slice_pat_semantics = false;
3234let mut as_deref = None;
3235let mut slicing = None;
3236if let ty::Ref(_, ty, _) = expected_ty.kind()
3237 && let ty::Array(..) | ty::Slice(..) = ty.kind()
3238 {
3239slice_pat_semantics = true;
3240 } else if self3241 .autoderef(span, expected_ty)
3242 .silence_errors()
3243 .any(|(ty, _)| #[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Slice(..) | ty::Array(..) => true,
_ => false,
}matches!(ty.kind(), ty::Slice(..) | ty::Array(..)))
3244 && let Some(span) = ti.span
3245 && let Some(_) = ti.origin_expr
3246 {
3247let resolved_ty = self.resolve_vars_if_possible(ti.expected);
3248let (is_slice_or_array_or_vector, resolved_ty) =
3249self.is_slice_or_array_or_vector(resolved_ty);
3250match resolved_ty.kind() {
3251 ty::Adt(adt_def, _)
3252if self.tcx.is_diagnostic_item(sym::Option, adt_def.did())
3253 || self.tcx.is_diagnostic_item(sym::Result, adt_def.did()) =>
3254 {
3255// Slicing won't work here, but `.as_deref()` might (issue #91328).
3256as_deref = Some(errors::AsDerefSuggestion { span: span.shrink_to_hi() });
3257 }
3258_ => (),
3259 }
32603261let is_top_level = current_depth <= 1;
3262if is_slice_or_array_or_vector && is_top_level {
3263slicing = Some(errors::SlicingSuggestion { span: span.shrink_to_hi() });
3264 }
3265 }
3266self.dcx().emit_err(errors::ExpectedArrayOrSlice {
3267span,
3268 ty: expected_ty,
3269slice_pat_semantics,
3270as_deref,
3271slicing,
3272 })
3273 }
32743275fn is_slice_or_array_or_vector(&self, ty: Ty<'tcx>) -> (bool, Ty<'tcx>) {
3276match ty.kind() {
3277 ty::Adt(adt_def, _) if self.tcx.is_diagnostic_item(sym::Vec, adt_def.did()) => {
3278 (true, ty)
3279 }
3280 ty::Ref(_, ty, _) => self.is_slice_or_array_or_vector(*ty),
3281 ty::Slice(..) | ty::Array(..) => (true, ty),
3282_ => (false, ty),
3283 }
3284 }
32853286/// Record a pattern that's invalid under Rust 2024 match ergonomics, along with a problematic
3287 /// span, so that the pattern migration lint can desugar it during THIR construction.
3288fn add_rust_2024_migration_desugared_pat(
3289&self,
3290 pat_id: HirId,
3291 subpat: &'tcx Pat<'tcx>,
3292 final_char: char,
3293 def_br_mutbl: Mutability,
3294 ) {
3295// Try to trim the span we're labeling to just the `&` or binding mode that's an issue.
3296let from_expansion = subpat.span.from_expansion();
3297let trimmed_span = if from_expansion {
3298// If the subpattern is from an expansion, highlight the whole macro call instead.
3299subpat.span
3300 } else {
3301let trimmed = self.tcx.sess.source_map().span_through_char(subpat.span, final_char);
3302// The edition of the trimmed span should be the same as `subpat.span`; this will be a
3303 // a hard error if the subpattern is of edition >= 2024. We set it manually to be sure:
3304trimmed.with_ctxt(subpat.span.ctxt())
3305 };
33063307let mut typeck_results = self.typeck_results.borrow_mut();
3308let mut table = typeck_results.rust_2024_migration_desugared_pats_mut();
3309// FIXME(ref_pat_eat_one_layer_2024): The migration diagnostic doesn't know how to track the
3310 // default binding mode in the presence of Rule 3 or Rule 5. As a consequence, the labels it
3311 // gives for default binding modes are wrong, as well as suggestions based on the default
3312 // binding mode. This keeps it from making those suggestions, as doing so could panic.
3313let info = table.entry(pat_id).or_insert_with(|| ty::Rust2024IncompatiblePatInfo {
3314 primary_labels: Vec::new(),
3315 bad_ref_modifiers: false,
3316 bad_mut_modifiers: false,
3317 bad_ref_pats: false,
3318 suggest_eliding_modes: !self.tcx.features().ref_pat_eat_one_layer_2024()
3319 && !self.tcx.features().ref_pat_eat_one_layer_2024_structural(),
3320 });
33213322let pat_kind = if let PatKind::Binding(user_bind_annot, _, _, _) = subpat.kind {
3323// If the user-provided binding modifier doesn't match the default binding mode, we'll
3324 // need to suggest reference patterns, which can affect other bindings.
3325 // For simplicity, we opt to suggest making the pattern fully explicit.
3326info.suggest_eliding_modes &= #[allow(non_exhaustive_omitted_patterns)] match user_bind_annot {
BindingMode(ByRef::Yes(_, mutbl), Mutability::Not) if
mutbl == def_br_mutbl => true,
_ => false,
}matches!(
3327 user_bind_annot,
3328 BindingMode(ByRef::Yes(_, mutbl), Mutability::Not) if mutbl == def_br_mutbl
3329 );
3330if user_bind_annot == BindingMode(ByRef::No, Mutability::Mut) {
3331info.bad_mut_modifiers = true;
3332"`mut` binding modifier"
3333} else {
3334info.bad_ref_modifiers = true;
3335match user_bind_annot.1 {
3336 Mutability::Not => "explicit `ref` binding modifier",
3337 Mutability::Mut => "explicit `ref mut` binding modifier",
3338 }
3339 }
3340 } else {
3341info.bad_ref_pats = true;
3342// For simplicity, we don't try to suggest eliding reference patterns. Thus, we'll
3343 // suggest adding them instead, which can affect the types assigned to bindings.
3344 // As such, we opt to suggest making the pattern fully explicit.
3345info.suggest_eliding_modes = false;
3346"reference pattern"
3347};
3348// Only provide a detailed label if the problematic subpattern isn't from an expansion.
3349 // In the case that it's from a macro, we'll add a more detailed note in the emitter.
3350let primary_label = if from_expansion {
3351// We can't suggest eliding modifiers within expansions.
3352info.suggest_eliding_modes = false;
3353// NB: This wording assumes the only expansions that can produce problematic reference
3354 // patterns and bindings are macros. If a desugaring or AST pass is added that can do
3355 // so, we may want to inspect the span's source callee or macro backtrace.
3356"occurs within macro expansion".to_owned()
3357 } else {
3358::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} not allowed when implicitly borrowing",
pat_kind))
})format!("{pat_kind} not allowed when implicitly borrowing")3359 };
3360info.primary_labels.push((trimmed_span, primary_label));
3361 }
3362}