1use std::cmp;
2use std::collections::hash_map::Entry::{Occupied, Vacant};
34use rustc_abi::FieldIdx;
5use rustc_astas ast;
6use rustc_data_structures::fx::FxHashMap;
7use rustc_errors::codes::*;
8use rustc_errors::{
9Applicability, Diag, ErrorGuaranteed, MultiSpan, pluralize, struct_span_code_err,
10};
11use rustc_hir::def::{CtorKind, DefKind, Res};
12use rustc_hir::def_id::DefId;
13use rustc_hir::pat_util::EnumerateAndAdjustIterator;
14use rustc_hir::{
15selfas hir, BindingMode, ByRef, ExprKind, HirId, LangItem, Mutability, Pat, PatExpr,
16PatExprKind, PatKind, expr_needs_parens,
17};
18use rustc_hir_analysis::autoderef::report_autoderef_recursion_limit_error;
19use rustc_infer::infer::RegionVariableOrigin;
20use rustc_middle::traits::PatternOriginExpr;
21use rustc_middle::ty::{self, Pinnedness, Ty, TypeVisitableExt};
22use rustc_middle::{bug, span_bug};
23use rustc_session::lint::builtin::NON_EXHAUSTIVE_OMITTED_PATTERNS;
24use rustc_session::parse::feature_err;
25use rustc_span::edit_distance::find_best_match_for_name;
26use rustc_span::edition::Edition;
27use rustc_span::source_map::Spanned;
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_receiver_is_total_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_receiver_is_total_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_receiver_is_total_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_receiver_is_total_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_receiver_is_total_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(lt) => self.check_pat_lit(pat.span, lt, expected, &pat_info.top_info),
615 PatKind::Range(lhs, rhs, _) => {
616self.check_pat_range(pat.span, lhs, rhs, expected, &pat_info.top_info)
617 }
618 PatKind::Binding(ba, var_id, ident, sub) => {
619self.check_pat_ident(pat, ba, var_id, ident, sub, expected, pat_info)
620 }
621 PatKind::TupleStruct(ref qpath, subpats, ddpos) => match opt_path_res.unwrap() {
622Ok(ResolvedPat { ty, kind: ResolvedPatKind::TupleStruct { res, variant } }) => self623 .check_pat_tuple_struct(
624pat, qpath, subpats, ddpos, res, ty, variant, expected, pat_info,
625 ),
626Err(guar) => {
627let ty_err = Ty::new_error(self.tcx, guar);
628for subpat in subpats {
629self.check_pat(subpat, ty_err, pat_info);
630 }
631ty_err632 }
633Ok(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:?}"),
634 },
635 PatKind::Struct(_, fields, has_rest_pat) => match opt_path_res.unwrap() {
636Ok(ResolvedPat { ty, kind: ResolvedPatKind::Struct { variant } }) => self637 .check_pat_struct(
638pat,
639fields,
640has_rest_pat.is_some(),
641ty,
642variant,
643expected,
644pat_info,
645 ),
646Err(guar) => {
647let ty_err = Ty::new_error(self.tcx, guar);
648for field in fields {
649self.check_pat(field.pat, ty_err, pat_info);
650 }
651ty_err652 }
653Ok(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:?}"),
654 },
655 PatKind::Guard(pat, cond) => {
656self.check_pat(pat, expected, pat_info);
657self.check_expr_has_type_or_error(cond, self.tcx.types.bool, |_| {});
658expected659 }
660 PatKind::Or(pats) => {
661for pat in pats {
662self.check_pat(pat, expected, pat_info);
663 }
664expected665 }
666 PatKind::Tuple(elements, ddpos) => {
667self.check_pat_tuple(pat.span, elements, ddpos, expected, pat_info)
668 }
669 PatKind::Box(inner) => self.check_pat_box(pat.span, inner, expected, pat_info),
670 PatKind::Deref(inner) => self.check_pat_deref(pat.span, inner, expected, pat_info),
671 PatKind::Ref(inner, pinned, mutbl) => {
672self.check_pat_ref(pat, inner, pinned, mutbl, expected, pat_info)
673 }
674 PatKind::Slice(before, slice, after) => {
675self.check_pat_slice(pat.span, before, slice, after, expected, pat_info)
676 }
677 }
678 }
679680fn adjust_pat_info(
681&self,
682 inner_pinnedness: Pinnedness,
683 inner_mutability: Mutability,
684 pat_info: PatInfo<'tcx>,
685 ) -> PatInfo<'tcx> {
686let mut binding_mode = match pat_info.binding_mode {
687// If default binding mode is by value, make it `ref`, `ref mut`, `ref pin const`
688 // or `ref pin mut` (depending on whether we observe `&`, `&mut`, `&pin const` or
689 // `&pin mut`).
690ByRef::No => ByRef::Yes(inner_pinnedness, inner_mutability),
691 ByRef::Yes(pinnedness, mutability) => {
692let pinnedness = match pinnedness {
693// When `ref`, stay a `ref` (on `&`) or downgrade to `ref pin` (on `&pin`).
694Pinnedness::Not => inner_pinnedness,
695// When `ref pin`, stay a `ref pin`.
696 // This is because we cannot get an `&mut T` from `&mut &pin mut T` unless `T: Unpin`.
697 // Note that `&T` and `&mut T` are `Unpin`, which implies
698 // `& &pin const T` <-> `&pin const &T` and `&mut &pin mut T` <-> `&pin mut &mut T`
699 // (i.e. mutually coercible).
700Pinnedness::Pinned => Pinnedness::Pinned,
701 };
702703let mutability = match mutability {
704// When `ref mut`, stay a `ref mut` (on `&mut`) or downgrade to `ref` (on `&`).
705Mutability::Mut => inner_mutability,
706// Once a `ref`, always a `ref`.
707 // This is because a `& &mut` cannot mutate the underlying value.
708Mutability::Not => Mutability::Not,
709 };
710 ByRef::Yes(pinnedness, mutability)
711 }
712 };
713714let PatInfo { mut max_ref_mutbl, mut max_pinnedness, .. } = pat_info;
715if self.downgrade_mut_inside_shared() {
716binding_mode = binding_mode.cap_ref_mutability(max_ref_mutbl.as_mutbl());
717 }
718match binding_mode {
719 ByRef::Yes(_, Mutability::Not) => max_ref_mutbl = MutblCap::Not,
720 ByRef::Yes(Pinnedness::Pinned, _) => max_pinnedness = PinnednessCap::Pinned,
721_ => {}
722 }
723{
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:723",
"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(723u32),
::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);
724PatInfo { binding_mode, max_pinnedness, max_ref_mutbl, ..pat_info }
725 }
726727fn check_deref_pattern(
728&self,
729 pat: &'tcx Pat<'tcx>,
730 opt_path_res: Option<Result<ResolvedPat<'tcx>, ErrorGuaranteed>>,
731 adjust_mode: AdjustMode,
732 expected: Ty<'tcx>,
733mut inner_ty: Ty<'tcx>,
734 pat_adjust_kind: PatAdjust,
735 pat_info: PatInfo<'tcx>,
736 ) -> Ty<'tcx> {
737if 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!(
738 !matches!(pat_adjust_kind, PatAdjust::BuiltinDeref),
739"unexpected deref pattern for builtin reference type {expected:?}",
740 );
741742let mut typeck_results = self.typeck_results.borrow_mut();
743let mut pat_adjustments_table = typeck_results.pat_adjustments_mut();
744let pat_adjustments = pat_adjustments_table.entry(pat.hir_id).or_default();
745// We may reach the recursion limit if a user matches on a type `T` satisfying
746 // `T: Deref<Target = T>`; error gracefully in this case.
747 // FIXME(deref_patterns): If `deref_patterns` stabilizes, it may make sense to move
748 // this check out of this branch. Alternatively, this loop could be implemented with
749 // autoderef and this check removed. For now though, don't break code compiling on
750 // stable with lots of `&`s and a low recursion limit, if anyone's done that.
751if self.tcx.recursion_limit().value_within_limit(pat_adjustments.len()) {
752// Preserve the smart pointer type for THIR lowering and closure upvar analysis.
753pat_adjustments.push(PatAdjustment { kind: pat_adjust_kind, source: expected });
754 } else {
755let guar = report_autoderef_recursion_limit_error(self.tcx, pat.span, expected);
756inner_ty = Ty::new_error(self.tcx, guar);
757 }
758drop(typeck_results);
759760// Recurse, using the old pat info to keep `current_depth` to its old value.
761 // Peeling smart pointers does not update the default binding mode.
762self.check_pat_inner(pat, opt_path_res, adjust_mode, inner_ty, pat_info)
763 }
764765/// How should the binding mode and expected type be adjusted?
766 ///
767 /// When the pattern contains a path, `opt_path_res` must be `Some(path_res)`.
768fn calc_adjust_mode(
769&self,
770 pat: &'tcx Pat<'tcx>,
771 opt_path_res: Option<Result<ResolvedPat<'tcx>, ErrorGuaranteed>>,
772 ) -> AdjustMode {
773match &pat.kind {
774// Type checking these product-like types successfully always require
775 // that the expected type be of those types and not reference types.
776PatKind::Tuple(..) | PatKind::Range(..) | PatKind::Slice(..) => AdjustMode::peel_all(),
777// When checking an explicit deref pattern, only peel reference types.
778 // FIXME(deref_patterns): If box patterns and deref patterns need to coexist, box
779 // patterns may want `PeelKind::Implicit`, stopping on encountering a box.
780PatKind::Box(_) | PatKind::Deref(_) => {
781 AdjustMode::Peel { kind: PeelKind::ExplicitDerefPat }
782 }
783// A never pattern behaves somewhat like a literal or unit variant.
784PatKind::Never => AdjustMode::peel_all(),
785// For patterns with paths, how we peel the scrutinee depends on the path's resolution.
786PatKind::Struct(..)
787 | PatKind::TupleStruct(..)
788 | PatKind::Expr(PatExpr { kind: PatExprKind::Path(_), .. }) => {
789// If there was an error resolving the path, default to peeling everything.
790opt_path_res.unwrap().map_or(AdjustMode::peel_all(), |pr| pr.adjust_mode())
791 }
792793// String and byte-string literals result in types `&str` and `&[u8]` respectively.
794 // All other literals result in non-reference types.
795 // As a result, we allow `if let 0 = &&0 {}` but not `if let "foo" = &&"foo" {}` unless
796 // `deref_patterns` is enabled.
797PatKind::Expr(lt) => {
798// Path patterns have already been handled, and inline const blocks currently
799 // aren't possible to write, so any handling for them would be untested.
800if truecfg!(debug_assertions)801 && self.tcx.features().deref_patterns()
802 && !#[allow(non_exhaustive_omitted_patterns)] match lt.kind {
PatExprKind::Lit { .. } => true,
_ => false,
}matches!(lt.kind, PatExprKind::Lit { .. })803 {
804::rustc_middle::util::bug::span_bug_fmt(lt.span,
format_args!("FIXME(deref_patterns): adjust mode unimplemented for {0:?}",
lt.kind));span_bug!(
805lt.span,
806"FIXME(deref_patterns): adjust mode unimplemented for {:?}",
807 lt.kind
808 );
809 }
810// Call `resolve_vars_if_possible` here for inline const blocks.
811let lit_ty = self.resolve_vars_if_possible(self.check_pat_expr_unadjusted(lt));
812// If `deref_patterns` is enabled, allow `if let "foo" = &&"foo" {}`.
813if self.tcx.features().deref_patterns() {
814let mut peeled_ty = lit_ty;
815let mut pat_ref_layers = 0;
816while let ty::Ref(_, inner_ty, mutbl) =
817*self.try_structurally_resolve_type(pat.span, peeled_ty).kind()
818 {
819// We rely on references at the head of constants being immutable.
820if true {
if !mutbl.is_not() {
::core::panicking::panic("assertion failed: mutbl.is_not()")
};
};debug_assert!(mutbl.is_not());
821 pat_ref_layers += 1;
822 peeled_ty = inner_ty;
823 }
824 AdjustMode::Peel {
825 kind: PeelKind::Implicit { until_adt: None, pat_ref_layers },
826 }
827 } else {
828if lit_ty.is_ref() { AdjustMode::Pass } else { AdjustMode::peel_all() }
829 }
830 }
831832// Ref patterns are complicated, we handle them in `check_pat_ref`.
833PatKind::Ref(..)
834// No need to do anything on a missing pattern.
835| PatKind::Missing836// A `_` pattern works with any expected type, so there's no need to do anything.
837| PatKind::Wild838// A malformed pattern doesn't have an expected type, so let's just accept any type.
839| PatKind::Err(_)
840// Bindings also work with whatever the expected type is,
841 // and moreover if we peel references off, that will give us the wrong binding type.
842 // Also, we can have a subpattern `binding @ pat`.
843 // Each side of the `@` should be treated independently (like with OR-patterns).
844| PatKind::Binding(..)
845// An OR-pattern just propagates to each individual alternative.
846 // This is maximally flexible, allowing e.g., `Some(mut x) | &Some(mut x)`.
847 // In that example, `Some(mut x)` results in `Peel` whereas `&Some(mut x)` in `Reset`.
848| PatKind::Or(_)
849// Like or-patterns, guard patterns just propagate to their subpatterns.
850| PatKind::Guard(..) => AdjustMode::Pass,
851 }
852 }
853854/// Assuming `expected` is a reference type, determine whether to peel it before matching.
855fn should_peel_ref(&self, peel_kind: PeelKind, mut expected: Ty<'tcx>) -> bool {
856if true {
if !expected.is_ref() {
::core::panicking::panic("assertion failed: expected.is_ref()")
};
};debug_assert!(expected.is_ref());
857let pat_ref_layers = match peel_kind {
858 PeelKind::ExplicitDerefPat => 0,
859 PeelKind::Implicit { pat_ref_layers, .. } => pat_ref_layers,
860 };
861862// Most patterns don't have reference types, so we'll want to peel all references from the
863 // scrutinee before matching. To optimize for the common case, return early.
864if pat_ref_layers == 0 {
865return true;
866 }
867if 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!(
868self.tcx.features().deref_patterns(),
869"Peeling for patterns with reference types is gated by `deref_patterns`."
870);
871872// If the pattern has as many or more layers of reference as the expected type, we can match
873 // without peeling more, unless we find a smart pointer or `&mut` that we also need to peel.
874 // We don't treat `&` and `&mut` as interchangeable, but by peeling `&mut`s before matching,
875 // we can still, e.g., match on a `&mut str` with a string literal pattern. This is because
876 // string literal patterns may be used where `str` is expected.
877let mut expected_ref_layers = 0;
878while let ty::Ref(_, inner_ty, mutbl) = *expected.kind() {
879if mutbl.is_mut() {
880// Mutable references can't be in the final value of constants, thus they can't be
881 // at the head of their types, thus we should always peel `&mut`.
882return true;
883 }
884 expected_ref_layers += 1;
885 expected = inner_ty;
886 }
887pat_ref_layers < expected_ref_layers || self.should_peel_smart_pointer(peel_kind, expected)
888 }
889890/// Determine whether `expected` is a smart pointer type that should be peeled before matching.
891fn should_peel_smart_pointer(&self, peel_kind: PeelKind, expected: Ty<'tcx>) -> bool {
892// Explicit `deref!(_)` patterns match against smart pointers; don't peel in that case.
893if let PeelKind::Implicit { until_adt, .. } = peel_kind894// For simplicity, only apply overloaded derefs if `expected` is a known ADT.
895 // FIXME(deref_patterns): we'll get better diagnostics for users trying to
896 // implicitly deref generics if we allow them here, but primitives, tuples, and
897 // inference vars definitely should be stopped. Figure out what makes most sense.
898&& let ty::Adt(scrutinee_adt, _) = *expected.kind()
899// Don't peel if the pattern type already matches the scrutinee. E.g., stop here if
900 // matching on a `Cow<'a, T>` scrutinee with a `Cow::Owned(_)` pattern.
901&& until_adt != Some(scrutinee_adt.did())
902// At this point, the pattern isn't able to match `expected` without peeling. Check
903 // that it implements `Deref` before assuming it's a smart pointer, to get a normal
904 // type error instead of a missing impl error if not. This only checks for `Deref`,
905 // not `DerefPure`: we require that too, but we want a trait error if it's missing.
906&& let Some(deref_trait) = self.tcx.lang_items().deref_trait()
907 && self.type_implements_trait(deref_trait, [expected], self.param_env).may_apply()
908 {
909true
910} else {
911false
912}
913 }
914915fn check_pat_expr_unadjusted(&self, lt: &'tcx hir::PatExpr<'tcx>) -> Ty<'tcx> {
916let ty = match <.kind {
917 rustc_hir::PatExprKind::Lit { lit, negated } => {
918let ty = self.check_expr_lit(lit, Expectation::NoExpectation);
919if *negated {
920self.register_bound(
921ty,
922self.tcx.require_lang_item(LangItem::Neg, lt.span),
923ObligationCause::dummy_with_span(lt.span),
924 );
925 }
926ty927 }
928 rustc_hir::PatExprKind::Path(qpath) => {
929let (res, opt_ty, segments) =
930self.resolve_ty_and_res_fully_qualified_call(qpath, lt.hir_id, lt.span);
931self.instantiate_value_path(segments, opt_ty, res, lt.span, lt.span, lt.hir_id).0
932}
933 };
934self.write_ty(lt.hir_id, ty);
935ty936 }
937938fn check_pat_lit(
939&self,
940 span: Span,
941 lt: &hir::PatExpr<'tcx>,
942 expected: Ty<'tcx>,
943 ti: &TopInfo<'tcx>,
944 ) -> Ty<'tcx> {
945// We've already computed the type above (when checking for a non-ref pat),
946 // so avoid computing it again.
947let ty = self.node_ty(lt.hir_id);
948949// Byte string patterns behave the same way as array patterns
950 // They can denote both statically and dynamically-sized byte arrays.
951 // Additionally, when `deref_patterns` is enabled, byte string literal patterns may have
952 // types `[u8]` or `[u8; N]`, in order to type, e.g., `deref!(b"..."): Vec<u8>`.
953let mut pat_ty = ty;
954if let hir::PatExprKind::Lit {
955 lit: Spanned { node: ast::LitKind::ByteStr(..), .. }, ..
956 } = lt.kind
957 {
958let tcx = self.tcx;
959let expected = self.structurally_resolve_type(span, expected);
960match *expected.kind() {
961// Allow `b"...": &[u8]`
962ty::Ref(_, inner_ty, _)
963if self.try_structurally_resolve_type(span, inner_ty).is_slice() =>
964 {
965{
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:965",
"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(965u32),
::tracing_core::__macro_support::Option::Some("rustc_hir_typeck::pat"),
::tracing_core::field::FieldSet::new(&["message",
"lt.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(<.hir_id.local_id)
as &dyn Value))])
});
} else { ; }
};trace!(?lt.hir_id.local_id, "polymorphic byte string lit");
966pat_ty = Ty::new_imm_ref(
967tcx,
968tcx.lifetimes.re_static,
969Ty::new_slice(tcx, tcx.types.u8),
970 );
971 }
972// Allow `b"...": [u8; 3]` for `deref_patterns`
973ty::Array(..) if tcx.features().deref_patterns() => {
974pat_ty = match *ty.kind() {
975 ty::Ref(_, inner_ty, _) => inner_ty,
976_ => ::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:?}"),
977 }
978 }
979// Allow `b"...": [u8]` for `deref_patterns`
980ty::Slice(..) if tcx.features().deref_patterns() => {
981pat_ty = Ty::new_slice(tcx, tcx.types.u8);
982 }
983// Otherwise, `b"...": &[u8; 3]`
984_ => {}
985 }
986 }
987988// When `deref_patterns` is enabled, in order to allow `deref!("..."): String`, we allow
989 // string literal patterns to have type `str`. This is accounted for when lowering to MIR.
990if self.tcx.features().deref_patterns()
991 && let hir::PatExprKind::Lit {
992 lit: Spanned { node: ast::LitKind::Str(..), .. }, ..
993 } = lt.kind
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_verbose(
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<[_]>::into_vec(::alloc::boxed::box_new([(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::Const1571 | DefKind::AssocConst1572 | 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) => (Some(def.did()), Some(def_id)),
1664_ => (None, None),
1665 },
1666_ => (None, None),
1667 };
16681669let 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!(
1670 type_def_id.and_then(|id| self.tcx.as_lang_item(id)),
1671Some(
1672 LangItem::Range
1673 | LangItem::RangeFrom
1674 | LangItem::RangeTo
1675 | LangItem::RangeFull
1676 | LangItem::RangeInclusiveStruct
1677 | LangItem::RangeToInclusive,
1678 )
1679 );
1680if is_range {
1681if !self.maybe_suggest_range_literal(&mut e, item_def_id, *ident) {
1682let msg = "constants only support matching by type, \
1683 if you meant to match against a range of values, \
1684 consider using a range pattern like `min ..= max` in the match block";
1685e.note(msg);
1686 }
1687 } else {
1688let msg = "introduce a new binding instead";
1689let sugg = ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("other_{0}",
ident.as_str().to_lowercase()))
})format!("other_{}", ident.as_str().to_lowercase());
1690e.span_suggestion(
1691ident.span,
1692msg,
1693sugg,
1694 Applicability::HasPlaceholders,
1695 );
1696 }
1697 }
1698 };
1699 }
1700 }
1701e.emit();
1702 }
17031704fn resolve_pat_tuple_struct(
1705&self,
1706 pat: &'tcx Pat<'tcx>,
1707 qpath: &'tcx hir::QPath<'tcx>,
1708 ) -> Result<ResolvedPat<'tcx>, ErrorGuaranteed> {
1709let tcx = self.tcx;
1710let report_unexpected_res = |res: Res| {
1711let expected = "tuple struct or tuple variant";
1712let e = report_unexpected_variant_res(tcx, res, None, qpath, pat.span, E0164, expected);
1713Err(e)
1714 };
17151716// Resolve the path and check the definition for errors.
1717let (res, opt_ty, segments) =
1718self.resolve_ty_and_res_fully_qualified_call(qpath, pat.hir_id, pat.span);
1719if res == Res::Err {
1720let e = self.dcx().span_delayed_bug(pat.span, "`Res::Err` but no error emitted");
1721self.set_tainted_by_errors(e);
1722return Err(e);
1723 }
17241725// Type-check the path.
1726let (pat_ty, res) =
1727self.instantiate_value_path(segments, opt_ty, res, pat.span, pat.span, pat.hir_id);
1728if !pat_ty.is_fn() {
1729return report_unexpected_res(res);
1730 }
17311732let variant = match res {
1733 Res::Err => {
1734self.dcx().span_bug(pat.span, "`Res::Err` but no error emitted");
1735 }
1736 Res::Def(DefKind::AssocConst | DefKind::AssocFn, _) => {
1737return report_unexpected_res(res);
1738 }
1739 Res::Def(DefKind::Ctor(_, CtorKind::Fn), _) => tcx.expect_variant_res(res),
1740_ => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern resolution: {0:?}",
res))bug!("unexpected pattern resolution: {:?}", res),
1741 };
17421743// Replace constructor type with constructed type for tuple struct patterns.
1744let pat_ty = pat_ty.fn_sig(tcx).output();
1745let pat_ty = pat_ty.no_bound_vars().expect("expected fn type");
17461747Ok(ResolvedPat { ty: pat_ty, kind: ResolvedPatKind::TupleStruct { res, variant } })
1748 }
17491750fn check_pat_tuple_struct(
1751&self,
1752 pat: &'tcx Pat<'tcx>,
1753 qpath: &'tcx hir::QPath<'tcx>,
1754 subpats: &'tcx [Pat<'tcx>],
1755 ddpos: hir::DotDotPos,
1756 res: Res,
1757 pat_ty: Ty<'tcx>,
1758 variant: &'tcx VariantDef,
1759 expected: Ty<'tcx>,
1760 pat_info: PatInfo<'tcx>,
1761 ) -> Ty<'tcx> {
1762let tcx = self.tcx;
1763let on_error = |e| {
1764for pat in subpats {
1765self.check_pat(pat, Ty::new_error(tcx, e), pat_info);
1766 }
1767 };
17681769// Type-check the tuple struct pattern against the expected type.
1770let had_err = self.demand_eqtype_pat(pat.span, expected, pat_ty, &pat_info.top_info);
17711772// Type-check subpatterns.
1773if subpats.len() == variant.fields.len()
1774 || subpats.len() < variant.fields.len() && ddpos.as_opt_usize().is_some()
1775 {
1776let ty::Adt(_, args) = pat_ty.kind() else {
1777::rustc_middle::util::bug::bug_fmt(format_args!("unexpected pattern type {0:?}",
pat_ty));bug!("unexpected pattern type {:?}", pat_ty);
1778 };
1779for (i, subpat) in subpats.iter().enumerate_and_adjust(variant.fields.len(), ddpos) {
1780let field = &variant.fields[FieldIdx::from_usize(i)];
1781let field_ty = self.field_ty(subpat.span, field, args);
1782self.check_pat(subpat, field_ty, pat_info);
17831784self.tcx.check_stability(
1785 variant.fields[FieldIdx::from_usize(i)].did,
1786Some(subpat.hir_id),
1787 subpat.span,
1788None,
1789 );
1790 }
1791if let Err(e) = had_err {
1792on_error(e);
1793return Ty::new_error(tcx, e);
1794 }
1795 } else {
1796let e = self.emit_err_pat_wrong_number_of_fields(
1797pat.span,
1798res,
1799qpath,
1800subpats,
1801&variant.fields.raw,
1802expected,
1803had_err,
1804 );
1805on_error(e);
1806return Ty::new_error(tcx, e);
1807 }
1808pat_ty1809 }
18101811fn emit_err_pat_wrong_number_of_fields(
1812&self,
1813 pat_span: Span,
1814 res: Res,
1815 qpath: &hir::QPath<'_>,
1816 subpats: &'tcx [Pat<'tcx>],
1817 fields: &'tcx [ty::FieldDef],
1818 expected: Ty<'tcx>,
1819 had_err: Result<(), ErrorGuaranteed>,
1820 ) -> ErrorGuaranteed {
1821let subpats_ending = if subpats.len() == 1 { "" } else { "s" }pluralize!(subpats.len());
1822let fields_ending = if fields.len() == 1 { "" } else { "s" }pluralize!(fields.len());
18231824let subpat_spans = if subpats.is_empty() {
1825<[_]>::into_vec(::alloc::boxed::box_new([pat_span]))vec![pat_span]1826 } else {
1827subpats.iter().map(|p| p.span).collect()
1828 };
1829let last_subpat_span = *subpat_spans.last().unwrap();
1830let res_span = self.tcx.def_span(res.def_id());
1831let def_ident_span = self.tcx.def_ident_span(res.def_id()).unwrap_or(res_span);
1832let field_def_spans = if fields.is_empty() {
1833<[_]>::into_vec(::alloc::boxed::box_new([res_span]))vec![res_span]1834 } else {
1835fields.iter().map(|f| f.ident(self.tcx).span).collect()
1836 };
1837let last_field_def_span = *field_def_spans.last().unwrap();
18381839let 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!(
1840self.dcx(),
1841 MultiSpan::from_spans(subpat_spans),
1842 E0023,
1843"this pattern has {} field{}, but the corresponding {} has {} field{}",
1844 subpats.len(),
1845 subpats_ending,
1846 res.descr(),
1847 fields.len(),
1848 fields_ending,
1849 );
1850err.span_label(
1851last_subpat_span,
1852::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()),
1853 );
1854if self.tcx.sess.source_map().is_multiline(qpath.span().between(last_subpat_span)) {
1855err.span_label(qpath.span(), "");
1856 }
1857if self.tcx.sess.source_map().is_multiline(def_ident_span.between(last_field_def_span)) {
1858err.span_label(def_ident_span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0} defined here", res.descr()))
})format!("{} defined here", res.descr()));
1859 }
1860for span in &field_def_spans[..field_def_spans.len() - 1] {
1861 err.span_label(*span, "");
1862 }
1863err.span_label(
1864last_field_def_span,
1865::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),
1866 );
18671868// Identify the case `Some(x, y)` where the expected type is e.g. `Option<(T, U)>`.
1869 // More generally, the expected type wants a tuple variant with one field of an
1870 // N-arity-tuple, e.g., `V_i((p_0, .., p_N))`. Meanwhile, the user supplied a pattern
1871 // with the subpatterns directly in the tuple variant pattern, e.g., `V_i(p_0, .., p_N)`.
1872let missing_parentheses = match (expected.kind(), fields, had_err) {
1873// #67037: only do this if we could successfully type-check the expected type against
1874 // the tuple struct pattern. Otherwise the args could get out of range on e.g.,
1875 // `let P() = U;` where `P != U` with `struct Box<T>(T);`.
1876(ty::Adt(_, args), [field], Ok(())) => {
1877let field_ty = self.field_ty(pat_span, field, args);
1878match field_ty.kind() {
1879 ty::Tuple(fields) => fields.len() == subpats.len(),
1880_ => false,
1881 }
1882 }
1883_ => false,
1884 };
1885if missing_parentheses {
1886let (left, right) = match subpats {
1887// This is the zero case; we aim to get the "hi" part of the `QPath`'s
1888 // span as the "lo" and then the "hi" part of the pattern's span as the "hi".
1889 // This looks like:
1890 //
1891 // help: missing parentheses
1892 // |
1893 // L | let A(()) = A(());
1894 // | ^ ^
1895[] => (qpath.span().shrink_to_hi(), pat_span),
1896// Easy case. Just take the "lo" of the first sub-pattern and the "hi" of the
1897 // last sub-pattern. In the case of `A(x)` the first and last may coincide.
1898 // This looks like:
1899 //
1900 // help: missing parentheses
1901 // |
1902 // L | let A((x, y)) = A((1, 2));
1903 // | ^ ^
1904[first, ..] => (first.span.shrink_to_lo(), subpats.last().unwrap().span),
1905 };
1906err.multipart_suggestion(
1907"missing parentheses",
1908<[_]>::into_vec(::alloc::boxed::box_new([(left, "(".to_string()),
(right.shrink_to_hi(), ")".to_string())]))vec![(left, "(".to_string()), (right.shrink_to_hi(), ")".to_string())],
1909 Applicability::MachineApplicable,
1910 );
1911 } else if fields.len() > subpats.len() && pat_span != DUMMY_SP {
1912let after_fields_span = pat_span.with_hi(pat_span.hi() - BytePos(1)).shrink_to_hi();
1913let all_fields_span = match subpats {
1914 [] => after_fields_span,
1915 [field] => field.span,
1916 [first, .., last] => first.span.to(last.span),
1917 };
19181919// Check if all the fields in the pattern are wildcards.
1920let all_wildcards = subpats.iter().all(|pat| #[allow(non_exhaustive_omitted_patterns)] match pat.kind {
PatKind::Wild => true,
_ => false,
}matches!(pat.kind, PatKind::Wild));
1921let first_tail_wildcard =
1922subpats.iter().enumerate().fold(None, |acc, (pos, pat)| match (acc, &pat.kind) {
1923 (None, PatKind::Wild) => Some(pos),
1924 (Some(_), PatKind::Wild) => acc,
1925_ => None,
1926 });
1927let tail_span = match first_tail_wildcard {
1928None => after_fields_span,
1929Some(0) => subpats[0].span.to(after_fields_span),
1930Some(pos) => subpats[pos - 1].span.shrink_to_hi().to(after_fields_span),
1931 };
19321933// FIXME: heuristic-based suggestion to check current types for where to add `_`.
1934let mut wildcard_sugg = ::alloc::vec::from_elem("_", fields.len() - subpats.len())vec!["_"; fields.len() - subpats.len()].join(", ");
1935if !subpats.is_empty() {
1936wildcard_sugg = String::from(", ") + &wildcard_sugg;
1937 }
19381939err.span_suggestion_verbose(
1940after_fields_span,
1941"use `_` to explicitly ignore each field",
1942wildcard_sugg,
1943 Applicability::MaybeIncorrect,
1944 );
19451946// Only suggest `..` if more than one field is missing
1947 // or the pattern consists of all wildcards.
1948if fields.len() - subpats.len() > 1 || all_wildcards {
1949if subpats.is_empty() || all_wildcards {
1950err.span_suggestion_verbose(
1951all_fields_span,
1952"use `..` to ignore all fields",
1953"..",
1954 Applicability::MaybeIncorrect,
1955 );
1956 } else {
1957err.span_suggestion_verbose(
1958tail_span,
1959"use `..` to ignore the rest of the fields",
1960", ..",
1961 Applicability::MaybeIncorrect,
1962 );
1963 }
1964 }
1965 }
19661967err.emit()
1968 }
19691970fn check_pat_tuple(
1971&self,
1972 span: Span,
1973 elements: &'tcx [Pat<'tcx>],
1974 ddpos: hir::DotDotPos,
1975 expected: Ty<'tcx>,
1976 pat_info: PatInfo<'tcx>,
1977 ) -> Ty<'tcx> {
1978let tcx = self.tcx;
1979let mut expected_len = elements.len();
1980if ddpos.as_opt_usize().is_some() {
1981// Require known type only when `..` is present.
1982if let ty::Tuple(tys) = self.structurally_resolve_type(span, expected).kind() {
1983expected_len = tys.len();
1984 }
1985 }
1986let max_len = cmp::max(expected_len, elements.len());
19871988let element_tys_iter = (0..max_len).map(|_| self.next_ty_var(span));
1989let element_tys = tcx.mk_type_list_from_iter(element_tys_iter);
1990let pat_ty = Ty::new_tup(tcx, element_tys);
1991if let Err(reported) = self.demand_eqtype_pat(span, expected, pat_ty, &pat_info.top_info) {
1992// Walk subpatterns with an expected type of `err` in this case to silence
1993 // further errors being emitted when using the bindings. #50333
1994for (_, elem) in elements.iter().enumerate_and_adjust(max_len, ddpos) {
1995self.check_pat(elem, Ty::new_error(tcx, reported), pat_info);
1996 }
1997Ty::new_error(tcx, reported)
1998 } else {
1999for (i, elem) in elements.iter().enumerate_and_adjust(max_len, ddpos) {
2000self.check_pat(elem, element_tys[i], pat_info);
2001 }
2002pat_ty2003 }
2004 }
20052006fn check_struct_pat_fields(
2007&self,
2008 adt_ty: Ty<'tcx>,
2009 pat: &'tcx Pat<'tcx>,
2010 variant: &'tcx ty::VariantDef,
2011 fields: &'tcx [hir::PatField<'tcx>],
2012 has_rest_pat: bool,
2013 pat_info: PatInfo<'tcx>,
2014 ) -> Result<(), ErrorGuaranteed> {
2015let tcx = self.tcx;
20162017let ty::Adt(adt, args) = adt_ty.kind() else {
2018::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");
2019 };
20202021// Index the struct fields' types.
2022let field_map = variant2023 .fields
2024 .iter_enumerated()
2025 .map(|(i, field)| (field.ident(self.tcx).normalize_to_macros_2_0(), (i, field)))
2026 .collect::<FxHashMap<_, _>>();
20272028// Keep track of which fields have already appeared in the pattern.
2029let mut used_fields = FxHashMap::default();
2030let mut result = Ok(());
20312032let mut inexistent_fields = ::alloc::vec::Vec::new()vec![];
2033// Typecheck each field.
2034for field in fields {
2035let span = field.span;
2036let ident = tcx.adjust_ident(field.ident, variant.def_id);
2037let field_ty = match used_fields.entry(ident) {
2038 Occupied(occupied) => {
2039let guar = self.error_field_already_bound(span, field.ident, *occupied.get());
2040 result = Err(guar);
2041 Ty::new_error(tcx, guar)
2042 }
2043 Vacant(vacant) => {
2044 vacant.insert(span);
2045 field_map
2046 .get(&ident)
2047 .map(|(i, f)| {
2048self.write_field_index(field.hir_id, *i);
2049self.tcx.check_stability(f.did, Some(field.hir_id), span, None);
2050self.field_ty(span, f, args)
2051 })
2052 .unwrap_or_else(|| {
2053 inexistent_fields.push(field);
2054 Ty::new_misc_error(tcx)
2055 })
2056 }
2057 };
20582059self.check_pat(field.pat, field_ty, pat_info);
2060 }
20612062let mut unmentioned_fields = variant2063 .fields
2064 .iter()
2065 .map(|field| (field, field.ident(self.tcx).normalize_to_macros_2_0()))
2066 .filter(|(_, ident)| !used_fields.contains_key(ident))
2067 .collect::<Vec<_>>();
20682069let inexistent_fields_err = if !inexistent_fields.is_empty()
2070 && !inexistent_fields.iter().any(|field| field.ident.name == kw::Underscore)
2071 {
2072// we don't care to report errors for a struct if the struct itself is tainted
2073variant.has_errors()?;
2074Some(self.error_inexistent_fields(
2075adt.variant_descr(),
2076&inexistent_fields,
2077&mut unmentioned_fields,
2078pat,
2079variant,
2080args,
2081 ))
2082 } else {
2083None2084 };
20852086// Require `..` if struct has non_exhaustive attribute.
2087let non_exhaustive = variant.field_list_has_applicable_non_exhaustive();
2088if non_exhaustive && !has_rest_pat {
2089self.error_foreign_non_exhaustive_spat(pat, adt.variant_descr(), fields.is_empty());
2090 }
20912092let mut unmentioned_err = None;
2093// Report an error if an incorrect number of fields was specified.
2094if adt.is_union() {
2095if fields.len() != 1 {
2096self.dcx().emit_err(errors::UnionPatMultipleFields { span: pat.span });
2097 }
2098if has_rest_pat {
2099self.dcx().emit_err(errors::UnionPatDotDot { span: pat.span });
2100 }
2101 } else if !unmentioned_fields.is_empty() {
2102let accessible_unmentioned_fields: Vec<_> = unmentioned_fields2103 .iter()
2104 .copied()
2105 .filter(|(field, _)| self.is_field_suggestable(field, pat.hir_id, pat.span))
2106 .collect();
21072108if !has_rest_pat {
2109if accessible_unmentioned_fields.is_empty() {
2110unmentioned_err = Some(self.error_no_accessible_fields(pat, fields));
2111 } else {
2112unmentioned_err = Some(self.error_unmentioned_fields(
2113pat,
2114&accessible_unmentioned_fields,
2115accessible_unmentioned_fields.len() != unmentioned_fields.len(),
2116fields,
2117 ));
2118 }
2119 } else if non_exhaustive && !accessible_unmentioned_fields.is_empty() {
2120self.lint_non_exhaustive_omitted_patterns(
2121pat,
2122&accessible_unmentioned_fields,
2123adt_ty,
2124 )
2125 }
2126 }
2127match (inexistent_fields_err, unmentioned_err) {
2128 (Some(i), Some(u)) => {
2129if let Err(e) = self.error_tuple_variant_as_struct_pat(pat, fields, variant) {
2130// We don't want to show the nonexistent fields error when this was
2131 // `Foo { a, b }` when it should have been `Foo(a, b)`.
2132i.delay_as_bug();
2133u.delay_as_bug();
2134Err(e)
2135 } else {
2136i.emit();
2137Err(u.emit())
2138 }
2139 }
2140 (None, Some(u)) => {
2141if let Err(e) = self.error_tuple_variant_as_struct_pat(pat, fields, variant) {
2142u.delay_as_bug();
2143Err(e)
2144 } else {
2145Err(u.emit())
2146 }
2147 }
2148 (Some(err), None) => Err(err.emit()),
2149 (None, None) => {
2150self.error_tuple_variant_index_shorthand(variant, pat, fields)?;
2151result2152 }
2153 }
2154 }
21552156fn error_tuple_variant_index_shorthand(
2157&self,
2158 variant: &VariantDef,
2159 pat: &'_ Pat<'_>,
2160 fields: &[hir::PatField<'_>],
2161 ) -> Result<(), ErrorGuaranteed> {
2162// if this is a tuple struct, then all field names will be numbers
2163 // so if any fields in a struct pattern use shorthand syntax, they will
2164 // be invalid identifiers (for example, Foo { 0, 1 }).
2165if let (Some(CtorKind::Fn), PatKind::Struct(qpath, field_patterns, ..)) =
2166 (variant.ctor_kind(), &pat.kind)
2167 {
2168let has_shorthand_field_name = field_patterns.iter().any(|field| field.is_shorthand);
2169if has_shorthand_field_name {
2170let path = rustc_hir_pretty::qpath_to_string(&self.tcx, qpath);
2171let 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!(
2172self.dcx(),
2173 pat.span,
2174 E0769,
2175"tuple variant `{path}` written as struct variant",
2176 );
2177err.span_suggestion_verbose(
2178qpath.span().shrink_to_hi().to(pat.span.shrink_to_hi()),
2179"use the tuple variant pattern syntax instead",
2180::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)),
2181 Applicability::MaybeIncorrect,
2182 );
2183return Err(err.emit());
2184 }
2185 }
2186Ok(())
2187 }
21882189fn error_foreign_non_exhaustive_spat(&self, pat: &Pat<'_>, descr: &str, no_fields: bool) {
2190let sess = self.tcx.sess;
2191let sm = sess.source_map();
2192let sp_brace = sm.end_point(pat.span);
2193let sp_comma = sm.end_point(pat.span.with_hi(sp_brace.hi()));
2194let sugg = if no_fields || sp_brace != sp_comma { ".. }" } else { ", .. }" };
21952196{
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!(
2197self.dcx(),
2198 pat.span,
2199 E0638,
2200"`..` required with {descr} marked as non-exhaustive",
2201 )2202 .with_span_suggestion_verbose(
2203sp_comma,
2204"add `..` at the end of the field list to ignore all other fields",
2205sugg,
2206 Applicability::MachineApplicable,
2207 )
2208 .emit();
2209 }
22102211fn error_field_already_bound(
2212&self,
2213 span: Span,
2214 ident: Ident,
2215 other_field: Span,
2216 ) -> ErrorGuaranteed {
2217{
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!(
2218self.dcx(),
2219 span,
2220 E0025,
2221"field `{}` bound multiple times in the pattern",
2222 ident
2223 )2224 .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"))
2225 .with_span_label(other_field, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("first use of `{0}`", ident))
})format!("first use of `{ident}`"))
2226 .emit()
2227 }
22282229fn error_inexistent_fields(
2230&self,
2231 kind_name: &str,
2232 inexistent_fields: &[&hir::PatField<'tcx>],
2233 unmentioned_fields: &mut Vec<(&'tcx ty::FieldDef, Ident)>,
2234 pat: &'tcx Pat<'tcx>,
2235 variant: &ty::VariantDef,
2236 args: ty::GenericArgsRef<'tcx>,
2237 ) -> Diag<'a> {
2238let tcx = self.tcx;
2239let (field_names, t, plural) = if let [field] = inexistent_fields {
2240 (::alloc::__export::must_use({
::alloc::fmt::format(format_args!("a field named `{0}`", field.ident))
})format!("a field named `{}`", field.ident), "this", "")
2241 } else {
2242 (
2243::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!(
2244"fields named {}",
2245 inexistent_fields
2246 .iter()
2247 .map(|field| format!("`{}`", field.ident))
2248 .collect::<Vec<String>>()
2249 .join(", ")
2250 ),
2251"these",
2252"s",
2253 )
2254 };
2255let spans = inexistent_fields.iter().map(|field| field.ident.span).collect::<Vec<_>>();
2256let 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!(
2257self.dcx(),
2258 spans,
2259 E0026,
2260"{} `{}` does not have {}",
2261 kind_name,
2262 tcx.def_path_str(variant.def_id),
2263 field_names
2264 );
2265if let Some(pat_field) = inexistent_fields.last() {
2266err.span_label(
2267pat_field.ident.span,
2268::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!(
2269"{} `{}` does not have {} field{}",
2270 kind_name,
2271 tcx.def_path_str(variant.def_id),
2272 t,
2273 plural
2274 ),
2275 );
22762277if let [(field_def, field)] = unmentioned_fields.as_slice()
2278 && self.is_field_suggestable(field_def, pat.hir_id, pat.span)
2279 {
2280let suggested_name =
2281find_best_match_for_name(&[field.name], pat_field.ident.name, None);
2282if let Some(suggested_name) = suggested_name {
2283err.span_suggestion_verbose(
2284pat_field.ident.span,
2285"a field with a similar name exists",
2286suggested_name,
2287 Applicability::MaybeIncorrect,
2288 );
22892290// When we have a tuple struct used with struct we don't want to suggest using
2291 // the (valid) struct syntax with numeric field names. Instead we want to
2292 // suggest the expected syntax. We infer that this is the case by parsing the
2293 // `Ident` into an unsized integer. The suggestion will be emitted elsewhere in
2294 // `smart_resolve_context_dependent_help`.
2295if suggested_name.to_ident_string().parse::<usize>().is_err() {
2296// We don't want to throw `E0027` in case we have thrown `E0026` for them.
2297unmentioned_fields.retain(|&(_, x)| x.name != suggested_name);
2298 }
2299 } else if inexistent_fields.len() == 1 {
2300match pat_field.pat.kind {
2301 PatKind::Expr(_)
2302if !self.may_coerce(
2303self.typeck_results.borrow().node_type(pat_field.pat.hir_id),
2304self.field_ty(field.span, field_def, args),
2305 ) => {}
2306_ => {
2307err.span_suggestion_short(
2308pat_field.ident.span,
2309::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!(
2310"`{}` has a field named `{}`",
2311 tcx.def_path_str(variant.def_id),
2312 field.name,
2313 ),
2314field.name,
2315 Applicability::MaybeIncorrect,
2316 );
2317 }
2318 }
2319 }
2320 }
2321 }
2322if tcx.sess.teach(err.code.unwrap()) {
2323err.note(
2324"This error indicates that a struct pattern attempted to \
2325 extract a nonexistent field from a struct. Struct fields \
2326 are identified by the name used before the colon : so struct \
2327 patterns should resemble the declaration of the struct type \
2328 being matched.\n\n\
2329 If you are using shorthand field patterns but want to refer \
2330 to the struct field by a different name, you should rename \
2331 it explicitly.",
2332 );
2333 }
2334err2335 }
23362337fn error_tuple_variant_as_struct_pat(
2338&self,
2339 pat: &Pat<'_>,
2340 fields: &'tcx [hir::PatField<'tcx>],
2341 variant: &ty::VariantDef,
2342 ) -> Result<(), ErrorGuaranteed> {
2343if let (Some(CtorKind::Fn), PatKind::Struct(qpath, pattern_fields, ..)) =
2344 (variant.ctor_kind(), &pat.kind)
2345 {
2346let is_tuple_struct_match = !pattern_fields.is_empty()
2347 && pattern_fields.iter().map(|field| field.ident.name.as_str()).all(is_number);
2348if is_tuple_struct_match {
2349return Ok(());
2350 }
23512352// we don't care to report errors for a struct if the struct itself is tainted
2353variant.has_errors()?;
23542355let path = rustc_hir_pretty::qpath_to_string(&self.tcx, qpath);
2356let 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!(
2357self.dcx(),
2358 pat.span,
2359 E0769,
2360"tuple variant `{}` written as struct variant",
2361 path
2362 );
2363let (sugg, appl) = if fields.len() == variant.fields.len() {
2364 (
2365self.get_suggested_tuple_struct_pattern(fields, variant),
2366 Applicability::MachineApplicable,
2367 )
2368 } else {
2369 (
2370variant.fields.iter().map(|_| "_").collect::<Vec<&str>>().join(", "),
2371 Applicability::MaybeIncorrect,
2372 )
2373 };
2374err.span_suggestion_verbose(
2375qpath.span().shrink_to_hi().to(pat.span.shrink_to_hi()),
2376"use the tuple variant pattern syntax instead",
2377::alloc::__export::must_use({
::alloc::fmt::format(format_args!("({0})", sugg))
})format!("({sugg})"),
2378appl,
2379 );
2380return Err(err.emit());
2381 }
2382Ok(())
2383 }
23842385fn get_suggested_tuple_struct_pattern(
2386&self,
2387 fields: &[hir::PatField<'_>],
2388 variant: &VariantDef,
2389 ) -> String {
2390let variant_field_idents =
2391variant.fields.iter().map(|f| f.ident(self.tcx)).collect::<Vec<Ident>>();
2392fields2393 .iter()
2394 .map(|field| {
2395match self.tcx.sess.source_map().span_to_snippet(field.pat.span) {
2396Ok(f) => {
2397// Field names are numbers, but numbers
2398 // are not valid identifiers
2399if variant_field_idents.contains(&field.ident) {
2400String::from("_")
2401 } else {
2402f2403 }
2404 }
2405Err(_) => rustc_hir_pretty::pat_to_string(&self.tcx, field.pat),
2406 }
2407 })
2408 .collect::<Vec<String>>()
2409 .join(", ")
2410 }
24112412/// Returns a diagnostic reporting a struct pattern which is missing an `..` due to
2413 /// inaccessible fields.
2414 ///
2415 /// ```text
2416 /// error: pattern requires `..` due to inaccessible fields
2417 /// --> src/main.rs:10:9
2418 /// |
2419 /// LL | let foo::Foo {} = foo::Foo::default();
2420 /// | ^^^^^^^^^^^
2421 /// |
2422 /// help: add a `..`
2423 /// |
2424 /// LL | let foo::Foo { .. } = foo::Foo::default();
2425 /// | ^^^^^^
2426 /// ```
2427fn error_no_accessible_fields(
2428&self,
2429 pat: &Pat<'_>,
2430 fields: &'tcx [hir::PatField<'tcx>],
2431 ) -> Diag<'a> {
2432let mut err = self2433 .dcx()
2434 .struct_span_err(pat.span, "pattern requires `..` due to inaccessible fields");
24352436if let Some(field) = fields.last() {
2437err.span_suggestion_verbose(
2438field.span.shrink_to_hi(),
2439"ignore the inaccessible and unused fields",
2440", ..",
2441 Applicability::MachineApplicable,
2442 );
2443 } else {
2444let qpath_span = if let PatKind::Struct(qpath, ..) = &pat.kind {
2445qpath.span()
2446 } else {
2447::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");
2448 };
24492450// Shrink the span to exclude the `foo:Foo` in `foo::Foo { }`.
2451let span = pat.span.with_lo(qpath_span.shrink_to_hi().hi());
2452err.span_suggestion_verbose(
2453span,
2454"ignore the inaccessible and unused fields",
2455" { .. }",
2456 Applicability::MachineApplicable,
2457 );
2458 }
2459err2460 }
24612462/// Report that a pattern for a `#[non_exhaustive]` struct marked with `non_exhaustive_omitted_patterns`
2463 /// is not exhaustive enough.
2464 ///
2465 /// Nb: the partner lint for enums lives in `compiler/rustc_mir_build/src/thir/pattern/usefulness.rs`.
2466fn lint_non_exhaustive_omitted_patterns(
2467&self,
2468 pat: &Pat<'_>,
2469 unmentioned_fields: &[(&ty::FieldDef, Ident)],
2470 ty: Ty<'tcx>,
2471 ) {
2472fn joined_uncovered_patterns(witnesses: &[&Ident]) -> String {
2473const LIMIT: usize = 3;
2474match witnesses {
2475 [] => {
2476{
::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!(
2477"expected an uncovered pattern, otherwise why are we emitting an error?"
2478)2479 }
2480 [witness] => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", witness))
})format!("`{witness}`"),
2481 [head @ .., tail] if head.len() < LIMIT => {
2482let head: Vec<_> = head.iter().map(<_>::to_string).collect();
2483::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and `{1}`",
head.join("`, `"), tail))
})format!("`{}` and `{}`", head.join("`, `"), tail)2484 }
2485_ => {
2486let (head, tail) = witnesses.split_at(LIMIT);
2487let head: Vec<_> = head.iter().map(<_>::to_string).collect();
2488::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}` and {1} more",
head.join("`, `"), tail.len()))
})format!("`{}` and {} more", head.join("`, `"), tail.len())2489 }
2490 }
2491 }
2492let joined_patterns = joined_uncovered_patterns(
2493&unmentioned_fields.iter().map(|(_, i)| i).collect::<Vec<_>>(),
2494 );
24952496self.tcx.node_span_lint(NON_EXHAUSTIVE_OMITTED_PATTERNS, pat.hir_id, pat.span, |lint| {
2497lint.primary_message("some fields are not explicitly listed");
2498lint.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));
2499lint.help(
2500"ensure that all fields are mentioned explicitly by adding the suggested fields",
2501 );
2502lint.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!(
2503"the pattern is of type `{ty}` and the `non_exhaustive_omitted_patterns` attribute was found",
2504 ));
2505 });
2506 }
25072508/// Returns a diagnostic reporting a struct pattern which does not mention some fields.
2509 ///
2510 /// ```text
2511 /// error[E0027]: pattern does not mention field `bar`
2512 /// --> src/main.rs:15:9
2513 /// |
2514 /// LL | let foo::Foo {} = foo::Foo::new();
2515 /// | ^^^^^^^^^^^ missing field `bar`
2516 /// ```
2517fn error_unmentioned_fields(
2518&self,
2519 pat: &Pat<'_>,
2520 unmentioned_fields: &[(&ty::FieldDef, Ident)],
2521 have_inaccessible_fields: bool,
2522 fields: &'tcx [hir::PatField<'tcx>],
2523 ) -> Diag<'a> {
2524let inaccessible = if have_inaccessible_fields { " and inaccessible fields" } else { "" };
2525let field_names = if let [(_, field)] = unmentioned_fields {
2526::alloc::__export::must_use({
::alloc::fmt::format(format_args!("field `{0}`{1}", field,
inaccessible))
})format!("field `{field}`{inaccessible}")2527 } else {
2528let fields = unmentioned_fields2529 .iter()
2530 .map(|(_, name)| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("`{0}`", name))
})format!("`{name}`"))
2531 .collect::<Vec<String>>()
2532 .join(", ");
2533::alloc::__export::must_use({
::alloc::fmt::format(format_args!("fields {0}{1}", fields,
inaccessible))
})format!("fields {fields}{inaccessible}")2534 };
2535let 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!(
2536self.dcx(),
2537 pat.span,
2538 E0027,
2539"pattern does not mention {}",
2540 field_names
2541 );
2542err.span_label(pat.span, ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("missing {0}", field_names))
})format!("missing {field_names}"));
2543let len = unmentioned_fields.len();
2544let (prefix, postfix, sp) = match fields {
2545 [] => match &pat.kind {
2546 PatKind::Struct(path, [], None) => {
2547 (" { ", " }", path.span().shrink_to_hi().until(pat.span.shrink_to_hi()))
2548 }
2549_ => return err,
2550 },
2551 [.., field] => {
2552// Account for last field having a trailing comma or parse recovery at the tail of
2553 // the pattern to avoid invalid suggestion (#78511).
2554let tail = field.span.shrink_to_hi().with_hi(pat.span.hi());
2555match &pat.kind {
2556 PatKind::Struct(..) => (", ", " }", tail),
2557_ => return err,
2558 }
2559 }
2560 };
2561err.span_suggestion(
2562sp,
2563::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!(
2564"include the missing field{} in the pattern{}",
2565pluralize!(len),
2566if have_inaccessible_fields { " and ignore the inaccessible fields" } else { "" }
2567 ),
2568::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!(
2569"{}{}{}{}",
2570 prefix,
2571 unmentioned_fields
2572 .iter()
2573 .map(|(_, name)| {
2574let field_name = name.to_string();
2575if is_number(&field_name) { format!("{field_name}: _") } else { field_name }
2576 })
2577 .collect::<Vec<_>>()
2578 .join(", "),
2579if have_inaccessible_fields { ", .." } else { "" },
2580 postfix,
2581 ),
2582 Applicability::MachineApplicable,
2583 );
2584err.span_suggestion(
2585sp,
2586::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!(
2587"if you don't care about {these} missing field{s}, you can explicitly ignore {them}",
2588 these = pluralize!("this", len),
2589 s = pluralize!(len),
2590 them = if len == 1 { "it" } else { "them" },
2591 ),
2592::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!(
2593"{}{}{}{}",
2594 prefix,
2595 unmentioned_fields
2596 .iter()
2597 .map(|(_, name)| {
2598let field_name = name.to_string();
2599format!("{field_name}: _")
2600 })
2601 .collect::<Vec<_>>()
2602 .join(", "),
2603if have_inaccessible_fields { ", .." } else { "" },
2604 postfix,
2605 ),
2606 Applicability::MachineApplicable,
2607 );
2608err.span_suggestion(
2609sp,
2610"or always ignore missing fields here",
2611::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{0}..{1}", prefix, postfix))
})format!("{prefix}..{postfix}"),
2612 Applicability::MachineApplicable,
2613 );
2614err2615 }
26162617fn check_pat_box(
2618&self,
2619 span: Span,
2620 inner: &'tcx Pat<'tcx>,
2621 expected: Ty<'tcx>,
2622 pat_info: PatInfo<'tcx>,
2623 ) -> Ty<'tcx> {
2624let tcx = self.tcx;
2625let (box_ty, inner_ty) = self2626 .check_dereferenceable(span, expected, inner)
2627 .and_then(|()| {
2628// Here, `demand::subtype` is good enough, but I don't
2629 // think any errors can be introduced by using `demand::eqtype`.
2630let inner_ty = self.next_ty_var(inner.span);
2631let box_ty = Ty::new_box(tcx, inner_ty);
2632self.demand_eqtype_pat(span, expected, box_ty, &pat_info.top_info)?;
2633Ok((box_ty, inner_ty))
2634 })
2635 .unwrap_or_else(|guar| {
2636let err = Ty::new_error(tcx, guar);
2637 (err, err)
2638 });
2639self.check_pat(inner, inner_ty, pat_info);
2640box_ty2641 }
26422643fn check_pat_deref(
2644&self,
2645 span: Span,
2646 inner: &'tcx Pat<'tcx>,
2647 expected: Ty<'tcx>,
2648 pat_info: PatInfo<'tcx>,
2649 ) -> Ty<'tcx> {
2650let target_ty = self.deref_pat_target(span, expected);
2651self.check_pat(inner, target_ty, pat_info);
2652self.register_deref_mut_bounds_if_needed(span, inner, [expected]);
2653expected2654 }
26552656fn deref_pat_target(&self, span: Span, source_ty: Ty<'tcx>) -> Ty<'tcx> {
2657// Register a `DerefPure` bound, which is required by all `deref!()` pats.
2658let tcx = self.tcx;
2659self.register_bound(
2660source_ty,
2661tcx.require_lang_item(hir::LangItem::DerefPure, span),
2662self.misc(span),
2663 );
2664// The expected type for the deref pat's inner pattern is `<expected as Deref>::Target`.
2665let target_ty = Ty::new_projection(
2666tcx,
2667tcx.require_lang_item(hir::LangItem::DerefTarget, span),
2668 [source_ty],
2669 );
2670let target_ty = self.normalize(span, target_ty);
2671self.try_structurally_resolve_type(span, target_ty)
2672 }
26732674/// Check if the interior of a deref pattern (either explicit or implicit) has any `ref mut`
2675 /// bindings, which would require `DerefMut` to be emitted in MIR building instead of just
2676 /// `Deref`. We do this *after* checking the inner pattern, since we want to make sure to
2677 /// account for `ref mut` binding modes inherited from implicitly dereferencing `&mut` refs.
2678fn register_deref_mut_bounds_if_needed(
2679&self,
2680 span: Span,
2681 inner: &'tcx Pat<'tcx>,
2682 derefed_tys: impl IntoIterator<Item = Ty<'tcx>>,
2683 ) {
2684if self.typeck_results.borrow().pat_has_ref_mut_binding(inner) {
2685for mutably_derefed_ty in derefed_tys {
2686self.register_bound(
2687 mutably_derefed_ty,
2688self.tcx.require_lang_item(hir::LangItem::DerefMut, span),
2689self.misc(span),
2690 );
2691 }
2692 }
2693 }
26942695// Precondition: Pat is Ref(inner)
2696fn check_pat_ref(
2697&self,
2698 pat: &'tcx Pat<'tcx>,
2699 inner: &'tcx Pat<'tcx>,
2700 pat_pinned: Pinnedness,
2701 pat_mutbl: Mutability,
2702mut expected: Ty<'tcx>,
2703mut pat_info: PatInfo<'tcx>,
2704 ) -> Ty<'tcx> {
2705let tcx = self.tcx;
27062707let pat_prefix_span =
2708inner.span.find_ancestor_inside(pat.span).map(|end| pat.span.until(end));
27092710let ref_pat_matches_mut_ref = self.ref_pat_matches_mut_ref();
2711if ref_pat_matches_mut_ref && pat_mutbl == Mutability::Not {
2712// If `&` patterns can match against mutable reference types (RFC 3627, Rule 5), we need
2713 // to prevent subpatterns from binding with `ref mut`. Subpatterns of a shared reference
2714 // pattern should have read-only access to the scrutinee, and the borrow checker won't
2715 // catch it in this case.
2716pat_info.max_ref_mutbl = pat_info.max_ref_mutbl.cap_to_weakly_not(pat_prefix_span);
2717 }
27182719expected = self.try_structurally_resolve_type(pat.span, expected);
2720// Determine whether we're consuming an inherited reference and resetting the default
2721 // binding mode, based on edition and enabled experimental features.
2722if let ByRef::Yes(inh_pin, inh_mut) = pat_info.binding_mode
2723 && pat_pinned == inh_pin2724 {
2725match self.ref_pat_matches_inherited_ref(pat.span.edition()) {
2726 InheritedRefMatchRule::EatOuter => {
2727// ref pattern attempts to consume inherited reference
2728if pat_mutbl > inh_mut {
2729// Tried to match inherited `ref` with `&mut`
2730 // NB: This assumes that `&` patterns can match against mutable references
2731 // (RFC 3627, Rule 5). If we implement a pattern typing ruleset with Rule 4E
2732 // but not Rule 5, we'll need to check that here.
2733if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2734self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2735 }
27362737pat_info.binding_mode = ByRef::No;
2738self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2739self.check_pat(inner, expected, pat_info);
2740return expected;
2741 }
2742 InheritedRefMatchRule::EatInner => {
2743if let ty::Ref(_, _, r_mutbl) = *expected.kind()
2744 && pat_mutbl <= r_mutbl2745 {
2746// Match against the reference type; don't consume the inherited ref.
2747 // NB: The check for compatible pattern and ref type mutability assumes that
2748 // `&` patterns can match against mutable references (RFC 3627, Rule 5). If
2749 // we implement a pattern typing ruleset with Rule 4 (including the fallback
2750 // to matching the inherited ref when the inner ref can't match) but not
2751 // Rule 5, we'll need to check that here.
2752if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2753// NB: For RFC 3627's Rule 3, we limit the default binding mode's ref
2754 // mutability to `pat_info.max_ref_mutbl`. If we implement a pattern typing
2755 // ruleset with Rule 4 but not Rule 3, we'll need to check that here.
2756if true {
if !self.downgrade_mut_inside_shared() {
::core::panicking::panic("assertion failed: self.downgrade_mut_inside_shared()")
};
};debug_assert!(self.downgrade_mut_inside_shared());
2757let mutbl_cap = cmp::min(r_mutbl, pat_info.max_ref_mutbl.as_mutbl());
2758pat_info.binding_mode = pat_info.binding_mode.cap_ref_mutability(mutbl_cap);
2759 } else {
2760// The reference pattern can't match against the expected type, so try
2761 // matching against the inherited ref instead.
2762if pat_mutbl > inh_mut {
2763// We can't match an inherited shared reference with `&mut`.
2764 // NB: This assumes that `&` patterns can match against mutable
2765 // references (RFC 3627, Rule 5). If we implement a pattern typing
2766 // ruleset with Rule 4 but not Rule 5, we'll need to check that here.
2767 // FIXME(ref_pat_eat_one_layer_2024_structural): If we already tried
2768 // matching the real reference, the error message should explain that
2769 // falling back to the inherited reference didn't work. This should be
2770 // the same error as the old-Edition version below.
2771if true {
if !ref_pat_matches_mut_ref {
::core::panicking::panic("assertion failed: ref_pat_matches_mut_ref")
};
};debug_assert!(ref_pat_matches_mut_ref);
2772self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2773 }
27742775pat_info.binding_mode = ByRef::No;
2776self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2777self.check_pat(inner, expected, pat_info);
2778return expected;
2779 }
2780 }
2781 InheritedRefMatchRule::EatBoth { consider_inherited_ref: true } => {
2782// Reset binding mode on old editions
2783pat_info.binding_mode = ByRef::No;
27842785if let ty::Ref(_, inner_ty, _) = *expected.kind() {
2786// Consume both the inherited and inner references.
2787if pat_mutbl.is_mut() && inh_mut.is_mut() {
2788// As a special case, a `&mut` reference pattern will be able to match
2789 // against a reference type of any mutability if the inherited ref is
2790 // mutable. Since this allows us to match against a shared reference
2791 // type, we refer to this as "falling back" to matching the inherited
2792 // reference, though we consume the real reference as well. We handle
2793 // this here to avoid adding this case to the common logic below.
2794self.check_pat(inner, inner_ty, pat_info);
2795return expected;
2796 } else {
2797// Otherwise, use the common logic below for matching the inner
2798 // reference type.
2799 // FIXME(ref_pat_eat_one_layer_2024_structural): If this results in a
2800 // mutability mismatch, the error message should explain that falling
2801 // back to the inherited reference didn't work. This should be the same
2802 // error as the Edition 2024 version above.
2803}
2804 } else {
2805// The expected type isn't a reference type, so only match against the
2806 // inherited reference.
2807if pat_mutbl > inh_mut {
2808// We can't match a lone inherited shared reference with `&mut`.
2809self.error_inherited_ref_mutability_mismatch(pat, pat_prefix_span);
2810 }
28112812self.typeck_results.borrow_mut().skipped_ref_pats_mut().insert(pat.hir_id);
2813self.check_pat(inner, expected, pat_info);
2814return expected;
2815 }
2816 }
2817 InheritedRefMatchRule::EatBoth { consider_inherited_ref: false } => {
2818// Reset binding mode on stable Rust. This will be a type error below if
2819 // `expected` is not a reference type.
2820pat_info.binding_mode = ByRef::No;
2821self.add_rust_2024_migration_desugared_pat(
2822pat_info.top_info.hir_id,
2823pat,
2824match pat_mutbl {
2825 Mutability::Not => '&', // last char of `&`
2826Mutability::Mut => 't', // last char of `&mut`
2827},
2828inh_mut,
2829 )
2830 }
2831 }
2832 }
28332834let (ref_ty, inner_ty) = match self.check_dereferenceable(pat.span, expected, inner) {
2835Ok(()) => {
2836// `demand::subtype` would be good enough, but using `eqtype` turns
2837 // out to be equally general. See (note_1) for details.
28382839 // Take region, inner-type from expected type if we can,
2840 // to avoid creating needless variables. This also helps with
2841 // the bad interactions of the given hack detailed in (note_1).
2842{
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:2842",
"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(2842u32),
::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);
2843match expected.maybe_pinned_ref() {
2844Some((r_ty, r_pinned, r_mutbl, _))
2845if ((ref_pat_matches_mut_ref && r_mutbl >= pat_mutbl)
2846 || r_mutbl == pat_mutbl)
2847 && pat_pinned == r_pinned =>
2848 {
2849if r_mutbl == Mutability::Not {
2850pat_info.max_ref_mutbl = MutblCap::Not;
2851 }
2852if r_pinned == Pinnedness::Pinned {
2853pat_info.max_pinnedness = PinnednessCap::Pinned;
2854 }
28552856 (expected, r_ty)
2857 }
2858_ => {
2859let inner_ty = self.next_ty_var(inner.span);
2860let ref_ty = self.new_ref_ty(pat.span, pat_pinned, pat_mutbl, inner_ty);
2861{
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:2861",
"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(2861u32),
::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);
2862let err = self.demand_eqtype_pat_diag(
2863pat.span,
2864expected,
2865ref_ty,
2866&pat_info.top_info,
2867 );
28682869// Look for a case like `fn foo(&foo: u32)` and suggest
2870 // `fn foo(foo: &u32)`
2871if let Err(mut err) = err {
2872self.borrow_pat_suggestion(&mut err, pat);
2873err.emit();
2874 }
2875 (ref_ty, inner_ty)
2876 }
2877 }
2878 }
2879Err(guar) => {
2880let err = Ty::new_error(tcx, guar);
2881 (err, err)
2882 }
2883 };
28842885self.check_pat(inner, inner_ty, pat_info);
2886ref_ty2887 }
28882889/// Create a reference or pinned reference type with a fresh region variable.
2890fn new_ref_ty(
2891&self,
2892 span: Span,
2893 pinnedness: Pinnedness,
2894 mutbl: Mutability,
2895 ty: Ty<'tcx>,
2896 ) -> Ty<'tcx> {
2897let region = self.next_region_var(RegionVariableOrigin::PatternRegion(span));
2898let ref_ty = Ty::new_ref(self.tcx, region, ty, mutbl);
2899if pinnedness.is_pinned() {
2900return self.new_pinned_ty(span, ref_ty);
2901 }
2902ref_ty2903 }
29042905/// Create a pinned type.
2906fn new_pinned_ty(&self, span: Span, ty: Ty<'tcx>) -> Ty<'tcx> {
2907Ty::new_adt(
2908self.tcx,
2909self.tcx.adt_def(self.tcx.require_lang_item(LangItem::Pin, span)),
2910self.tcx.mk_args(&[ty.into()]),
2911 )
2912 }
29132914fn error_inherited_ref_mutability_mismatch(
2915&self,
2916 pat: &'tcx Pat<'tcx>,
2917 pat_prefix_span: Option<Span>,
2918 ) -> ErrorGuaranteed {
2919let err_msg = "mismatched types";
2920let err = if let Some(span) = pat_prefix_span {
2921let mut err = self.dcx().struct_span_err(span, err_msg);
2922err.code(E0308);
2923err.note("cannot match inherited `&` with `&mut` pattern");
2924err.span_suggestion_verbose(
2925span,
2926"replace this `&mut` pattern with `&`",
2927"&",
2928 Applicability::MachineApplicable,
2929 );
2930err2931 } else {
2932self.dcx().struct_span_err(pat.span, err_msg)
2933 };
2934err.emit()
2935 }
29362937fn try_resolve_slice_ty_to_array_ty(
2938&self,
2939 before: &'tcx [Pat<'tcx>],
2940 slice: Option<&'tcx Pat<'tcx>>,
2941 span: Span,
2942 ) -> Option<Ty<'tcx>> {
2943if slice.is_some() {
2944return None;
2945 }
29462947let tcx = self.tcx;
2948let len = before.len();
2949let inner_ty = self.next_ty_var(span);
29502951Some(Ty::new_array(tcx, inner_ty, len.try_into().unwrap()))
2952 }
29532954/// Used to determines whether we can infer the expected type in the slice pattern to be of type array.
2955 /// This is only possible if we're in an irrefutable pattern. If we were to allow this in refutable
2956 /// patterns we wouldn't e.g. report ambiguity in the following situation:
2957 ///
2958 /// ```ignore(rust)
2959 /// struct Zeroes;
2960 /// const ARR: [usize; 2] = [0; 2];
2961 /// const ARR2: [usize; 2] = [2; 2];
2962 ///
2963 /// impl Into<&'static [usize; 2]> for Zeroes {
2964 /// fn into(self) -> &'static [usize; 2] {
2965 /// &ARR
2966 /// }
2967 /// }
2968 ///
2969 /// impl Into<&'static [usize]> for Zeroes {
2970 /// fn into(self) -> &'static [usize] {
2971 /// &ARR2
2972 /// }
2973 /// }
2974 ///
2975 /// fn main() {
2976 /// let &[a, b]: &[usize] = Zeroes.into() else {
2977 /// ..
2978 /// };
2979 /// }
2980 /// ```
2981 ///
2982 /// If we're in an irrefutable pattern we prefer the array impl candidate given that
2983 /// the slice impl candidate would be rejected anyway (if no ambiguity existed).
2984fn pat_is_irrefutable(&self, decl_origin: Option<DeclOrigin<'_>>) -> bool {
2985match decl_origin {
2986Some(DeclOrigin::LocalDecl { els: None }) => true,
2987Some(DeclOrigin::LocalDecl { els: Some(_) } | DeclOrigin::LetExpr) | None => false,
2988 }
2989 }
29902991/// Type check a slice pattern.
2992 ///
2993 /// Syntactically, these look like `[pat_0, ..., pat_n]`.
2994 /// Semantically, we are type checking a pattern with structure:
2995 /// ```ignore (not-rust)
2996 /// [before_0, ..., before_n, (slice, after_0, ... after_n)?]
2997 /// ```
2998 /// The type of `slice`, if it is present, depends on the `expected` type.
2999 /// If `slice` is missing, then so is `after_i`.
3000 /// If `slice` is present, it can still represent 0 elements.
3001fn check_pat_slice(
3002&self,
3003 span: Span,
3004 before: &'tcx [Pat<'tcx>],
3005 slice: Option<&'tcx Pat<'tcx>>,
3006 after: &'tcx [Pat<'tcx>],
3007 expected: Ty<'tcx>,
3008 pat_info: PatInfo<'tcx>,
3009 ) -> Ty<'tcx> {
3010let expected = self.try_structurally_resolve_type(span, expected);
30113012// If the pattern is irrefutable and `expected` is an infer ty, we try to equate it
3013 // to an array if the given pattern allows it. See issue #76342
3014if self.pat_is_irrefutable(pat_info.decl_origin) && expected.is_ty_var() {
3015if let Some(resolved_arr_ty) =
3016self.try_resolve_slice_ty_to_array_ty(before, slice, span)
3017 {
3018{
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:3018",
"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(3018u32),
::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);
3019let _ = self.demand_eqtype(span, expected, resolved_arr_ty);
3020 }
3021 }
30223023let expected = self.structurally_resolve_type(span, expected);
3024{
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:3024",
"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(3024u32),
::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);
30253026let (element_ty, opt_slice_ty, inferred) = match *expected.kind() {
3027// An array, so we might have something like `let [a, b, c] = [0, 1, 2];`.
3028ty::Array(element_ty, len) => {
3029let min = before.len() as u64 + after.len() as u64;
3030let (opt_slice_ty, expected) =
3031self.check_array_pat_len(span, element_ty, expected, slice, len, min);
3032// `opt_slice_ty.is_none()` => `slice.is_none()`.
3033 // Note, though, that opt_slice_ty could be `Some(error_ty)`.
3034if !(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());
3035 (element_ty, opt_slice_ty, expected)
3036 }
3037 ty::Slice(element_ty) => (element_ty, Some(expected), expected),
3038// The expected type must be an array or slice, but was neither, so error.
3039_ => {
3040let guar = expected.error_reported().err().unwrap_or_else(|| {
3041self.error_expected_array_or_slice(span, expected, pat_info)
3042 });
3043let err = Ty::new_error(self.tcx, guar);
3044 (err, Some(err), err)
3045 }
3046 };
30473048// Type check all the patterns before `slice`.
3049for elt in before {
3050self.check_pat(elt, element_ty, pat_info);
3051 }
3052// Type check the `slice`, if present, against its expected type.
3053if let Some(slice) = slice {
3054self.check_pat(slice, opt_slice_ty.unwrap(), pat_info);
3055 }
3056// Type check the elements after `slice`, if present.
3057for elt in after {
3058self.check_pat(elt, element_ty, pat_info);
3059 }
3060inferred3061 }
30623063/// Type check the length of an array pattern.
3064 ///
3065 /// Returns both the type of the variable length pattern (or `None`), and the potentially
3066 /// inferred array type. We only return `None` for the slice type if `slice.is_none()`.
3067fn check_array_pat_len(
3068&self,
3069 span: Span,
3070 element_ty: Ty<'tcx>,
3071 arr_ty: Ty<'tcx>,
3072 slice: Option<&'tcx Pat<'tcx>>,
3073 len: ty::Const<'tcx>,
3074 min_len: u64,
3075 ) -> (Option<Ty<'tcx>>, Ty<'tcx>) {
3076let len = self.try_structurally_resolve_const(span, len).try_to_target_usize(self.tcx);
30773078let guar = if let Some(len) = len {
3079// Now we know the length...
3080if slice.is_none() {
3081// ...and since there is no variable-length pattern,
3082 // we require an exact match between the number of elements
3083 // in the array pattern and as provided by the matched type.
3084if min_len == len {
3085return (None, arr_ty);
3086 }
30873088self.error_scrutinee_inconsistent_length(span, min_len, len)
3089 } else if let Some(pat_len) = len.checked_sub(min_len) {
3090// The variable-length pattern was there,
3091 // so it has an array type with the remaining elements left as its size...
3092return (Some(Ty::new_array(self.tcx, element_ty, pat_len)), arr_ty);
3093 } else {
3094// ...however, in this case, there were no remaining elements.
3095 // That is, the slice pattern requires more than the array type offers.
3096self.error_scrutinee_with_rest_inconsistent_length(span, min_len, len)
3097 }
3098 } else if slice.is_none() {
3099// We have a pattern with a fixed length,
3100 // which we can use to infer the length of the array.
3101let updated_arr_ty = Ty::new_array(self.tcx, element_ty, min_len);
3102self.demand_eqtype(span, updated_arr_ty, arr_ty);
3103return (None, updated_arr_ty);
3104 } else {
3105// We have a variable-length pattern and don't know the array length.
3106 // This happens if we have e.g.,
3107 // `let [a, b, ..] = arr` where `arr: [T; N]` where `const N: usize`.
3108self.error_scrutinee_unfixed_length(span)
3109 };
31103111// If we get here, we must have emitted an error.
3112(Some(Ty::new_error(self.tcx, guar)), arr_ty)
3113 }
31143115fn error_scrutinee_inconsistent_length(
3116&self,
3117 span: Span,
3118 min_len: u64,
3119 size: u64,
3120 ) -> ErrorGuaranteed {
3121{
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!(
3122self.dcx(),
3123 span,
3124 E0527,
3125"pattern requires {} element{} but array has {}",
3126 min_len,
3127pluralize!(min_len),
3128 size,
3129 )3130 .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)))
3131 .emit()
3132 }
31333134fn error_scrutinee_with_rest_inconsistent_length(
3135&self,
3136 span: Span,
3137 min_len: u64,
3138 size: u64,
3139 ) -> ErrorGuaranteed {
3140{
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!(
3141self.dcx(),
3142 span,
3143 E0528,
3144"pattern requires at least {} element{} but array has {}",
3145 min_len,
3146pluralize!(min_len),
3147 size,
3148 )3149 .with_span_label(
3150span,
3151::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),),
3152 )
3153 .emit()
3154 }
31553156fn error_scrutinee_unfixed_length(&self, span: Span) -> ErrorGuaranteed {
3157{
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!(
3158self.dcx(),
3159 span,
3160 E0730,
3161"cannot pattern-match on an array without a fixed length",
3162 )3163 .emit()
3164 }
31653166fn error_expected_array_or_slice(
3167&self,
3168 span: Span,
3169 expected_ty: Ty<'tcx>,
3170 pat_info: PatInfo<'tcx>,
3171 ) -> ErrorGuaranteed {
3172let PatInfo { top_info: ti, current_depth, .. } = pat_info;
31733174let mut slice_pat_semantics = false;
3175let mut as_deref = None;
3176let mut slicing = None;
3177if let ty::Ref(_, ty, _) = expected_ty.kind()
3178 && let ty::Array(..) | ty::Slice(..) = ty.kind()
3179 {
3180slice_pat_semantics = true;
3181 } else if self3182 .autoderef(span, expected_ty)
3183 .silence_errors()
3184 .any(|(ty, _)| #[allow(non_exhaustive_omitted_patterns)] match ty.kind() {
ty::Slice(..) | ty::Array(..) => true,
_ => false,
}matches!(ty.kind(), ty::Slice(..) | ty::Array(..)))
3185 && let Some(span) = ti.span
3186 && let Some(_) = ti.origin_expr
3187 {
3188let resolved_ty = self.resolve_vars_if_possible(ti.expected);
3189let (is_slice_or_array_or_vector, resolved_ty) =
3190self.is_slice_or_array_or_vector(resolved_ty);
3191match resolved_ty.kind() {
3192 ty::Adt(adt_def, _)
3193if self.tcx.is_diagnostic_item(sym::Option, adt_def.did())
3194 || self.tcx.is_diagnostic_item(sym::Result, adt_def.did()) =>
3195 {
3196// Slicing won't work here, but `.as_deref()` might (issue #91328).
3197as_deref = Some(errors::AsDerefSuggestion { span: span.shrink_to_hi() });
3198 }
3199_ => (),
3200 }
32013202let is_top_level = current_depth <= 1;
3203if is_slice_or_array_or_vector && is_top_level {
3204slicing = Some(errors::SlicingSuggestion { span: span.shrink_to_hi() });
3205 }
3206 }
3207self.dcx().emit_err(errors::ExpectedArrayOrSlice {
3208span,
3209 ty: expected_ty,
3210slice_pat_semantics,
3211as_deref,
3212slicing,
3213 })
3214 }
32153216fn is_slice_or_array_or_vector(&self, ty: Ty<'tcx>) -> (bool, Ty<'tcx>) {
3217match ty.kind() {
3218 ty::Adt(adt_def, _) if self.tcx.is_diagnostic_item(sym::Vec, adt_def.did()) => {
3219 (true, ty)
3220 }
3221 ty::Ref(_, ty, _) => self.is_slice_or_array_or_vector(*ty),
3222 ty::Slice(..) | ty::Array(..) => (true, ty),
3223_ => (false, ty),
3224 }
3225 }
32263227/// Record a pattern that's invalid under Rust 2024 match ergonomics, along with a problematic
3228 /// span, so that the pattern migration lint can desugar it during THIR construction.
3229fn add_rust_2024_migration_desugared_pat(
3230&self,
3231 pat_id: HirId,
3232 subpat: &'tcx Pat<'tcx>,
3233 final_char: char,
3234 def_br_mutbl: Mutability,
3235 ) {
3236// Try to trim the span we're labeling to just the `&` or binding mode that's an issue.
3237let from_expansion = subpat.span.from_expansion();
3238let trimmed_span = if from_expansion {
3239// If the subpattern is from an expansion, highlight the whole macro call instead.
3240subpat.span
3241 } else {
3242let trimmed = self.tcx.sess.source_map().span_through_char(subpat.span, final_char);
3243// The edition of the trimmed span should be the same as `subpat.span`; this will be a
3244 // a hard error if the subpattern is of edition >= 2024. We set it manually to be sure:
3245trimmed.with_ctxt(subpat.span.ctxt())
3246 };
32473248let mut typeck_results = self.typeck_results.borrow_mut();
3249let mut table = typeck_results.rust_2024_migration_desugared_pats_mut();
3250// FIXME(ref_pat_eat_one_layer_2024): The migration diagnostic doesn't know how to track the
3251 // default binding mode in the presence of Rule 3 or Rule 5. As a consequence, the labels it
3252 // gives for default binding modes are wrong, as well as suggestions based on the default
3253 // binding mode. This keeps it from making those suggestions, as doing so could panic.
3254let info = table.entry(pat_id).or_insert_with(|| ty::Rust2024IncompatiblePatInfo {
3255 primary_labels: Vec::new(),
3256 bad_ref_modifiers: false,
3257 bad_mut_modifiers: false,
3258 bad_ref_pats: false,
3259 suggest_eliding_modes: !self.tcx.features().ref_pat_eat_one_layer_2024()
3260 && !self.tcx.features().ref_pat_eat_one_layer_2024_structural(),
3261 });
32623263let pat_kind = if let PatKind::Binding(user_bind_annot, _, _, _) = subpat.kind {
3264// If the user-provided binding modifier doesn't match the default binding mode, we'll
3265 // need to suggest reference patterns, which can affect other bindings.
3266 // For simplicity, we opt to suggest making the pattern fully explicit.
3267info.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!(
3268 user_bind_annot,
3269 BindingMode(ByRef::Yes(_, mutbl), Mutability::Not) if mutbl == def_br_mutbl
3270 );
3271if user_bind_annot == BindingMode(ByRef::No, Mutability::Mut) {
3272info.bad_mut_modifiers = true;
3273"`mut` binding modifier"
3274} else {
3275info.bad_ref_modifiers = true;
3276match user_bind_annot.1 {
3277 Mutability::Not => "explicit `ref` binding modifier",
3278 Mutability::Mut => "explicit `ref mut` binding modifier",
3279 }
3280 }
3281 } else {
3282info.bad_ref_pats = true;
3283// For simplicity, we don't try to suggest eliding reference patterns. Thus, we'll
3284 // suggest adding them instead, which can affect the types assigned to bindings.
3285 // As such, we opt to suggest making the pattern fully explicit.
3286info.suggest_eliding_modes = false;
3287"reference pattern"
3288};
3289// Only provide a detailed label if the problematic subpattern isn't from an expansion.
3290 // In the case that it's from a macro, we'll add a more detailed note in the emitter.
3291let primary_label = if from_expansion {
3292// We can't suggest eliding modifiers within expansions.
3293info.suggest_eliding_modes = false;
3294// NB: This wording assumes the only expansions that can produce problematic reference
3295 // patterns and bindings are macros. If a desugaring or AST pass is added that can do
3296 // so, we may want to inspect the span's source callee or macro backtrace.
3297"occurs within macro expansion".to_owned()
3298 } else {
3299::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")3300 };
3301info.primary_labels.push((trimmed_span, primary_label));
3302 }
3303}