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};
23use rustc_middle::{bug, span_bug};
24use rustc_session::lint::builtin::NON_EXHAUSTIVE_OMITTED_PATTERNS;
25use rustc_session::parse::feature_err;
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.try_structurally_resolve_type(pat.span, 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!(
807lt.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.try_structurally_resolve_type(pat.span, 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, 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> {
948match 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.try_structurally_resolve_type(span, 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.try_structurally_resolve_type(span, 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(err) = self.demand_suptype_with_origin(&cause, expected, pat_ty) {
1011err.emit();
1012 }
10131014pat_ty1015 }
10161017fn check_pat_range(
1018&self,
1019 span: Span,
1020 lhs: Option<&'tcx hir::PatExpr<'tcx>>,
1021 rhs: Option<&'tcx hir::PatExpr<'tcx>>,
1022 expected: Ty<'tcx>,
1023 ti: &TopInfo<'tcx>,
1024 ) -> Ty<'tcx> {
1025let calc_side = |opt_expr: Option<&'tcx hir::PatExpr<'tcx>>| match opt_expr {
1026None => None,
1027Some(expr) => {
1028let ty = self.check_pat_expr_unadjusted(expr);
1029// Check that the end-point is possibly of numeric or char type.
1030 // The early check here is not for correctness, but rather better
1031 // diagnostics (e.g. when `&str` is being matched, `expected` will
1032 // be peeled to `str` while ty here is still `&str`, if we don't
1033 // err early here, a rather confusing unification error will be
1034 // emitted instead).
1035let ty = self.try_structurally_resolve_type(expr.span, ty);
1036let fail =
1037 !(ty.is_numeric() || ty.is_char() || ty.is_ty_var() || ty.references_error());
1038Some((fail, ty, expr.span))
1039 }
1040 };
1041let mut lhs = calc_side(lhs);
1042let mut rhs = calc_side(rhs);
10431044if let (Some((true, ..)), _) | (_, Some((true, ..))) = (lhs, rhs) {
1045// There exists a side that didn't meet our criteria that the end-point
1046 // be of a numeric or char type, as checked in `calc_side` above.
1047let guar = self.emit_err_pat_range(span, lhs, rhs);
1048return Ty::new_error(self.tcx, guar);
1049 }
10501051// Unify each side with `expected`.
1052 // Subtyping doesn't matter here, as the value is some kind of scalar.
1053let demand_eqtype = |x: &mut _, y| {
1054if let Some((ref mut fail, x_ty, x_span)) = *x1055 && let Err(mut err) = self.demand_eqtype_pat_diag(x_span, expected, x_ty, ti)
1056 {
1057if let Some((_, y_ty, y_span)) = y {
1058self.endpoint_has_type(&mut err, y_span, y_ty);
1059 }
1060err.emit();
1061*fail = true;
1062 }
1063 };
1064demand_eqtype(&mut lhs, rhs);
1065demand_eqtype(&mut rhs, lhs);
10661067if let (Some((true, ..)), _) | (_, Some((true, ..))) = (lhs, rhs) {
1068return Ty::new_misc_error(self.tcx);
1069 }
10701071// Find the unified type and check if it's of numeric or char type again.
1072 // This check is needed if both sides are inference variables.
1073 // We require types to be resolved here so that we emit inference failure
1074 // rather than "_ is not a char or numeric".
1075let ty = self.structurally_resolve_type(span, expected);
1076if !(ty.is_numeric() || ty.is_char() || ty.references_error()) {
1077if let Some((ref mut fail, _, _)) = lhs {
1078*fail = true;
1079 }
1080if let Some((ref mut fail, _, _)) = rhs {
1081*fail = true;
1082 }
1083let guar = self.emit_err_pat_range(span, lhs, rhs);
1084return Ty::new_error(self.tcx, guar);
1085 }
1086ty1087 }
10881089fn endpoint_has_type(&self, err: &mut Diag<'_>, span: Span, ty: Ty<'_>) {
1090if !ty.references_error() {
1091err.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("this is of type `{0}`", ty))
})format!("this is of type `{ty}`"));
1092 }
1093 }
10941095fn emit_err_pat_range(
1096&self,
1097 span: Span,
1098 lhs: Option<(bool, Ty<'tcx>, Span)>,
1099 rhs: Option<(bool, Ty<'tcx>, Span)>,
1100 ) -> ErrorGuaranteed {
1101let span = match (lhs, rhs) {
1102 (Some((true, ..)), Some((true, ..))) => span,
1103 (Some((true, _, sp)), _) => sp,
1104 (_, Some((true, _, sp))) => sp,
1105_ => ::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?"),
1106 };
1107let 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!(
1108self.dcx(),
1109 span,
1110 E0029,
1111"only `char` and numeric types are allowed in range patterns"
1112);
1113let msg = |ty| {
1114let ty = self.resolve_vars_if_possible(ty);
1115::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")1116 };
1117let mut one_side_err = |first_span, first_ty, second: Option<(bool, Ty<'tcx>, Span)>| {
1118err.span_label(first_span, msg(first_ty));
1119if let Some((_, ty, sp)) = second {
1120let ty = self.resolve_vars_if_possible(ty);
1121self.endpoint_has_type(&mut err, sp, ty);
1122 }
1123 };
1124match (lhs, rhs) {
1125 (Some((true, lhs_ty, lhs_sp)), Some((true, rhs_ty, rhs_sp))) => {
1126err.span_label(lhs_sp, msg(lhs_ty));
1127err.span_label(rhs_sp, msg(rhs_ty));
1128 }
1129 (Some((true, lhs_ty, lhs_sp)), rhs) => one_side_err(lhs_sp, lhs_ty, rhs),
1130 (lhs, Some((true, rhs_ty, rhs_sp))) => one_side_err(rhs_sp, rhs_ty, lhs),
1131_ => ::rustc_middle::util::bug::span_bug_fmt(span,
format_args!("Impossible, verified above."))span_bug!(span, "Impossible, verified above."),
1132 }
1133if (lhs, rhs).references_error() {
1134err.downgrade_to_delayed_bug();
1135 }
1136if self.tcx.sess.teach(err.code.unwrap()) {
1137err.note(
1138"In a match expression, only numbers and characters can be matched \
1139 against a range. This is because the compiler checks that the range \
1140 is non-empty at compile-time, and is unable to evaluate arbitrary \
1141 comparison functions. If you want to capture values of an orderable \
1142 type between two end-points, you can use a guard.",
1143 );
1144 }
1145err.emit()
1146 }
11471148fn check_pat_ident(
1149&self,
1150 pat: &'tcx Pat<'tcx>,
1151 user_bind_annot: BindingMode,
1152 var_id: HirId,
1153 ident: Ident,
1154 sub: Option<&'tcx Pat<'tcx>>,
1155 expected: Ty<'tcx>,
1156 pat_info: PatInfo<'tcx>,
1157 ) -> Ty<'tcx> {
1158let PatInfo { binding_mode: def_br, top_info: ti, .. } = pat_info;
11591160// Determine the binding mode...
1161let bm = match user_bind_annot {
1162BindingMode(ByRef::No, Mutability::Mut) if let ByRef::Yes(_, def_br_mutbl) = def_br => {
1163// Only mention the experimental `mut_ref` feature if if we're in edition 2024 and
1164 // using other experimental matching features compatible with it.
1165if pat.span.at_least_rust_2024()
1166 && (self.tcx.features().ref_pat_eat_one_layer_2024()
1167 || self.tcx.features().ref_pat_eat_one_layer_2024_structural())
1168 {
1169if !self.tcx.features().mut_ref() {
1170feature_err(
1171self.tcx.sess,
1172 sym::mut_ref,
1173pat.span.until(ident.span),
1174"binding cannot be both mutable and by-reference",
1175 )
1176 .emit();
1177 }
11781179BindingMode(def_br, Mutability::Mut)
1180 } else {
1181// `mut` resets the binding mode on edition <= 2021
1182self.add_rust_2024_migration_desugared_pat(
1183pat_info.top_info.hir_id,
1184pat,
1185't', // last char of `mut`
1186def_br_mutbl,
1187 );
1188BindingMode(ByRef::No, Mutability::Mut)
1189 }
1190 }
1191BindingMode(ByRef::No, mutbl) => BindingMode(def_br, mutbl),
1192BindingMode(ByRef::Yes(_, user_br_mutbl), _) => {
1193if let ByRef::Yes(_, def_br_mutbl) = def_br {
1194// `ref`/`ref mut` overrides the binding mode on edition <= 2021
1195self.add_rust_2024_migration_desugared_pat(
1196pat_info.top_info.hir_id,
1197pat,
1198match user_br_mutbl {
1199 Mutability::Not => 'f', // last char of `ref`
1200Mutability::Mut => 't', // last char of `ref mut`
1201},
1202def_br_mutbl,
1203 );
1204 }
1205user_bind_annot1206 }
1207 };
12081209// If there exists a pinned reference in the pattern but the binding is not pinned,
1210 // it means the binding is unpinned and thus requires an `Unpin` bound.
1211if pat_info.max_pinnedness == PinnednessCap::Pinned1212 && #[allow(non_exhaustive_omitted_patterns)] match bm.0 {
ByRef::Yes(Pinnedness::Not, _) => true,
_ => false,
}matches!(bm.0, ByRef::Yes(Pinnedness::Not, _))1213 {
1214self.register_bound(
1215expected,
1216self.tcx.require_lang_item(hir::LangItem::Unpin, pat.span),
1217self.misc(pat.span),
1218 )
1219 }
12201221if #[allow(non_exhaustive_omitted_patterns)] match bm.0 {
ByRef::Yes(_, Mutability::Mut) => true,
_ => false,
}matches!(bm.0, ByRef::Yes(_, Mutability::Mut))1222 && let MutblCap::WeaklyNot(and_pat_span) = pat_info.max_ref_mutbl
1223 {
1224let 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!(
1225self.dcx(),
1226 ident.span,
1227 E0596,
1228"cannot borrow as mutable inside an `&` pattern"
1229);
12301231if let Some(span) = and_pat_span {
1232err.span_suggestion(
1233span,
1234"replace this `&` with `&mut`",
1235"&mut ",
1236 Applicability::MachineApplicable,
1237 );
1238 }
1239err.emit();
1240 }
12411242// ...and store it in a side table:
1243self.typeck_results.borrow_mut().pat_binding_modes_mut().insert(pat.hir_id, bm);
12441245{
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:1245",
"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(1245u32),
::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);
12461247let local_ty = self.local_ty(pat.span, pat.hir_id);
1248let eq_ty = match bm.0 {
1249 ByRef::Yes(pinnedness, mutbl) => {
1250// If the binding is like `ref x | ref mut x`,
1251 // then `x` is assigned a value of type `&M T` where M is the
1252 // mutability and T is the expected type.
1253 //
1254 // Under pin ergonomics, if the binding is like `ref pin const|mut x`,
1255 // then `x` is assigned a value of type `&pin M T` where M is the
1256 // mutability and T is the expected type.
1257 //
1258 // `x` is assigned a value of type `&M T`, hence `&M T <: typeof(x)`
1259 // is required. However, we use equality, which is stronger.
1260 // See (note_1) for an explanation.
1261self.new_ref_ty(pat.span, pinnedness, mutbl, expected)
1262 }
1263// Otherwise, the type of x is the expected type `T`.
1264ByRef::No => expected, // As above, `T <: typeof(x)` is required, but we use equality, see (note_1).
1265};
12661267// We have a concrete type for the local, so we do not need to taint it and hide follow up errors *using* the local.
1268let _ = self.demand_eqtype_pat(pat.span, eq_ty, local_ty, &ti);
12691270// If there are multiple arms, make sure they all agree on
1271 // what the type of the binding `x` ought to be.
1272if var_id != pat.hir_id {
1273self.check_binding_alt_eq_ty(user_bind_annot, pat.span, var_id, local_ty, &ti);
1274 }
12751276if let Some(p) = sub {
1277self.check_pat(p, expected, pat_info);
1278 }
12791280local_ty1281 }
12821283/// When a variable is bound several times in a `PatKind::Or`, it'll resolve all of the
1284 /// subsequent bindings of the same name to the first usage. Verify that all of these
1285 /// bindings have the same type by comparing them all against the type of that first pat.
1286fn check_binding_alt_eq_ty(
1287&self,
1288 ba: BindingMode,
1289 span: Span,
1290 var_id: HirId,
1291 ty: Ty<'tcx>,
1292 ti: &TopInfo<'tcx>,
1293 ) {
1294let var_ty = self.local_ty(span, var_id);
1295if let Err(mut err) = self.demand_eqtype_pat_diag(span, var_ty, ty, ti) {
1296let var_ty = self.resolve_vars_if_possible(var_ty);
1297let 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");
1298err.span_label(self.tcx.hir_span(var_id), msg);
1299let in_match = self.tcx.hir_parent_iter(var_id).any(|(_, n)| {
1300#[allow(non_exhaustive_omitted_patterns)] match n {
hir::Node::Expr(hir::Expr {
kind: hir::ExprKind::Match(.., hir::MatchSource::Normal), .. }) =>
true,
_ => false,
}matches!(
1301 n,
1302 hir::Node::Expr(hir::Expr {
1303 kind: hir::ExprKind::Match(.., hir::MatchSource::Normal),
1304 ..
1305 })
1306 )1307 });
1308let pre = if in_match { "in the same arm, " } else { "" };
1309err.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"));
1310self.suggest_adding_missing_ref_or_removing_ref(
1311&mut err,
1312span,
1313var_ty,
1314self.resolve_vars_if_possible(ty),
1315ba,
1316 );
1317err.emit();
1318 }
1319 }
13201321fn suggest_adding_missing_ref_or_removing_ref(
1322&self,
1323 err: &mut Diag<'_>,
1324 span: Span,
1325 expected: Ty<'tcx>,
1326 actual: Ty<'tcx>,
1327 ba: BindingMode,
1328 ) {
1329match (expected.kind(), actual.kind(), ba) {
1330 (ty::Ref(_, inner_ty, _), _, BindingMode::NONE)
1331if self.can_eq(self.param_env, *inner_ty, actual) =>
1332 {
1333err.span_suggestion_verbose(
1334span.shrink_to_lo(),
1335"consider adding `ref`",
1336"ref ",
1337 Applicability::MaybeIncorrect,
1338 );
1339 }
1340 (_, ty::Ref(_, inner_ty, _), BindingMode::REF)
1341if self.can_eq(self.param_env, expected, *inner_ty) =>
1342 {
1343err.span_suggestion_verbose(
1344span.with_hi(span.lo() + BytePos(4)),
1345"consider removing `ref`",
1346"",
1347 Applicability::MaybeIncorrect,
1348 );
1349 }
1350_ => (),
1351 }
1352 }
13531354/// Precondition: pat is a `Ref(_)` pattern
1355// FIXME(pin_ergonomics): add suggestions for `&pin mut` or `&pin const` patterns
1356fn borrow_pat_suggestion(&self, err: &mut Diag<'_>, pat: &Pat<'_>) {
1357let tcx = self.tcx;
1358if let PatKind::Ref(inner, pinned, mutbl) = pat.kind
1359 && let PatKind::Binding(_, _, binding, ..) = inner.kind
1360 {
1361let binding_parent = tcx.parent_hir_node(pat.hir_id);
1362{
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:1362",
"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(1362u32),
::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);
13631364let pin_and_mut = pinned.prefix_str(mutbl).trim_end();
13651366let mut_var_suggestion = 'block: {
1367if mutbl.is_not() {
1368break 'block None;
1369 }
13701371let ident_kind = match binding_parent {
1372 hir::Node::Param(_) => "parameter",
1373 hir::Node::LetStmt(_) => "variable",
1374 hir::Node::Arm(_) => "binding",
13751376// Provide diagnostics only if the parent pattern is struct-like,
1377 // i.e. where `mut binding` makes sense
1378hir::Node::Pat(Pat { kind, .. }) => match kind {
1379 PatKind::Struct(..)
1380 | PatKind::TupleStruct(..)
1381 | PatKind::Or(..)
1382 | PatKind::Guard(..)
1383 | PatKind::Tuple(..)
1384 | PatKind::Slice(..) => "binding",
13851386 PatKind::Missing1387 | PatKind::Wild1388 | PatKind::Never1389 | PatKind::Binding(..)
1390 | PatKind::Box(..)
1391 | PatKind::Deref(_)
1392 | PatKind::Ref(..)
1393 | PatKind::Expr(..)
1394 | PatKind::Range(..)
1395 | PatKind::Err(_) => break 'block None,
1396 },
13971398// Don't provide suggestions in other cases
1399_ => break 'block None,
1400 };
14011402Some((
1403pat.span,
1404::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"),
1405::alloc::__export::must_use({
::alloc::fmt::format(format_args!("mut {0}", binding))
})format!("mut {binding}"),
1406 ))
1407 };
14081409match binding_parent {
1410// Check that there is explicit type (ie this is not a closure param with inferred type)
1411 // so we don't suggest moving something to the type that does not exist
1412hir::Node::Param(hir::Param { ty_span, pat, .. }) if pat.span != *ty_span => {
1413err.multipart_suggestion(
1414::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"),
1415::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(), mutbl.ref_prefix_str().to_owned())]))vec![
1416 (pat.span.until(inner.span), "".to_owned()),
1417 (ty_span.shrink_to_lo(), mutbl.ref_prefix_str().to_owned()),
1418 ],
1419 Applicability::MachineApplicable1420 );
14211422if let Some((sp, msg, sugg)) = mut_var_suggestion {
1423err.span_note(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: `{1}`", msg, sugg))
})format!("{msg}: `{sugg}`"));
1424 }
1425 }
1426 hir::Node::Pat(pt) if let PatKind::TupleStruct(_, pat_arr, _) = pt.kind => {
1427for i in pat_arr.iter() {
1428if let PatKind::Ref(the_ref, _, _) = i.kind
1429 && let PatKind::Binding(mt, _, ident, _) = the_ref.kind
1430 {
1431let BindingMode(_, mtblty) = mt;
1432 err.span_suggestion_verbose(
1433 i.span,
1434::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"),
1435 mtblty.prefix_str().to_string() + &ident.name.to_string(),
1436 Applicability::MaybeIncorrect,
1437 );
1438 }
1439 }
1440if let Some((sp, msg, sugg)) = mut_var_suggestion {
1441err.span_note(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: `{1}`", msg, sugg))
})format!("{msg}: `{sugg}`"));
1442 }
1443 }
1444 hir::Node::Param(_) | hir::Node::Arm(_) | hir::Node::Pat(_) => {
1445// rely on match ergonomics or it might be nested `&&pat`
1446err.span_suggestion_verbose(
1447pat.span.until(inner.span),
1448::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"),
1449"",
1450 Applicability::MaybeIncorrect,
1451 );
14521453if let Some((sp, msg, sugg)) = mut_var_suggestion {
1454err.span_note(sp, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}: `{1}`", msg, sugg))
})format!("{msg}: `{sugg}`"));
1455 }
1456 }
1457_ if let Some((sp, msg, sugg)) = mut_var_suggestion => {
1458err.span_suggestion(sp, msg, sugg, Applicability::MachineApplicable);
1459 }
1460_ => {} // don't provide suggestions in other cases #55175
1461}
1462 }
1463 }
14641465fn check_dereferenceable(
1466&self,
1467 span: Span,
1468 expected: Ty<'tcx>,
1469 inner: &Pat<'_>,
1470 ) -> Result<(), ErrorGuaranteed> {
1471if let PatKind::Binding(..) = inner.kind
1472 && let Some(pointee_ty) = self.shallow_resolve(expected).builtin_deref(true)
1473 && let ty::Dynamic(..) = pointee_ty.kind()
1474 {
1475// This is "x = dyn SomeTrait" being reduced from
1476 // "let &x = &dyn SomeTrait" or "let box x = Box<dyn SomeTrait>", an error.
1477let type_str = self.ty_to_string(expected);
1478let 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!(
1479self.dcx(),
1480 span,
1481 E0033,
1482"type `{}` cannot be dereferenced",
1483 type_str
1484 );
1485err.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"));
1486if self.tcx.sess.teach(err.code.unwrap()) {
1487err.note(CANNOT_IMPLICITLY_DEREF_POINTER_TRAIT_OBJ);
1488 }
1489return Err(err.emit());
1490 }
1491Ok(())
1492 }
14931494fn resolve_pat_struct(
1495&self,
1496 pat: &'tcx Pat<'tcx>,
1497 qpath: &hir::QPath<'tcx>,
1498 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1499// Resolve the path and check the definition for errors.
1500let (variant, pat_ty) = self.check_struct_path(qpath, pat.hir_id)?;
1501Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::Struct { variant } })
1502 }
15031504fn check_pat_struct(
1505&self,
1506 pat: &'tcx Pat<'tcx>,
1507 fields: &'tcx [hir::PatField<'tcx>],
1508 has_rest_pat: bool,
1509 pat_ty: Ty<'tcx>,
1510 variant: &'tcx VariantDef,
1511 expected: Ty<'tcx>,
1512 pat_info: PatInfo<'tcx>,
1513 ) -> Ty<'tcx> {
1514// Type-check the path.
1515let had_err = self.demand_eqtype_pat(pat.span, expected, pat_ty, &pat_info.top_info);
15161517// Type-check subpatterns.
1518match self.check_struct_pat_fields(pat_ty, pat, variant, fields, has_rest_pat, pat_info) {
1519Ok(()) => match had_err {
1520Ok(()) => pat_ty,
1521Err(guar) => Ty::new_error(self.tcx, guar),
1522 },
1523Err(guar) => Ty::new_error(self.tcx, guar),
1524 }
1525 }
15261527fn resolve_pat_path(
1528&self,
1529 path_id: HirId,
1530 span: Span,
1531 qpath: &'tcx hir::QPath<'_>,
1532 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1533let tcx = self.tcx;
15341535let (res, opt_ty, segments) =
1536self.resolve_ty_and_res_fully_qualified_call(qpath, path_id, span);
1537match res {
1538 Res::Err => {
1539let e =
1540self.dcx().span_delayed_bug(qpath.span(), "`Res::Err` but no error emitted");
1541self.set_tainted_by_errors(e);
1542return Err(e);
1543 }
1544 Res::Def(DefKind::AssocFn | DefKind::Ctor(_, CtorKind::Fn) | DefKind::Variant, _) => {
1545let expected = "unit struct, unit variant or constant";
1546let e = report_unexpected_variant_res(tcx, res, None, qpath, span, E0533, expected);
1547return Err(e);
1548 }
1549 Res::SelfCtor(def_id) => {
1550if let ty::Adt(adt_def, _) = *tcx.type_of(def_id).skip_binder().kind()
1551 && adt_def.is_struct()
1552 && let Some((CtorKind::Const, _)) = adt_def.non_enum_variant().ctor
1553 {
1554// Ok, we allow unit struct ctors in patterns only.
1555} else {
1556let e = report_unexpected_variant_res(
1557tcx,
1558res,
1559None,
1560qpath,
1561span,
1562E0533,
1563"unit struct",
1564 );
1565return Err(e);
1566 }
1567 }
1568 Res::Def(
1569 DefKind::Ctor(_, CtorKind::Const)
1570 | DefKind::Const { .. }
1571 | DefKind::AssocConst { .. }
1572 | DefKind::ConstParam,
1573_,
1574 ) => {} // OK
1575_ => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern resolution: {0:?}",
res))bug!("unexpected pattern resolution: {:?}", res),
1576 }
15771578// Find the type of the path pattern, for later checking.
1579let (pat_ty, pat_res) =
1580self.instantiate_value_path(segments, opt_ty, res, span, span, path_id);
1581Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::Path { res, pat_res, segments } })
1582 }
15831584fn check_pat_path(
1585&self,
1586 pat_id_for_diag: HirId,
1587 span: Span,
1588 resolved: &ResolvedPat<'tcx>,
1589 expected: Ty<'tcx>,
1590 ti: &TopInfo<'tcx>,
1591 ) -> Ty<'tcx> {
1592if let Err(err) =
1593self.demand_suptype_with_origin(&self.pattern_cause(ti, span), expected, resolved.ty)
1594 {
1595self.emit_bad_pat_path(err, pat_id_for_diag, span, resolved);
1596 }
1597resolved.ty
1598 }
15991600fn maybe_suggest_range_literal(
1601&self,
1602 e: &mut Diag<'_>,
1603 opt_def_id: Option<hir::def_id::DefId>,
1604 ident: Ident,
1605 ) -> bool {
1606if let Some(def_id) = opt_def_id1607 && let Some(hir::Node::Item(hir::Item {
1608 kind: hir::ItemKind::Const(_, _, _, ct_rhs),
1609 ..
1610 })) = self.tcx.hir_get_if_local(def_id)
1611 && let hir::Node::Expr(expr) = self.tcx.hir_node(ct_rhs.hir_id())
1612 && hir::is_range_literal(expr)
1613 {
1614let span = self.tcx.hir_span(ct_rhs.hir_id());
1615if let Ok(snip) = self.tcx.sess.source_map().span_to_snippet(span) {
1616e.span_suggestion_verbose(
1617ident.span,
1618"you may want to move the range into the match block",
1619snip,
1620 Applicability::MachineApplicable,
1621 );
1622return true;
1623 }
1624 }
1625false
1626}
16271628fn emit_bad_pat_path(
1629&self,
1630mut e: Diag<'_>,
1631 hir_id: HirId,
1632 pat_span: Span,
1633 resolved_pat: &ResolvedPat<'tcx>,
1634 ) {
1635let ResolvedPatKind::Path { res, pat_res, segments } = resolved_pat.kind else {
1636::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:?}");
1637 };
16381639if let Some(span) = self.tcx.hir_res_span(pat_res) {
1640e.span_label(span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} defined here", res.descr()))
})format!("{} defined here", res.descr()));
1641if let [hir::PathSegment { ident, .. }] = segments {
1642e.span_label(
1643pat_span,
1644::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!(
1645"`{}` is interpreted as {} {}, not a new binding",
1646 ident,
1647 res.article(),
1648 res.descr(),
1649 ),
1650 );
1651match self.tcx.parent_hir_node(hir_id) {
1652 hir::Node::PatField(..) => {
1653e.span_suggestion_verbose(
1654ident.span.shrink_to_hi(),
1655"bind the struct field to a different name instead",
1656::alloc::__export::must_use({
::alloc::fmt::format(format_args!(": other_{0}",
ident.as_str().to_lowercase()))
})format!(": other_{}", ident.as_str().to_lowercase()),
1657 Applicability::HasPlaceholders,
1658 );
1659 }
1660_ => {
1661let (type_def_id, item_def_id) = match resolved_pat.ty.kind() {
1662 ty::Adt(def, _) => match res {
1663 Res::Def(DefKind::Const { .. }, def_id) => {
1664 (Some(def.did()), Some(def_id))
1665 }
1666_ => (None, None),
1667 },
1668_ => (None, None),
1669 };
16701671let 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!(
1672 type_def_id.and_then(|id| self.tcx.as_lang_item(id)),
1673Some(
1674 LangItem::Range
1675 | LangItem::RangeFrom
1676 | LangItem::RangeTo
1677 | LangItem::RangeFull
1678 | LangItem::RangeInclusiveStruct
1679 | LangItem::RangeToInclusive,
1680 )
1681 );
1682if is_range {
1683if !self.maybe_suggest_range_literal(&mut e, item_def_id, *ident) {
1684let msg = "constants only support matching by type, \
1685 if you meant to match against a range of values, \
1686 consider using a range pattern like `min ..= max` in the match block";
1687e.note(msg);
1688 }
1689 } else {
1690let msg = "introduce a new binding instead";
1691let sugg = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("other_{0}",
ident.as_str().to_lowercase()))
})format!("other_{}", ident.as_str().to_lowercase());
1692e.span_suggestion(
1693ident.span,
1694msg,
1695sugg,
1696 Applicability::HasPlaceholders,
1697 );
1698 }
1699 }
1700 };
1701 }
1702 }
1703e.emit();
1704 }
17051706fn resolve_pat_tuple_struct(
1707&self,
1708 pat: &'tcx Pat<'tcx>,
1709 qpath: &'tcx hir::QPath<'tcx>,
1710 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1711let tcx = self.tcx;
1712let report_unexpected_res = |res: Res| {
1713let expected = "tuple struct or tuple variant";
1714let e = report_unexpected_variant_res(tcx, res, None, qpath, pat.span, E0164, expected);
1715Err(e)
1716 };
17171718// Resolve the path and check the definition for errors.
1719let (res, opt_ty, segments) =
1720self.resolve_ty_and_res_fully_qualified_call(qpath, pat.hir_id, pat.span);
1721if res == Res::Err {
1722let e = self.dcx().span_delayed_bug(pat.span, "`Res::Err` but no error emitted");
1723self.set_tainted_by_errors(e);
1724return Err(e);
1725 }
17261727// Type-check the path.
1728let (pat_ty, res) =
1729self.instantiate_value_path(segments, opt_ty, res, pat.span, pat.span, pat.hir_id);
1730if !pat_ty.is_fn() {
1731return report_unexpected_res(res);
1732 }
17331734let variant = match res {
1735 Res::Err => {
1736self.dcx().span_bug(pat.span, "`Res::Err` but no error emitted");
1737 }
1738 Res::Def(DefKind::AssocConst { .. } | DefKind::AssocFn, _) => {
1739return report_unexpected_res(res);
1740 }
1741 Res::Def(DefKind::Ctor(_, CtorKind::Fn), _) => tcx.expect_variant_res(res),
1742_ => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern resolution: {0:?}",
res))bug!("unexpected pattern resolution: {:?}", res),
1743 };
17441745// Replace constructor type with constructed type for tuple struct patterns.
1746let pat_ty = pat_ty.fn_sig(tcx).output();
1747let pat_ty = pat_ty.no_bound_vars().expect("expected fn type");
17481749Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::TupleStruct { res, variant } })
1750 }
17511752fn check_pat_tuple_struct(
1753&self,
1754 pat: &'tcx Pat<'tcx>,
1755 qpath: &'tcx hir::QPath<'tcx>,
1756 subpats: &'tcx [Pat<'tcx>],
1757 ddpos: hir::DotDotPos,
1758 res: Res,
1759 pat_ty: Ty<'tcx>,
1760 variant: &'tcx VariantDef,
1761 expected: Ty<'tcx>,
1762 pat_info: PatInfo<'tcx>,
1763 ) -> Ty<'tcx> {
1764let tcx = self.tcx;
1765let on_error = |e| {
1766for pat in subpats {
1767self.check_pat(pat, Ty::new_error(tcx, e), pat_info);
1768 }
1769 };
17701771// Type-check the tuple struct pattern against the expected type.
1772let had_err = self.demand_eqtype_pat(pat.span, expected, pat_ty, &pat_info.top_info);
17731774// Type-check subpatterns.
1775if subpats.len() == variant.fields.len()
1776 || subpats.len() < variant.fields.len() && ddpos.as_opt_usize().is_some()
1777 {
1778let ty::Adt(_, args) = pat_ty.kind() else {
1779::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern type {0:?}",
pat_ty));bug!("unexpected pattern type {:?}", pat_ty);
1780 };
1781for (i, subpat) in subpats.iter().enumerate_and_adjust(variant.fields.len(), ddpos) {
1782let field = &variant.fields[FieldIdx::from_usize(i)];
1783let field_ty = self.field_ty(subpat.span, field, args);
1784self.check_pat(subpat, field_ty, pat_info);
17851786self.tcx.check_stability(
1787 variant.fields[FieldIdx::from_usize(i)].did,
1788Some(subpat.hir_id),
1789 subpat.span,
1790None,
1791 );
1792 }
1793if let Err(e) = had_err {
1794on_error(e);
1795return Ty::new_error(tcx, e);
1796 }
1797 } else {
1798let e = self.emit_err_pat_wrong_number_of_fields(
1799pat.span,
1800res,
1801qpath,
1802subpats,
1803&variant.fields.raw,
1804expected,
1805had_err,
1806 );
1807on_error(e);
1808return Ty::new_error(tcx, e);
1809 }
1810pat_ty1811 }
18121813fn emit_err_pat_wrong_number_of_fields(
1814&self,
1815 pat_span: Span,
1816 res: Res,
1817 qpath: &hir::QPath<'_>,
1818 subpats: &'tcx [Pat<'tcx>],
1819 fields: &'tcx [ty::FieldDef],
1820 expected: Ty<'tcx>,
1821 had_err: Result<(), ErrorGuaranteed>,
1822 ) -> ErrorGuaranteed {
1823let subpats_ending = if subpats.len() == 1 { "" } else { "s" }pluralize!(subpats.len());
1824let fields_ending = if fields.len() == 1 { "" } else { "s" }pluralize!(fields.len());
18251826let subpat_spans = if subpats.is_empty() {
1827::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[pat_span]))vec![pat_span]1828 } else {
1829subpats.iter().map(|p| p.span).collect()
1830 };
1831let last_subpat_span = *subpat_spans.last().unwrap();
1832let res_span = self.tcx.def_span(res.def_id());
1833let def_ident_span = self.tcx.def_ident_span(res.def_id()).unwrap_or(res_span);
1834let field_def_spans = if fields.is_empty() {
1835::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[res_span]))vec![res_span]1836 } else {
1837fields.iter().map(|f| f.ident(self.tcx).span).collect()
1838 };
1839let last_field_def_span = *field_def_spans.last().unwrap();
18401841let 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!(
1842self.dcx(),
1843 MultiSpan::from_spans(subpat_spans),
1844 E0023,
1845"this pattern has {} field{}, but the corresponding {} has {} field{}",
1846 subpats.len(),
1847 subpats_ending,
1848 res.descr(),
1849 fields.len(),
1850 fields_ending,
1851 );
1852err.span_label(
1853last_subpat_span,
1854::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()),
1855 );
1856if self.tcx.sess.source_map().is_multiline(qpath.span().between(last_subpat_span)) {
1857err.span_label(qpath.span(), "");
1858 }
1859if self.tcx.sess.source_map().is_multiline(def_ident_span.between(last_field_def_span)) {
1860err.span_label(def_ident_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} defined here", res.descr()))
})format!("{} defined here", res.descr()));
1861 }
1862for span in &field_def_spans[..field_def_spans.len() - 1] {
1863 err.span_label(*span, "");
1864 }
1865err.span_label(
1866last_field_def_span,
1867::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),
1868 );
18691870// Identify the case `Some(x, y)` where the expected type is e.g. `Option<(T, U)>`.
1871 // More generally, the expected type wants a tuple variant with one field of an
1872 // N-arity-tuple, e.g., `V_i((p_0, .., p_N))`. Meanwhile, the user supplied a pattern
1873 // with the subpatterns directly in the tuple variant pattern, e.g., `V_i(p_0, .., p_N)`.
1874let missing_parentheses = match (expected.kind(), fields, had_err) {
1875// #67037: only do this if we could successfully type-check the expected type against
1876 // the tuple struct pattern. Otherwise the args could get out of range on e.g.,
1877 // `let P() = U;` where `P != U` with `struct Box<T>(T);`.
1878(ty::Adt(_, args), [field], Ok(())) => {
1879let field_ty = self.field_ty(pat_span, field, args);
1880match field_ty.kind() {
1881 ty::Tuple(fields) => fields.len() == subpats.len(),
1882_ => false,
1883 }
1884 }
1885_ => false,
1886 };
1887if missing_parentheses {
1888let (left, right) = match subpats {
1889// This is the zero case; we aim to get the "hi" part of the `QPath`'s
1890 // span as the "lo" and then the "hi" part of the pattern's span as the "hi".
1891 // This looks like:
1892 //
1893 // help: missing parentheses
1894 // |
1895 // L | let A(()) = A(());
1896 // | ^ ^
1897[] => (qpath.span().shrink_to_hi(), pat_span),
1898// Easy case. Just take the "lo" of the first sub-pattern and the "hi" of the
1899 // last sub-pattern. In the case of `A(x)` the first and last may coincide.
1900 // This looks like:
1901 //
1902 // help: missing parentheses
1903 // |
1904 // L | let A((x, y)) = A((1, 2));
1905 // | ^ ^
1906[first, ..] => (first.span.shrink_to_lo(), subpats.last().unwrap().span),
1907 };
1908err.multipart_suggestion(
1909"missing parentheses",
1910::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())],
1911 Applicability::MachineApplicable,
1912 );
1913 } else if fields.len() > subpats.len() && pat_span != DUMMY_SP {
1914let after_fields_span = pat_span.with_hi(pat_span.hi() - BytePos(1)).shrink_to_hi();
1915let all_fields_span = match subpats {
1916 [] => after_fields_span,
1917 [field] => field.span,
1918 [first, .., last] => first.span.to(last.span),
1919 };
19201921// Check if all the fields in the pattern are wildcards.
1922let all_wildcards = subpats.iter().all(|pat| #[allow(non_exhaustive_omitted_patterns)] match pat.kind {
PatKind::Wild => true,
_ => false,
}matches!(pat.kind, PatKind::Wild));
1923let first_tail_wildcard =
1924subpats.iter().enumerate().fold(None, |acc, (pos, pat)| match (acc, &pat.kind) {
1925 (None, PatKind::Wild) => Some(pos),
1926 (Some(_), PatKind::Wild) => acc,
1927_ => None,
1928 });
1929let tail_span = match first_tail_wildcard {
1930None => after_fields_span,
1931Some(0) => subpats[0].span.to(after_fields_span),
1932Some(pos) => subpats[pos - 1].span.shrink_to_hi().to(after_fields_span),
1933 };
19341935// FIXME: heuristic-based suggestion to check current types for where to add `_`.
1936let mut wildcard_sugg = ::alloc::vec::from_elem("_", fields.len() - subpats.len())vec!["_"; fields.len() - subpats.len()].join(", ");
1937if !subpats.is_empty() {
1938wildcard_sugg = String::from(", ") + &wildcard_sugg;
1939 }
19401941err.span_suggestion_verbose(
1942after_fields_span,
1943"use `_` to explicitly ignore each field",
1944wildcard_sugg,
1945 Applicability::MaybeIncorrect,
1946 );
19471948// Only suggest `..` if more than one field is missing
1949 // or the pattern consists of all wildcards.
1950if fields.len() - subpats.len() > 1 || all_wildcards {
1951if subpats.is_empty() || all_wildcards {
1952err.span_suggestion_verbose(
1953all_fields_span,
1954"use `..` to ignore all fields",
1955"..",
1956 Applicability::MaybeIncorrect,
1957 );
1958 } else {
1959err.span_suggestion_verbose(
1960tail_span,
1961"use `..` to ignore the rest of the fields",
1962", ..",
1963 Applicability::MaybeIncorrect,
1964 );
1965 }
1966 }
1967 }
19681969err.emit()
1970 }
19711972fn check_pat_tuple(
1973&self,
1974 span: Span,
1975 elements: &'tcx [Pat<'tcx>],
1976 ddpos: hir::DotDotPos,
1977 expected: Ty<'tcx>,
1978 pat_info: PatInfo<'tcx>,
1979 ) -> Ty<'tcx> {
1980let tcx = self.tcx;
1981let mut expected_len = elements.len();
1982if ddpos.as_opt_usize().is_some() {
1983// Require known type only when `..` is present.
1984if let ty::Tuple(tys) = self.structurally_resolve_type(span, expected).kind() {
1985expected_len = tys.len();
1986 }
1987 }
1988let max_len = cmp::max(expected_len, elements.len());
19891990let element_tys_iter = (0..max_len).map(|_| self.next_ty_var(span));
1991let element_tys = tcx.mk_type_list_from_iter(element_tys_iter);
1992let pat_ty = Ty::new_tup(tcx, element_tys);
1993if let Err(reported) = self.demand_eqtype_pat(span, expected, pat_ty, &pat_info.top_info) {
1994// Walk subpatterns with an expected type of `err` in this case to silence
1995 // further errors being emitted when using the bindings. #50333
1996for (_, elem) in elements.iter().enumerate_and_adjust(max_len, ddpos) {
1997self.check_pat(elem, Ty::new_error(tcx, reported), pat_info);
1998 }
1999Ty::new_error(tcx, reported)
2000 } else {
2001for (i, elem) in elements.iter().enumerate_and_adjust(max_len, ddpos) {
2002self.check_pat(elem, element_tys[i], pat_info);
2003 }
2004pat_ty2005 }
2006 }
20072008fn check_struct_pat_fields(
2009&self,
2010 adt_ty: Ty<'tcx>,
2011 pat: &'tcx Pat<'tcx>,
2012 variant: &'tcx ty::VariantDef,
2013 fields: &'tcx [hir::PatField<'tcx>],
2014 has_rest_pat: bool,
2015 pat_info: PatInfo<'tcx>,
2016 ) -> Result<(), ErrorGuaranteed> {
2017let tcx = self.tcx;
20182019let ty::Adt(adt, args) = adt_ty.kind() else {
2020::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");
2021 };
20222023// Index the struct fields' types.
2024let field_map = variant2025 .fields
2026 .iter_enumerated()
2027 .map(|(i, field)| (field.ident(self.tcx).normalize_to_macros_2_0(), (i, field)))
2028 .collect::<FxHashMap<_, _>>();
20292030// Keep track of which fields have already appeared in the pattern.
2031let mut used_fields = FxHashMap::default();
2032let mut result = Ok(());
20332034let mut inexistent_fields = ::alloc::vec::Vec::new()vec![];
2035// Typecheck each field.
2036for field in fields {
2037let span = field.span;
2038let ident = tcx.adjust_ident(field.ident, variant.def_id);
2039let field_ty = match used_fields.entry(ident) {
2040 Occupied(occupied) => {
2041let guar = self.error_field_already_bound(span, field.ident, *occupied.get());
2042 result = Err(guar);
2043 Ty::new_error(tcx, guar)
2044 }
2045 Vacant(vacant) => {
2046 vacant.insert(span);
2047 field_map
2048 .get(&ident)
2049 .map(|(i, f)| {
2050self.write_field_index(field.hir_id, *i);
2051self.tcx.check_stability(f.did, Some(field.hir_id), span, None);
2052self.field_ty(span, f, args)
2053 })
2054 .unwrap_or_else(|| {
2055 inexistent_fields.push(field);
2056 Ty::new_misc_error(tcx)
2057 })
2058 }
2059 };
20602061self.check_pat(field.pat, field_ty, pat_info);
2062 }
20632064let mut unmentioned_fields = variant2065 .fields
2066 .iter()
2067 .map(|field| (field, field.ident(self.tcx).normalize_to_macros_2_0()))
2068 .filter(|(_, ident)| !used_fields.contains_key(ident))
2069 .collect::<Vec<_>>();
20702071let inexistent_fields_err = if !inexistent_fields.is_empty()
2072 && !inexistent_fields.iter().any(|field| field.ident.name == kw::Underscore)
2073 {
2074// we don't care to report errors for a struct if the struct itself is tainted
2075variant.has_errors()?;
2076Some(self.error_inexistent_fields(
2077adt.variant_descr(),
2078&inexistent_fields,
2079&mut unmentioned_fields,
2080pat,
2081variant,
2082args,
2083 ))
2084 } else {
2085None2086 };
20872088// Require `..` if struct has non_exhaustive attribute.
2089let non_exhaustive = variant.field_list_has_applicable_non_exhaustive();
2090if non_exhaustive && !has_rest_pat {
2091self.error_foreign_non_exhaustive_spat(pat, adt.variant_descr(), fields.is_empty());
2092 }
20932094let mut unmentioned_err = None;
2095// Report an error if an incorrect number of fields was specified.
2096if adt.is_union() {
2097if fields.len() != 1 {
2098self.dcx().emit_err(errors::UnionPatMultipleFields { span: pat.span });
2099 }
2100if has_rest_pat {
2101self.dcx().emit_err(errors::UnionPatDotDot { span: pat.span });
2102 }
2103 } else if !unmentioned_fields.is_empty() {
2104let accessible_unmentioned_fields: Vec<_> = unmentioned_fields2105 .iter()
2106 .copied()
2107 .filter(|(field, _)| self.is_field_suggestable(field, pat.hir_id, pat.span))
2108 .collect();
21092110if !has_rest_pat {
2111if accessible_unmentioned_fields.is_empty() {
2112unmentioned_err = Some(self.error_no_accessible_fields(pat, fields));
2113 } else {
2114unmentioned_err = Some(self.error_unmentioned_fields(
2115pat,
2116&accessible_unmentioned_fields,
2117accessible_unmentioned_fields.len() != unmentioned_fields.len(),
2118fields,
2119 ));
2120 }
2121 } else if non_exhaustive && !accessible_unmentioned_fields.is_empty() {
2122self.lint_non_exhaustive_omitted_patterns(
2123pat,
2124&accessible_unmentioned_fields,
2125adt_ty,
2126 )
2127 }
2128 }
2129match (inexistent_fields_err, unmentioned_err) {
2130 (Some(i), Some(u)) => {
2131if let Err(e) = self.error_tuple_variant_as_struct_pat(pat, fields, variant) {
2132// We don't want to show the nonexistent fields error when this was
2133 // `Foo { a, b }` when it should have been `Foo(a, b)`.
2134i.delay_as_bug();
2135u.delay_as_bug();
2136Err(e)
2137 } else {
2138i.emit();
2139Err(u.emit())
2140 }
2141 }
2142 (None, Some(u)) => {
2143if let Err(e) = self.error_tuple_variant_as_struct_pat(pat, fields, variant) {
2144u.delay_as_bug();
2145Err(e)
2146 } else {
2147Err(u.emit())
2148 }
2149 }
2150 (Some(err), None) => Err(err.emit()),
2151 (None, None) => {
2152self.error_tuple_variant_index_shorthand(variant, pat, fields)?;
2153result2154 }
2155 }
2156 }
21572158fn error_tuple_variant_index_shorthand(
2159&self,
2160 variant: &VariantDef,
2161 pat: &'_ Pat<'_>,
2162 fields: &[hir::PatField<'_>],
2163 ) -> Result<(), ErrorGuaranteed> {
2164// if this is a tuple struct, then all field names will be numbers
2165 // so if any fields in a struct pattern use shorthand syntax, they will
2166 // be invalid identifiers (for example, Foo { 0, 1 }).
2167if let (Some(CtorKind::Fn), PatKind::Struct(qpath, field_patterns, ..)) =
2168 (variant.ctor_kind(), &pat.kind)
2169 {
2170let has_shorthand_field_name = field_patterns.iter().any(|field| field.is_shorthand);
2171if has_shorthand_field_name {
2172let path = rustc_hir_pretty::qpath_to_string(&self.tcx, qpath);
2173let 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!(
2174self.dcx(),
2175 pat.span,
2176 E0769,
2177"tuple variant `{path}` written as struct variant",
2178 );
2179err.span_suggestion_verbose(
2180qpath.span().shrink_to_hi().to(pat.span.shrink_to_hi()),
2181"use the tuple variant pattern syntax instead",
2182::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)),
2183 Applicability::MaybeIncorrect,
2184 );
2185return Err(err.emit());
2186 }
2187 }
2188Ok(())
2189 }
21902191fn error_foreign_non_exhaustive_spat(&self, pat: &Pat<'_>, descr: &str, no_fields: bool) {
2192let sess = self.tcx.sess;
2193let sm = sess.source_map();
2194let sp_brace = sm.end_point(pat.span);
2195let sp_comma = sm.end_point(pat.span.with_hi(sp_brace.hi()));
2196let sugg = if no_fields || sp_brace != sp_comma { ".. }" } else { ", .. }" };
21972198{
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!(
2199self.dcx(),
2200 pat.span,
2201 E0638,
2202"`..` required with {descr} marked as non-exhaustive",
2203 )2204 .with_span_suggestion_verbose(
2205sp_comma,
2206"add `..` at the end of the field list to ignore all other fields",
2207sugg,
2208 Applicability::MachineApplicable,
2209 )
2210 .emit();
2211 }
22122213fn error_field_already_bound(
2214&self,
2215 span: Span,
2216 ident: Ident,
2217 other_field: Span,
2218 ) -> ErrorGuaranteed {
2219{
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!(
2220self.dcx(),
2221 span,
2222 E0025,
2223"field `{}` bound multiple times in the pattern",
2224 ident
2225 )2226 .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"))
2227 .with_span_label(other_field, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("first use of `{0}`", ident))
})format!("first use of `{ident}`"))
2228 .emit()
2229 }
22302231fn error_inexistent_fields(
2232&self,
2233 kind_name: &str,
2234 inexistent_fields: &[&hir::PatField<'tcx>],
2235 unmentioned_fields: &mut Vec<(&'tcx ty::FieldDef, Ident)>,
2236 pat: &'tcx Pat<'tcx>,
2237 variant: &ty::VariantDef,
2238 args: ty::GenericArgsRef<'tcx>,
2239 ) -> Diag<'a> {
2240let tcx = self.tcx;
2241let (field_names, t, plural) = if let [field] = inexistent_fields {
2242 (::alloc::__export::must_use({
::alloc::fmt::format(format_args!("a field named `{0}`", field.ident))
})format!("a field named `{}`", field.ident), "this", "")
2243 } else {
2244 (
2245::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!(
2246"fields named {}",
2247 inexistent_fields
2248 .iter()
2249 .map(|field| format!("`{}`", field.ident))
2250 .collect::<Vec<String>>()
2251 .join(", ")
2252 ),
2253"these",
2254"s",
2255 )
2256 };
2257let spans = inexistent_fields.iter().map(|field| field.ident.span).collect::<Vec<_>>();
2258let 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!(
2259self.dcx(),
2260 spans,
2261 E0026,
2262"{} `{}` does not have {}",
2263 kind_name,
2264 tcx.def_path_str(variant.def_id),
2265 field_names
2266 );
2267if let Some(pat_field) = inexistent_fields.last() {
2268err.span_label(
2269pat_field.ident.span,
2270::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!(
2271"{} `{}` does not have {} field{}",
2272 kind_name,
2273 tcx.def_path_str(variant.def_id),
2274 t,
2275 plural
2276 ),
2277 );
22782279if let [(field_def, field)] = unmentioned_fields.as_slice()
2280 && self.is_field_suggestable(field_def, pat.hir_id, pat.span)
2281 {
2282let suggested_name =
2283find_best_match_for_name(&[field.name], pat_field.ident.name, None);
2284if let Some(suggested_name) = suggested_name {
2285err.span_suggestion_verbose(
2286pat_field.ident.span,
2287"a field with a similar name exists",
2288suggested_name,
2289 Applicability::MaybeIncorrect,
2290 );
22912292// When we have a tuple struct used with struct we don't want to suggest using
2293 // the (valid) struct syntax with numeric field names. Instead we want to
2294 // suggest the expected syntax. We infer that this is the case by parsing the
2295 // `Ident` into an unsized integer. The suggestion will be emitted elsewhere in
2296 // `smart_resolve_context_dependent_help`.
2297if suggested_name.to_ident_string().parse::<usize>().is_err() {
2298// We don't want to throw `E0027` in case we have thrown `E0026` for them.
2299unmentioned_fields.retain(|&(_, x)| x.name != suggested_name);
2300 }
2301 } else if inexistent_fields.len() == 1 {
2302match pat_field.pat.kind {
2303 PatKind::Expr(_)
2304if !self.may_coerce(
2305self.typeck_results.borrow().node_type(pat_field.pat.hir_id),
2306self.field_ty(field.span, field_def, args),
2307 ) => {}
2308_ => {
2309err.span_suggestion_short(
2310pat_field.ident.span,
2311::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!(
2312"`{}` has a field named `{}`",
2313 tcx.def_path_str(variant.def_id),
2314 field.name,
2315 ),
2316field.name,
2317 Applicability::MaybeIncorrect,
2318 );
2319 }
2320 }
2321 }
2322 }
2323 }
2324if tcx.sess.teach(err.code.unwrap()) {
2325err.note(
2326"This error indicates that a struct pattern attempted to \
2327 extract a nonexistent field from a struct. Struct fields \
2328 are identified by the name used before the colon : so struct \
2329 patterns should resemble the declaration of the struct type \
2330 being matched.\n\n\
2331 If you are using shorthand field patterns but want to refer \
2332 to the struct field by a different name, you should rename \
2333 it explicitly.",
2334 );
2335 }
2336err2337 }
23382339fn error_tuple_variant_as_struct_pat(
2340&self,
2341 pat: &Pat<'_>,
2342 fields: &'tcx [hir::PatField<'tcx>],
2343 variant: &ty::VariantDef,
2344 ) -> Result<(), ErrorGuaranteed> {
2345if let (Some(CtorKind::Fn), PatKind::Struct(qpath, pattern_fields, ..)) =
2346 (variant.ctor_kind(), &pat.kind)
2347 {
2348let is_tuple_struct_match = !pattern_fields.is_empty()
2349 && pattern_fields.iter().map(|field| field.ident.name.as_str()).all(is_number);
2350if is_tuple_struct_match {
2351return Ok(());
2352 }
23532354// we don't care to report errors for a struct if the struct itself is tainted
2355variant.has_errors()?;
23562357let path = rustc_hir_pretty::qpath_to_string(&self.tcx, qpath);
2358let 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!(
2359self.dcx(),
2360 pat.span,
2361 E0769,
2362"tuple variant `{}` written as struct variant",
2363 path
2364 );
2365let (sugg, appl) = if fields.len() == variant.fields.len() {
2366 (
2367self.get_suggested_tuple_struct_pattern(fields, variant),
2368 Applicability::MachineApplicable,
2369 )
2370 } else {
2371 (
2372variant.fields.iter().map(|_| "_").collect::<Vec<&str>>().join(", "),
2373 Applicability::MaybeIncorrect,
2374 )
2375 };
2376err.span_suggestion_verbose(
2377qpath.span().shrink_to_hi().to(pat.span.shrink_to_hi()),
2378"use the tuple variant pattern syntax instead",
2379::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})", sugg))
})format!("({sugg})"),
2380appl,
2381 );
2382return Err(err.emit());
2383 }
2384Ok(())
2385 }
23862387fn get_suggested_tuple_struct_pattern(
2388&self,
2389 fields: &[hir::PatField<'_>],
2390 variant: &VariantDef,
2391 ) -> String {
2392let variant_field_idents =
2393variant.fields.iter().map(|f| f.ident(self.tcx)).collect::<Vec<Ident>>();
2394fields2395 .iter()
2396 .map(|field| {
2397match self.tcx.sess.source_map().span_to_snippet(field.pat.span) {
2398Ok(f) => {
2399// Field names are numbers, but numbers
2400 // are not valid identifiers
2401if variant_field_idents.contains(&field.ident) {
2402String::from("_")
2403 } else {
2404f2405 }
2406 }
2407Err(_) => rustc_hir_pretty::pat_to_string(&self.tcx, field.pat),
2408 }
2409 })
2410 .collect::<Vec<String>>()
2411 .join(", ")
2412 }
24132414/// Returns a diagnostic reporting a struct pattern which is missing an `..` due to
2415 /// inaccessible fields.
2416 ///
2417 /// ```text
2418 /// error: pattern requires `..` due to inaccessible fields
2419 /// --> src/main.rs:10:9
2420 /// |
2421 /// LL | let foo::Foo {} = foo::Foo::default();
2422 /// | ^^^^^^^^^^^
2423 /// |
2424 /// help: add a `..`
2425 /// |
2426 /// LL | let foo::Foo { .. } = foo::Foo::default();
2427 /// | ^^^^^^
2428 /// ```
2429fn error_no_accessible_fields(
2430&self,
2431 pat: &Pat<'_>,
2432 fields: &'tcx [hir::PatField<'tcx>],
2433 ) -> Diag<'a> {
2434let mut err = self2435 .dcx()
2436 .struct_span_err(pat.span, "pattern requires `..` due to inaccessible fields");
24372438if let Some(field) = fields.last() {
2439let tail_span = field.span.shrink_to_hi().to(pat.span.shrink_to_hi());
2440let comma_hi_offset =
2441self.tcx.sess.source_map().span_to_snippet(tail_span).ok().and_then(|snippet| {
2442let trimmed = snippet.trim_start();
2443trimmed.starts_with(',').then(|| (snippet.len() - trimmed.len() + 1) as u32)
2444 });
2445err.span_suggestion_verbose(
2446if let Some(comma_hi_offset) = comma_hi_offset {
2447tail_span.with_hi(tail_span.lo() + BytePos(comma_hi_offset)).shrink_to_hi()
2448 } else {
2449field.span.shrink_to_hi()
2450 },
2451"ignore the inaccessible and unused fields",
2452if comma_hi_offset.is_some() { " .." } else { ", .." },
2453 Applicability::MachineApplicable,
2454 );
2455 } else {
2456let qpath_span = if let PatKind::Struct(qpath, ..) = &pat.kind {
2457qpath.span()
2458 } else {
2459::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");
2460 };
24612462// Shrink the span to exclude the `foo:Foo` in `foo::Foo { }`.
2463let span = pat.span.with_lo(qpath_span.shrink_to_hi().hi());
2464err.span_suggestion_verbose(
2465span,
2466"ignore the inaccessible and unused fields",
2467" { .. }",
2468 Applicability::MachineApplicable,
2469 );
2470 }
2471err2472 }
24732474/// Report that a pattern for a `#[non_exhaustive]` struct marked with `non_exhaustive_omitted_patterns`
2475 /// is not exhaustive enough.
2476 ///
2477 /// Nb: the partner lint for enums lives in `compiler/rustc_mir_build/src/thir/pattern/usefulness.rs`.
2478fn lint_non_exhaustive_omitted_patterns(
2479&self,
2480 pat: &Pat<'_>,
2481 unmentioned_fields: &[(&ty::FieldDef, Ident)],
2482 ty: Ty<'tcx>,
2483 ) {
2484struct FieldsNotListed<'a, 'b, 'tcx> {
2485 pat_span: Span,
2486 unmentioned_fields: &'a [(&'b ty::FieldDef, Ident)],
2487 joined_patterns: String,
2488 ty: Ty<'tcx>,
2489 }
24902491impl<'a, 'b, 'c, 'tcx> Diagnostic<'a, ()> for FieldsNotListed<'b, 'c, 'tcx> {
2492fn into_diag(self, dcx: DiagCtxtHandle<'a>, level: Level) -> Diag<'a, ()> {
2493let Self { pat_span, unmentioned_fields, joined_patterns, ty } = self;
2494Diag::new(dcx, level, "some fields are not explicitly listed")
2495 .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))
2496 .with_help(
2497"ensure that all fields are mentioned explicitly by adding the suggested fields",
2498 )
2499 .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!(
2500"the pattern is of type `{ty}` and the `non_exhaustive_omitted_patterns` attribute was found",
2501 ))
2502 }
2503 }
25042505fn joined_uncovered_patterns(witnesses: &[&Ident]) -> String {
2506const LIMIT: usize = 3;
2507match witnesses {
2508 [] => {
2509{
::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!(
2510"expected an uncovered pattern, otherwise why are we emitting an error?"
2511)2512 }
2513 [witness] => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", witness))
})format!("`{witness}`"),
2514 [head @ .., tail] if head.len() < LIMIT => {
2515let head: Vec<_> = head.iter().map(<_>::to_string).collect();
2516::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and `{1}`",
head.join("`, `"), tail))
})format!("`{}` and `{}`", head.join("`, `"), tail)2517 }
2518_ => {
2519let (head, tail) = witnesses.split_at(LIMIT);
2520let head: Vec<_> = head.iter().map(<_>::to_string).collect();
2521::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and {1} more",
head.join("`, `"), tail.len()))
})format!("`{}` and {} more", head.join("`, `"), tail.len())2522 }
2523 }
2524 }
2525let joined_patterns = joined_uncovered_patterns(
2526&unmentioned_fields.iter().map(|(_, i)| i).collect::<Vec<_>>(),
2527 );
25282529self.tcx.emit_node_span_lint(
2530NON_EXHAUSTIVE_OMITTED_PATTERNS,
2531pat.hir_id,
2532pat.span,
2533FieldsNotListed { pat_span: pat.span, unmentioned_fields, joined_patterns, ty },
2534 );
2535 }
25362537/// Returns a diagnostic reporting a struct pattern which does not mention some fields.
2538 ///
2539 /// ```text
2540 /// error[E0027]: pattern does not mention field `bar`
2541 /// --> src/main.rs:15:9
2542 /// |
2543 /// LL | let foo::Foo {} = foo::Foo::new();
2544 /// | ^^^^^^^^^^^ missing field `bar`
2545 /// ```
2546fn error_unmentioned_fields(
2547&self,
2548 pat: &Pat<'_>,
2549 unmentioned_fields: &[(&ty::FieldDef, Ident)],
2550 have_inaccessible_fields: bool,
2551 fields: &'tcx [hir::PatField<'tcx>],
2552 ) -> Diag<'a> {
2553let inaccessible = if have_inaccessible_fields { " and inaccessible fields" } else { "" };
2554let field_names = if let [(_, field)] = unmentioned_fields {
2555::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field `{0}`{1}", field,
inaccessible))
})format!("field `{field}`{inaccessible}")2556 } else {
2557let fields = unmentioned_fields2558 .iter()
2559 .map(|(_, name)| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", name))
})format!("`{name}`"))
2560 .collect::<Vec<String>>()
2561 .join(", ");
2562::alloc::__export::must_use({
::alloc::fmt::format(format_args!("fields {0}{1}", fields,
inaccessible))
})format!("fields {fields}{inaccessible}")2563 };
2564let 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!(
2565self.dcx(),
2566 pat.span,
2567 E0027,
2568"pattern does not mention {}",
2569 field_names
2570 );
2571err.span_label(pat.span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("missing {0}", field_names))
})format!("missing {field_names}"));
2572let len = unmentioned_fields.len();
2573let (prefix, postfix, sp) = match fields {
2574 [] => match &pat.kind {
2575 PatKind::Struct(path, [], None) => {
2576 (" { ", " }", path.span().shrink_to_hi().until(pat.span.shrink_to_hi()))
2577 }
2578_ => return err,
2579 },
2580 [.., field] => {
2581// Account for last field having a trailing comma or parse recovery at the tail of
2582 // the pattern to avoid invalid suggestion (#78511).
2583let tail = field.span.shrink_to_hi().with_hi(pat.span.hi());
2584match &pat.kind {
2585 PatKind::Struct(..) => (", ", " }", tail),
2586_ => return err,
2587 }
2588 }
2589 };
2590err.span_suggestion(
2591sp,
2592::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!(
2593"include the missing field{} in the pattern{}",
2594pluralize!(len),
2595if have_inaccessible_fields { " and ignore the inaccessible fields" } else { "" }
2596 ),
2597::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!(
2598"{}{}{}{}",
2599 prefix,
2600 unmentioned_fields
2601 .iter()
2602 .map(|(_, name)| {
2603let field_name = name.to_string();
2604if is_number(&field_name) { format!("{field_name}: _") } else { field_name }
2605 })
2606 .collect::<Vec<_>>()
2607 .join(", "),
2608if have_inaccessible_fields { ", .." } else { "" },
2609 postfix,
2610 ),
2611 Applicability::MachineApplicable,
2612 );
2613err.span_suggestion(
2614sp,
2615::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!(
2616"if you don't care about {these} missing field{s}, you can explicitly ignore {them}",
2617 these = pluralize!("this", len),
2618 s = pluralize!(len),
2619 them = if len == 1 { "it" } else { "them" },
2620 ),
2621::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!(
2622"{}{}{}{}",
2623 prefix,
2624 unmentioned_fields
2625 .iter()
2626 .map(|(_, name)| {
2627let field_name = name.to_string();
2628format!("{field_name}: _")
2629 })
2630 .collect::<Vec<_>>()
2631 .join(", "),
2632if have_inaccessible_fields { ", .." } else { "" },
2633 postfix,
2634 ),
2635 Applicability::MachineApplicable,
2636 );
2637err.span_suggestion(
2638sp,
2639"or always ignore missing fields here",
2640::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}..{1}", prefix, postfix))
})format!("{prefix}..{postfix}"),
2641 Applicability::MachineApplicable,
2642 );
2643err2644 }
26452646fn check_pat_box(
2647&self,
2648 span: Span,
2649 inner: &'tcx Pat<'tcx>,
2650 expected: Ty<'tcx>,
2651 pat_info: PatInfo<'tcx>,
2652 ) -> Ty<'tcx> {
2653let tcx = self.tcx;
2654let (box_ty, inner_ty) = self2655 .check_dereferenceable(span, expected, inner)
2656 .and_then(|()| {
2657// Here, `demand::subtype` is good enough, but I don't
2658 // think any errors can be introduced by using `demand::eqtype`.
2659let inner_ty = self.next_ty_var(inner.span);
2660let box_ty = Ty::new_box(tcx, inner_ty);
2661self.demand_eqtype_pat(span, expected, box_ty, &pat_info.top_info)?;
2662Ok((box_ty, inner_ty))
2663 })
2664 .unwrap_or_else(|guar| {
2665let err = Ty::new_error(tcx, guar);
2666 (err, err)
2667 });
2668self.check_pat(inner, inner_ty, pat_info);
2669box_ty2670 }
26712672fn check_pat_deref(
2673&self,
2674 span: Span,
2675 inner: &'tcx Pat<'tcx>,
2676 expected: Ty<'tcx>,
2677 pat_info: PatInfo<'tcx>,
2678 ) -> Ty<'tcx> {
2679let target_ty = self.deref_pat_target(span, expected);
2680self.check_pat(inner, target_ty, pat_info);
2681self.register_deref_mut_bounds_if_needed(span, inner, [expected]);
2682expected2683 }
26842685fn deref_pat_target(&self, span: Span, source_ty: Ty<'tcx>) -> Ty<'tcx> {
2686// Register a `DerefPure` bound, which is required by all `deref!()` pats.
2687let tcx = self.tcx;
2688self.register_bound(
2689source_ty,
2690tcx.require_lang_item(hir::LangItem::DerefPure, span),
2691self.misc(span),
2692 );
2693// The expected type for the deref pat's inner pattern is `<expected as Deref>::Target`.
2694let target_ty = Ty::new_projection(
2695tcx,
2696tcx.require_lang_item(hir::LangItem::DerefTarget, span),
2697 [source_ty],
2698 );
2699let target_ty = self.normalize(span, target_ty);
2700self.try_structurally_resolve_type(span, target_ty)
2701 }
27022703/// Check if the interior of a deref pattern (either explicit or implicit) has any `ref mut`
2704 /// bindings, which would require `DerefMut` to be emitted in MIR building instead of just
2705 /// `Deref`. We do this *after* checking the inner pattern, since we want to make sure to
2706 /// account for `ref mut` binding modes inherited from implicitly dereferencing `&mut` refs.
2707fn register_deref_mut_bounds_if_needed(
2708&self,
2709 span: Span,
2710 inner: &'tcx Pat<'tcx>,
2711 derefed_tys: impl IntoIterator<Item = Ty<'tcx>>,
2712 ) {
2713if self.typeck_results.borrow().pat_has_ref_mut_binding(inner) {
2714for mutably_derefed_ty in derefed_tys {
2715self.register_bound(
2716 mutably_derefed_ty,
2717self.tcx.require_lang_item(hir::LangItem::DerefMut, span),
2718self.misc(span),
2719 );
2720 }
2721 }
2722 }
27232724// Precondition: Pat is Ref(inner)
2725fn check_pat_ref(
2726&self,
2727 pat: &'tcx Pat<'tcx>,
2728 inner: &'tcx Pat<'tcx>,
2729 pat_pinned: Pinnedness,
2730 pat_mutbl: Mutability,
2731mut expected: Ty<'tcx>,
2732mut pat_info: PatInfo<'tcx>,
2733 ) -> Ty<'tcx> {
2734let tcx = self.tcx;
27352736let pat_prefix_span =
2737inner.span.find_ancestor_inside(pat.span).map(|end| pat.span.until(end));
27382739let ref_pat_matches_mut_ref = self.ref_pat_matches_mut_ref();
2740if ref_pat_matches_mut_ref && pat_mutbl == Mutability::Not {
2741// If `&` patterns can match against mutable reference types (RFC 3627, Rule 5), we need
2742 // to prevent subpatterns from binding with `ref mut`. Subpatterns of a shared reference
2743 // pattern should have read-only access to the scrutinee, and the borrow checker won't
2744 // catch it in this case.
2745pat_info.max_ref_mutbl = pat_info.max_ref_mutbl.cap_to_weakly_not(pat_prefix_span);
2746 }
27472748expected = self.try_structurally_resolve_type(pat.span, expected);
2749// Determine whether we're consuming an inherited reference and resetting the default
2750 // binding mode, based on edition and enabled experimental features.
2751if let ByRef::Yes(inh_pin, inh_mut) = pat_info.binding_mode
2752 && pat_pinned == inh_pin2753 {
2754match self.ref_pat_matches_inherited_ref(pat.span.edition()) {
2755 InheritedRefMatchRule::EatOuter => {
2756// ref pattern attempts to consume inherited reference
2757if pat_mutbl > inh_mut {
2758// Tried to match inherited `ref` with `&mut`
2759 // NB: This assumes that `&` patterns can match against mutable references
2760 // (RFC 3627, Rule 5). If we implement a pattern typing ruleset with Rule 4E
2761 // but not Rule 5, we'll need to check that here.
2762if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2763self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2764 }
27652766pat_info.binding_mode = ByRef::No;
2767self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2768self.check_pat(inner, expected, pat_info);
2769return expected;
2770 }
2771 InheritedRefMatchRule::EatInner => {
2772if let ty::Ref(_, _, r_mutbl) = *expected.kind()
2773 && pat_mutbl <= r_mutbl2774 {
2775// Match against the reference type; don't consume the inherited ref.
2776 // NB: The check for compatible pattern and ref type mutability assumes that
2777 // `&` patterns can match against mutable references (RFC 3627, Rule 5). If
2778 // we implement a pattern typing ruleset with Rule 4 (including the fallback
2779 // to matching the inherited ref when the inner ref can't match) but not
2780 // Rule 5, we'll need to check that here.
2781if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2782// NB: For RFC 3627's Rule 3, we limit the default binding mode's ref
2783 // mutability to `pat_info.max_ref_mutbl`. If we implement a pattern typing
2784 // ruleset with Rule 4 but not Rule 3, we'll need to check that here.
2785if true {
if !self.downgrade_mut_inside_shared() {
::core::panicking::panic("assertion failed: self.downgrade_mut_inside_shared()")
};
};debug_assert!(self.downgrade_mut_inside_shared());
2786let mutbl_cap = cmp::min(r_mutbl, pat_info.max_ref_mutbl.as_mutbl());
2787pat_info.binding_mode = pat_info.binding_mode.cap_ref_mutability(mutbl_cap);
2788 } else {
2789// The reference pattern can't match against the expected type, so try
2790 // matching against the inherited ref instead.
2791if pat_mutbl > inh_mut {
2792// We can't match an inherited shared reference with `&mut`.
2793 // NB: This assumes that `&` patterns can match against mutable
2794 // references (RFC 3627, Rule 5). If we implement a pattern typing
2795 // ruleset with Rule 4 but not Rule 5, we'll need to check that here.
2796 // FIXME(ref_pat_eat_one_layer_2024_structural): If we already tried
2797 // matching the real reference, the error message should explain that
2798 // falling back to the inherited reference didn't work. This should be
2799 // the same error as the old-Edition version below.
2800if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2801self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2802 }
28032804pat_info.binding_mode = ByRef::No;
2805self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2806self.check_pat(inner, expected, pat_info);
2807return expected;
2808 }
2809 }
2810 InheritedRefMatchRule::EatBoth { consider_inherited_ref: true } => {
2811// Reset binding mode on old editions
2812pat_info.binding_mode = ByRef::No;
28132814if let ty::Ref(_, inner_ty, _) = *expected.kind() {
2815// Consume both the inherited and inner references.
2816if pat_mutbl.is_mut() && inh_mut.is_mut() {
2817// As a special case, a `&mut` reference pattern will be able to match
2818 // against a reference type of any mutability if the inherited ref is
2819 // mutable. Since this allows us to match against a shared reference
2820 // type, we refer to this as "falling back" to matching the inherited
2821 // reference, though we consume the real reference as well. We handle
2822 // this here to avoid adding this case to the common logic below.
2823self.check_pat(inner, inner_ty, pat_info);
2824return expected;
2825 } else {
2826// Otherwise, use the common logic below for matching the inner
2827 // reference type.
2828 // FIXME(ref_pat_eat_one_layer_2024_structural): If this results in a
2829 // mutability mismatch, the error message should explain that falling
2830 // back to the inherited reference didn't work. This should be the same
2831 // error as the Edition 2024 version above.
2832}
2833 } else {
2834// The expected type isn't a reference type, so only match against the
2835 // inherited reference.
2836if pat_mutbl > inh_mut {
2837// We can't match a lone inherited shared reference with `&mut`.
2838self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2839 }
28402841self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2842self.check_pat(inner, expected, pat_info);
2843return expected;
2844 }
2845 }
2846 InheritedRefMatchRule::EatBoth { consider_inherited_ref: false } => {
2847// Reset binding mode on stable Rust. This will be a type error below if
2848 // `expected` is not a reference type.
2849pat_info.binding_mode = ByRef::No;
2850self.add_rust_2024_migration_desugared_pat(
2851pat_info.top_info.hir_id,
2852pat,
2853match pat_mutbl {
2854 Mutability::Not => '&', // last char of `&`
2855Mutability::Mut => 't', // last char of `&mut`
2856},
2857inh_mut,
2858 )
2859 }
2860 }
2861 }
28622863let (ref_ty, inner_ty) = match self.check_dereferenceable(pat.span, expected, inner) {
2864Ok(()) => {
2865// `demand::subtype` would be good enough, but using `eqtype` turns
2866 // out to be equally general. See (note_1) for details.
28672868 // Take region, inner-type from expected type if we can,
2869 // to avoid creating needless variables. This also helps with
2870 // the bad interactions of the given hack detailed in (note_1).
2871{
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:2871",
"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(2871u32),
::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);
2872match expected.maybe_pinned_ref() {
2873Some((r_ty, r_pinned, r_mutbl, _))
2874if ((ref_pat_matches_mut_ref && r_mutbl >= pat_mutbl)
2875 || r_mutbl == pat_mutbl)
2876 && pat_pinned == r_pinned =>
2877 {
2878if r_mutbl == Mutability::Not {
2879pat_info.max_ref_mutbl = MutblCap::Not;
2880 }
2881if r_pinned == Pinnedness::Pinned {
2882pat_info.max_pinnedness = PinnednessCap::Pinned;
2883 }
28842885 (expected, r_ty)
2886 }
2887_ => {
2888let inner_ty = self.next_ty_var(inner.span);
2889let ref_ty = self.new_ref_ty(pat.span, pat_pinned, pat_mutbl, inner_ty);
2890{
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:2890",
"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(2890u32),
::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);
2891let err = self.demand_eqtype_pat_diag(
2892pat.span,
2893expected,
2894ref_ty,
2895&pat_info.top_info,
2896 );
28972898// Look for a case like `fn foo(&foo: u32)` and suggest
2899 // `fn foo(foo: &u32)`
2900if let Err(mut err) = err {
2901self.borrow_pat_suggestion(&mut err, pat);
2902err.emit();
2903 }
2904 (ref_ty, inner_ty)
2905 }
2906 }
2907 }
2908Err(guar) => {
2909let err = Ty::new_error(tcx, guar);
2910 (err, err)
2911 }
2912 };
29132914self.check_pat(inner, inner_ty, pat_info);
2915ref_ty2916 }
29172918/// Create a reference or pinned reference type with a fresh region variable.
2919fn new_ref_ty(
2920&self,
2921 span: Span,
2922 pinnedness: Pinnedness,
2923 mutbl: Mutability,
2924 ty: Ty<'tcx>,
2925 ) -> Ty<'tcx> {
2926let region = self.next_region_var(RegionVariableOrigin::PatternRegion(span));
2927let ref_ty = Ty::new_ref(self.tcx, region, ty, mutbl);
2928if pinnedness.is_pinned() {
2929return self.new_pinned_ty(span, ref_ty);
2930 }
2931ref_ty2932 }
29332934/// Create a pinned type.
2935fn new_pinned_ty(&self, span: Span, ty: Ty<'tcx>) -> Ty<'tcx> {
2936Ty::new_adt(
2937self.tcx,
2938self.tcx.adt_def(self.tcx.require_lang_item(LangItem::Pin, span)),
2939self.tcx.mk_args(&[ty.into()]),
2940 )
2941 }
29422943fn error_inherited_ref_mutability_mismatch(
2944&self,
2945 pat: &'tcx Pat<'tcx>,
2946 pat_prefix_span: Option<Span>,
2947 ) -> ErrorGuaranteed {
2948let err_msg = "mismatched types";
2949let err = if let Some(span) = pat_prefix_span {
2950let mut err = self.dcx().struct_span_err(span, err_msg);
2951err.code(E0308);
2952err.note("cannot match inherited `&` with `&mut` pattern");
2953err.span_suggestion_verbose(
2954span,
2955"replace this `&mut` pattern with `&`",
2956"&",
2957 Applicability::MachineApplicable,
2958 );
2959err2960 } else {
2961self.dcx().struct_span_err(pat.span, err_msg)
2962 };
2963err.emit()
2964 }
29652966fn try_resolve_slice_ty_to_array_ty(
2967&self,
2968 before: &'tcx [Pat<'tcx>],
2969 slice: Option<&'tcx Pat<'tcx>>,
2970 span: Span,
2971 ) -> Option<Ty<'tcx>> {
2972if slice.is_some() {
2973return None;
2974 }
29752976let tcx = self.tcx;
2977let len = before.len();
2978let inner_ty = self.next_ty_var(span);
29792980Some(Ty::new_array(tcx, inner_ty, len.try_into().unwrap()))
2981 }
29822983/// Used to determines whether we can infer the expected type in the slice pattern to be of type array.
2984 /// This is only possible if we're in an irrefutable pattern. If we were to allow this in refutable
2985 /// patterns we wouldn't e.g. report ambiguity in the following situation:
2986 ///
2987 /// ```ignore(rust)
2988 /// struct Zeroes;
2989 /// const ARR: [usize; 2] = [0; 2];
2990 /// const ARR2: [usize; 2] = [2; 2];
2991 ///
2992 /// impl Into<&'static [usize; 2]> for Zeroes {
2993 /// fn into(self) -> &'static [usize; 2] {
2994 /// &ARR
2995 /// }
2996 /// }
2997 ///
2998 /// impl Into<&'static [usize]> for Zeroes {
2999 /// fn into(self) -> &'static [usize] {
3000 /// &ARR2
3001 /// }
3002 /// }
3003 ///
3004 /// fn main() {
3005 /// let &[a, b]: &[usize] = Zeroes.into() else {
3006 /// ..
3007 /// };
3008 /// }
3009 /// ```
3010 ///
3011 /// If we're in an irrefutable pattern we prefer the array impl candidate given that
3012 /// the slice impl candidate would be rejected anyway (if no ambiguity existed).
3013fn pat_is_irrefutable(&self, decl_origin: Option<DeclOrigin<'_>>) -> bool {
3014match decl_origin {
3015Some(DeclOrigin::LocalDecl { els: None }) => true,
3016Some(DeclOrigin::LocalDecl { els: Some(_) } | DeclOrigin::LetExpr) | None => false,
3017 }
3018 }
30193020/// Type check a slice pattern.
3021 ///
3022 /// Syntactically, these look like `[pat_0, ..., pat_n]`.
3023 /// Semantically, we are type checking a pattern with structure:
3024 /// ```ignore (not-rust)
3025 /// [before_0, ..., before_n, (slice, after_0, ... after_n)?]
3026 /// ```
3027 /// The type of `slice`, if it is present, depends on the `expected` type.
3028 /// If `slice` is missing, then so is `after_i`.
3029 /// If `slice` is present, it can still represent 0 elements.
3030fn check_pat_slice(
3031&self,
3032 span: Span,
3033 before: &'tcx [Pat<'tcx>],
3034 slice: Option<&'tcx Pat<'tcx>>,
3035 after: &'tcx [Pat<'tcx>],
3036 expected: Ty<'tcx>,
3037 pat_info: PatInfo<'tcx>,
3038 ) -> Ty<'tcx> {
3039let expected = self.try_structurally_resolve_type(span, expected);
30403041// If the pattern is irrefutable and `expected` is an infer ty, we try to equate it
3042 // to an array if the given pattern allows it. See issue #76342
3043if self.pat_is_irrefutable(pat_info.decl_origin) && expected.is_ty_var() {
3044if let Some(resolved_arr_ty) =
3045self.try_resolve_slice_ty_to_array_ty(before, slice, span)
3046 {
3047{
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:3047",
"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(3047u32),
::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);
3048let _ = self.demand_eqtype(span, expected, resolved_arr_ty);
3049 }
3050 }
30513052let expected = self.structurally_resolve_type(span, expected);
3053{
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:3053",
"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(3053u32),
::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);
30543055let (element_ty, opt_slice_ty, inferred) = match *expected.kind() {
3056// An array, so we might have something like `let [a, b, c] = [0, 1, 2];`.
3057ty::Array(element_ty, len) => {
3058let min = before.len() as u64 + after.len() as u64;
3059let (opt_slice_ty, expected) =
3060self.check_array_pat_len(span, element_ty, expected, slice, len, min);
3061// `opt_slice_ty.is_none()` => `slice.is_none()`.
3062 // Note, though, that opt_slice_ty could be `Some(error_ty)`.
3063if !(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());
3064 (element_ty, opt_slice_ty, expected)
3065 }
3066 ty::Slice(element_ty) => (element_ty, Some(expected), expected),
3067// The expected type must be an array or slice, but was neither, so error.
3068_ => {
3069let guar = expected.error_reported().err().unwrap_or_else(|| {
3070self.error_expected_array_or_slice(span, expected, pat_info)
3071 });
3072let err = Ty::new_error(self.tcx, guar);
3073 (err, Some(err), err)
3074 }
3075 };
30763077// Type check all the patterns before `slice`.
3078for elt in before {
3079self.check_pat(elt, element_ty, pat_info);
3080 }
3081// Type check the `slice`, if present, against its expected type.
3082if let Some(slice) = slice {
3083self.check_pat(slice, opt_slice_ty.unwrap(), pat_info);
3084 }
3085// Type check the elements after `slice`, if present.
3086for elt in after {
3087self.check_pat(elt, element_ty, pat_info);
3088 }
3089inferred3090 }
30913092/// Type check the length of an array pattern.
3093 ///
3094 /// Returns both the type of the variable length pattern (or `None`), and the potentially
3095 /// inferred array type. We only return `None` for the slice type if `slice.is_none()`.
3096fn check_array_pat_len(
3097&self,
3098 span: Span,
3099 element_ty: Ty<'tcx>,
3100 arr_ty: Ty<'tcx>,
3101 slice: Option<&'tcx Pat<'tcx>>,
3102 len: ty::Const<'tcx>,
3103 min_len: u64,
3104 ) -> (Option<Ty<'tcx>>, Ty<'tcx>) {
3105let len = self.try_structurally_resolve_const(span, len).try_to_target_usize(self.tcx);
31063107let guar = if let Some(len) = len {
3108// Now we know the length...
3109if slice.is_none() {
3110// ...and since there is no variable-length pattern,
3111 // we require an exact match between the number of elements
3112 // in the array pattern and as provided by the matched type.
3113if min_len == len {
3114return (None, arr_ty);
3115 }
31163117self.error_scrutinee_inconsistent_length(span, min_len, len)
3118 } else if let Some(pat_len) = len.checked_sub(min_len) {
3119// The variable-length pattern was there,
3120 // so it has an array type with the remaining elements left as its size...
3121return (Some(Ty::new_array(self.tcx, element_ty, pat_len)), arr_ty);
3122 } else {
3123// ...however, in this case, there were no remaining elements.
3124 // That is, the slice pattern requires more than the array type offers.
3125self.error_scrutinee_with_rest_inconsistent_length(span, min_len, len)
3126 }
3127 } else if slice.is_none() {
3128// We have a pattern with a fixed length,
3129 // which we can use to infer the length of the array.
3130let updated_arr_ty = Ty::new_array(self.tcx, element_ty, min_len);
3131self.demand_eqtype(span, updated_arr_ty, arr_ty);
3132return (None, updated_arr_ty);
3133 } else {
3134// We have a variable-length pattern and don't know the array length.
3135 // This happens if we have e.g.,
3136 // `let [a, b, ..] = arr` where `arr: [T; N]` where `const N: usize`.
3137self.error_scrutinee_unfixed_length(span)
3138 };
31393140// If we get here, we must have emitted an error.
3141(Some(Ty::new_error(self.tcx, guar)), arr_ty)
3142 }
31433144fn error_scrutinee_inconsistent_length(
3145&self,
3146 span: Span,
3147 min_len: u64,
3148 size: u64,
3149 ) -> ErrorGuaranteed {
3150{
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!(
3151self.dcx(),
3152 span,
3153 E0527,
3154"pattern requires {} element{} but array has {}",
3155 min_len,
3156pluralize!(min_len),
3157 size,
3158 )3159 .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)))
3160 .emit()
3161 }
31623163fn error_scrutinee_with_rest_inconsistent_length(
3164&self,
3165 span: Span,
3166 min_len: u64,
3167 size: u64,
3168 ) -> ErrorGuaranteed {
3169{
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!(
3170self.dcx(),
3171 span,
3172 E0528,
3173"pattern requires at least {} element{} but array has {}",
3174 min_len,
3175pluralize!(min_len),
3176 size,
3177 )3178 .with_span_label(
3179span,
3180::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),),
3181 )
3182 .emit()
3183 }
31843185fn error_scrutinee_unfixed_length(&self, span: Span) -> ErrorGuaranteed {
3186{
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!(
3187self.dcx(),
3188 span,
3189 E0730,
3190"cannot pattern-match on an array without a fixed length",
3191 )3192 .emit()
3193 }
31943195fn error_expected_array_or_slice(
3196&self,
3197 span: Span,
3198 expected_ty: Ty<'tcx>,
3199 pat_info: PatInfo<'tcx>,
3200 ) -> ErrorGuaranteed {
3201let PatInfo { top_info: ti, current_depth, .. } = pat_info;
32023203let mut slice_pat_semantics = false;
3204let mut as_deref = None;
3205let mut slicing = None;
3206if let ty::Ref(_, ty, _) = expected_ty.kind()
3207 && let ty::Array(..) | ty::Slice(..) = ty.kind()
3208 {
3209slice_pat_semantics = true;
3210 } else if self3211 .autoderef(span, expected_ty)
3212 .silence_errors()
3213 .any(|(ty, _)| #[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Slice(..) | ty::Array(..) => true,
_ => false,
}matches!(ty.kind(), ty::Slice(..) | ty::Array(..)))
3214 && let Some(span) = ti.span
3215 && let Some(_) = ti.origin_expr
3216 {
3217let resolved_ty = self.resolve_vars_if_possible(ti.expected);
3218let (is_slice_or_array_or_vector, resolved_ty) =
3219self.is_slice_or_array_or_vector(resolved_ty);
3220match resolved_ty.kind() {
3221 ty::Adt(adt_def, _)
3222if self.tcx.is_diagnostic_item(sym::Option, adt_def.did())
3223 || self.tcx.is_diagnostic_item(sym::Result, adt_def.did()) =>
3224 {
3225// Slicing won't work here, but `.as_deref()` might (issue #91328).
3226as_deref = Some(errors::AsDerefSuggestion { span: span.shrink_to_hi() });
3227 }
3228_ => (),
3229 }
32303231let is_top_level = current_depth <= 1;
3232if is_slice_or_array_or_vector && is_top_level {
3233slicing = Some(errors::SlicingSuggestion { span: span.shrink_to_hi() });
3234 }
3235 }
3236self.dcx().emit_err(errors::ExpectedArrayOrSlice {
3237span,
3238 ty: expected_ty,
3239slice_pat_semantics,
3240as_deref,
3241slicing,
3242 })
3243 }
32443245fn is_slice_or_array_or_vector(&self, ty: Ty<'tcx>) -> (bool, Ty<'tcx>) {
3246match ty.kind() {
3247 ty::Adt(adt_def, _) if self.tcx.is_diagnostic_item(sym::Vec, adt_def.did()) => {
3248 (true, ty)
3249 }
3250 ty::Ref(_, ty, _) => self.is_slice_or_array_or_vector(*ty),
3251 ty::Slice(..) | ty::Array(..) => (true, ty),
3252_ => (false, ty),
3253 }
3254 }
32553256/// Record a pattern that's invalid under Rust 2024 match ergonomics, along with a problematic
3257 /// span, so that the pattern migration lint can desugar it during THIR construction.
3258fn add_rust_2024_migration_desugared_pat(
3259&self,
3260 pat_id: HirId,
3261 subpat: &'tcx Pat<'tcx>,
3262 final_char: char,
3263 def_br_mutbl: Mutability,
3264 ) {
3265// Try to trim the span we're labeling to just the `&` or binding mode that's an issue.
3266let from_expansion = subpat.span.from_expansion();
3267let trimmed_span = if from_expansion {
3268// If the subpattern is from an expansion, highlight the whole macro call instead.
3269subpat.span
3270 } else {
3271let trimmed = self.tcx.sess.source_map().span_through_char(subpat.span, final_char);
3272// The edition of the trimmed span should be the same as `subpat.span`; this will be a
3273 // a hard error if the subpattern is of edition >= 2024. We set it manually to be sure:
3274trimmed.with_ctxt(subpat.span.ctxt())
3275 };
32763277let mut typeck_results = self.typeck_results.borrow_mut();
3278let mut table = typeck_results.rust_2024_migration_desugared_pats_mut();
3279// FIXME(ref_pat_eat_one_layer_2024): The migration diagnostic doesn't know how to track the
3280 // default binding mode in the presence of Rule 3 or Rule 5. As a consequence, the labels it
3281 // gives for default binding modes are wrong, as well as suggestions based on the default
3282 // binding mode. This keeps it from making those suggestions, as doing so could panic.
3283let info = table.entry(pat_id).or_insert_with(|| ty::Rust2024IncompatiblePatInfo {
3284 primary_labels: Vec::new(),
3285 bad_ref_modifiers: false,
3286 bad_mut_modifiers: false,
3287 bad_ref_pats: false,
3288 suggest_eliding_modes: !self.tcx.features().ref_pat_eat_one_layer_2024()
3289 && !self.tcx.features().ref_pat_eat_one_layer_2024_structural(),
3290 });
32913292let pat_kind = if let PatKind::Binding(user_bind_annot, _, _, _) = subpat.kind {
3293// If the user-provided binding modifier doesn't match the default binding mode, we'll
3294 // need to suggest reference patterns, which can affect other bindings.
3295 // For simplicity, we opt to suggest making the pattern fully explicit.
3296info.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!(
3297 user_bind_annot,
3298 BindingMode(ByRef::Yes(_, mutbl), Mutability::Not) if mutbl == def_br_mutbl
3299 );
3300if user_bind_annot == BindingMode(ByRef::No, Mutability::Mut) {
3301info.bad_mut_modifiers = true;
3302"`mut` binding modifier"
3303} else {
3304info.bad_ref_modifiers = true;
3305match user_bind_annot.1 {
3306 Mutability::Not => "explicit `ref` binding modifier",
3307 Mutability::Mut => "explicit `ref mut` binding modifier",
3308 }
3309 }
3310 } else {
3311info.bad_ref_pats = true;
3312// For simplicity, we don't try to suggest eliding reference patterns. Thus, we'll
3313 // suggest adding them instead, which can affect the types assigned to bindings.
3314 // As such, we opt to suggest making the pattern fully explicit.
3315info.suggest_eliding_modes = false;
3316"reference pattern"
3317};
3318// Only provide a detailed label if the problematic subpattern isn't from an expansion.
3319 // In the case that it's from a macro, we'll add a more detailed note in the emitter.
3320let primary_label = if from_expansion {
3321// We can't suggest eliding modifiers within expansions.
3322info.suggest_eliding_modes = false;
3323// NB: This wording assumes the only expansions that can produce problematic reference
3324 // patterns and bindings are macros. If a desugaring or AST pass is added that can do
3325 // so, we may want to inspect the span's source callee or macro backtrace.
3326"occurs within macro expansion".to_owned()
3327 } else {
3328::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")3329 };
3330info.primary_labels.push((trimmed_span, primary_label));
3331 }
3332}