Skip to main content

rustc_mir_build/thir/pattern/
check_match.rs

1use rustc_arena::{DroplessArena, TypedArena};
2use rustc_ast::Mutability;
3use rustc_data_structures::fx::FxIndexSet;
4use rustc_data_structures::stack::ensure_sufficient_stack;
5use rustc_errors::codes::*;
6use rustc_errors::{Applicability, ErrorGuaranteed, MultiSpan, msg, struct_span_code_err};
7use rustc_hir::def::*;
8use rustc_hir::def_id::{DefId, LocalDefId};
9use rustc_hir::{self as hir, BindingMode, ByRef, HirId, MatchSource};
10use rustc_infer::infer::TyCtxtInferExt;
11use rustc_middle::bug;
12use rustc_middle::thir::visit::Visitor;
13use rustc_middle::thir::*;
14use rustc_middle::ty::print::with_no_trimmed_paths;
15use rustc_middle::ty::{self, AdtDef, Ty, TyCtxt};
16use rustc_pattern_analysis::diagnostics::Uncovered;
17use rustc_pattern_analysis::rustc::{
18    Constructor, DeconstructedPat, MatchArm, RedundancyExplanation, RevealedTy,
19    RustcPatCtxt as PatCtxt, Usefulness, UsefulnessReport, WitnessPat,
20};
21use rustc_session::lint::builtin::{
22    BINDINGS_WITH_VARIANT_NAME, IRREFUTABLE_LET_PATTERNS, UNREACHABLE_PATTERNS,
23};
24use rustc_span::edit_distance::find_best_match_for_name;
25use rustc_span::hygiene::DesugaringKind;
26use rustc_span::{Ident, Span};
27use rustc_trait_selection::infer::InferCtxtExt;
28use tracing::instrument;
29
30use crate::diagnostics::*;
31
32pub(crate) fn check_match(tcx: TyCtxt<'_>, def_id: LocalDefId) -> Result<(), ErrorGuaranteed> {
33    let typeck_results = tcx.typeck(def_id);
34    let (thir, expr) = tcx.thir_body(def_id)?;
35    let thir = thir.borrow();
36    let pattern_arena = TypedArena::default();
37    let dropless_arena = DroplessArena::default();
38    let mut visitor = MatchVisitor {
39        tcx,
40        thir: &*thir,
41        typeck_results,
42        // FIXME(#132279): We're in a body, should handle opaques.
43        typing_env: ty::TypingEnv::non_body_analysis(tcx, def_id),
44        hir_source: tcx.local_def_id_to_hir_id(def_id),
45        let_source: LetSource::None,
46        pattern_arena: &pattern_arena,
47        dropless_arena: &dropless_arena,
48        error: Ok(()),
49    };
50    visitor.visit_expr(&thir[expr]);
51
52    let origin = match tcx.def_kind(def_id) {
53        DefKind::AssocFn | DefKind::Fn => "function argument",
54        DefKind::Closure => "closure argument",
55        // other types of MIR don't have function parameters, and we don't need to
56        // categorize those for the irrefutable check.
57        _ if thir.params.is_empty() => "",
58        kind => ::rustc_middle::util::bug::bug_fmt(format_args!("unexpected function parameters in THIR: {0:?} {1:?}",
        kind, def_id))bug!("unexpected function parameters in THIR: {kind:?} {def_id:?}"),
59    };
60
61    for param in thir.params.iter() {
62        if let Some(ref pattern) = param.pat {
63            visitor.check_binding_is_irrefutable(pattern, origin, None, None, None);
64        }
65    }
66    visitor.error
67}
68
69#[derive(#[automatically_derived]
impl ::core::fmt::Debug for RefutableFlag {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                RefutableFlag::Irrefutable => "Irrefutable",
                RefutableFlag::Refutable => "Refutable",
            })
    }
}Debug, #[automatically_derived]
impl ::core::marker::Copy for RefutableFlag { }Copy, #[automatically_derived]
impl ::core::clone::Clone for RefutableFlag {
    #[inline]
    fn clone(&self) -> RefutableFlag { *self }
}Clone, #[automatically_derived]
impl ::core::cmp::PartialEq for RefutableFlag {
    #[inline]
    fn eq(&self, other: &RefutableFlag) -> bool {
        let __self_discr = ::core::intrinsics::discriminant_value(self);
        let __arg1_discr = ::core::intrinsics::discriminant_value(other);
        __self_discr == __arg1_discr
    }
}PartialEq)]
70enum RefutableFlag {
71    Irrefutable,
72    Refutable,
73}
74use RefutableFlag::*;
75
76#[derive(#[automatically_derived]
impl ::core::clone::Clone for LetSource {
    #[inline]
    fn clone(&self) -> LetSource { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for LetSource { }Copy, #[automatically_derived]
impl ::core::fmt::Debug for LetSource {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::write_str(f,
            match self {
                LetSource::None => "None",
                LetSource::PlainLet => "PlainLet",
                LetSource::IfLet => "IfLet",
                LetSource::IfLetGuard => "IfLetGuard",
                LetSource::LetElse => "LetElse",
                LetSource::WhileLet => "WhileLet",
                LetSource::Else => "Else",
                LetSource::ElseIfLet => "ElseIfLet",
            })
    }
}Debug, #[automatically_derived]
impl ::core::cmp::PartialEq for LetSource {
    #[inline]
    fn eq(&self, other: &LetSource) -> 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 LetSource {
    #[inline]
    #[doc(hidden)]
    #[coverage(off)]
    fn assert_fields_are_eq(&self) {}
}Eq)]
77enum LetSource {
78    None,
79    PlainLet,
80    IfLet,
81    IfLetGuard,
82    LetElse,
83    WhileLet,
84    Else,
85    ElseIfLet,
86}
87
88struct MatchVisitor<'p, 'tcx> {
89    tcx: TyCtxt<'tcx>,
90    typing_env: ty::TypingEnv<'tcx>,
91    typeck_results: &'tcx ty::TypeckResults<'tcx>,
92    thir: &'p Thir<'tcx>,
93    hir_source: HirId,
94    let_source: LetSource,
95    pattern_arena: &'p TypedArena<DeconstructedPat<'p, 'tcx>>,
96    dropless_arena: &'p DroplessArena,
97    /// Tracks if we encountered an error while checking this body. That the first function to
98    /// report it stores it here. Some functions return `Result` to allow callers to short-circuit
99    /// on error, but callers don't need to store it here again.
100    error: Result<(), ErrorGuaranteed>,
101}
102
103// Visitor for a thir body. This calls `check_match` and `check_let` as appropriate.
104impl<'p, 'tcx> Visitor<'p, 'tcx> for MatchVisitor<'p, 'tcx> {
105    fn thir(&self) -> &'p Thir<'tcx> {
106        self.thir
107    }
108
109    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("visit_arm",
                                    "rustc_mir_build::thir::pattern::check_match",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/thir/pattern/check_match.rs"),
                                    ::tracing_core::__macro_support::Option::Some(109u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::thir::pattern::check_match"),
                                    ::tracing_core::field::FieldSet::new(&["arm"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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(&arm)
                                                            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;
        }
        {
            self.with_hir_source(arm.hir_id,
                |this|
                    {
                        if let Some(expr) = arm.guard {
                            this.with_let_source(LetSource::IfLetGuard,
                                |this| { this.visit_expr(&this.thir[expr]) });
                        }
                        this.visit_pat(&arm.pattern);
                        this.visit_expr(&self.thir[arm.body]);
                    });
        }
    }
}#[instrument(level = "trace", skip(self))]
110    fn visit_arm(&mut self, arm: &'p Arm<'tcx>) {
111        self.with_hir_source(arm.hir_id, |this| {
112            if let Some(expr) = arm.guard {
113                this.with_let_source(LetSource::IfLetGuard, |this| {
114                    this.visit_expr(&this.thir[expr])
115                });
116            }
117            this.visit_pat(&arm.pattern);
118            this.visit_expr(&self.thir[arm.body]);
119        });
120    }
121
122    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("visit_expr",
                                    "rustc_mir_build::thir::pattern::check_match",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/thir/pattern/check_match.rs"),
                                    ::tracing_core::__macro_support::Option::Some(122u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::thir::pattern::check_match"),
                                    ::tracing_core::field::FieldSet::new(&["ex"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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(&ex)
                                                            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;
        }
        {
            match ex.kind {
                ExprKind::Scope { value, hir_id, .. } => {
                    self.with_hir_source(hir_id,
                        |this| { this.visit_expr(&this.thir[value]); });
                    return;
                }
                ExprKind::If { cond, then, else_opt, if_then_scope: _ } => {
                    let let_source =
                        match ex.span.desugaring_kind() {
                            Some(DesugaringKind::WhileLoop) => LetSource::WhileLet,
                            _ =>
                                match self.let_source {
                                    LetSource::Else => LetSource::ElseIfLet,
                                    _ => LetSource::IfLet,
                                },
                        };
                    self.with_let_source(let_source,
                        |this| this.visit_expr(&self.thir[cond]));
                    self.with_let_source(LetSource::None,
                        |this| { this.visit_expr(&this.thir[then]); });
                    if let Some(else_) = else_opt {
                        self.with_let_source(LetSource::Else,
                            |this| { this.visit_expr(&this.thir[else_]) });
                    }
                    return;
                }
                ExprKind::Match { scrutinee, ref arms, match_source } => {
                    self.check_match(scrutinee, arms, match_source, ex.span);
                }
                ExprKind::LoopMatch {
                    match_data: LoopMatchMatchData {
                        scrutinee, ref arms, span
                        }, .. } => {
                    self.check_match(scrutinee, arms, MatchSource::Normal,
                        span);
                }
                ExprKind::Let { ref pat, expr } => {
                    self.check_let(pat, Some(expr), ex.span, None);
                }
                ExprKind::LogicalOp { op: LogicalOp::And, .. } if
                    !#[allow(non_exhaustive_omitted_patterns)] match self.let_source
                            {
                            LetSource::None => true,
                            _ => false,
                        } => {
                    let mut chain_refutabilities = Vec::new();
                    let Ok(()) =
                        self.visit_land(ex,
                            &mut chain_refutabilities) else { return };
                    if let [Some((_, Irrefutable))] = chain_refutabilities[..] {
                        self.lint_single_let(ex.span, None, None);
                    }
                    return;
                }
                _ => {}
            };
            self.with_let_source(LetSource::None,
                |this| visit::walk_expr(this, ex));
        }
    }
}#[instrument(level = "trace", skip(self))]
123    fn visit_expr(&mut self, ex: &'p Expr<'tcx>) {
124        match ex.kind {
125            ExprKind::Scope { value, hir_id, .. } => {
126                self.with_hir_source(hir_id, |this| {
127                    this.visit_expr(&this.thir[value]);
128                });
129                return;
130            }
131            ExprKind::If { cond, then, else_opt, if_then_scope: _ } => {
132                // Give a specific `let_source` for the condition.
133                let let_source = match ex.span.desugaring_kind() {
134                    Some(DesugaringKind::WhileLoop) => LetSource::WhileLet,
135                    _ => match self.let_source {
136                        LetSource::Else => LetSource::ElseIfLet,
137                        _ => LetSource::IfLet,
138                    },
139                };
140                self.with_let_source(let_source, |this| this.visit_expr(&self.thir[cond]));
141                self.with_let_source(LetSource::None, |this| {
142                    this.visit_expr(&this.thir[then]);
143                });
144                if let Some(else_) = else_opt {
145                    self.with_let_source(LetSource::Else, |this| {
146                        this.visit_expr(&this.thir[else_])
147                    });
148                }
149                return;
150            }
151            ExprKind::Match { scrutinee, ref arms, match_source } => {
152                self.check_match(scrutinee, arms, match_source, ex.span);
153            }
154            ExprKind::LoopMatch {
155                match_data: LoopMatchMatchData { scrutinee, ref arms, span },
156                ..
157            } => {
158                self.check_match(scrutinee, arms, MatchSource::Normal, span);
159            }
160            ExprKind::Let { ref pat, expr } => {
161                self.check_let(pat, Some(expr), ex.span, None);
162            }
163            ExprKind::LogicalOp { op: LogicalOp::And, .. }
164                if !matches!(self.let_source, LetSource::None) =>
165            {
166                let mut chain_refutabilities = Vec::new();
167                let Ok(()) = self.visit_land(ex, &mut chain_refutabilities) else { return };
168                // Lint only single irrefutable let binding.
169                if let [Some((_, Irrefutable))] = chain_refutabilities[..] {
170                    self.lint_single_let(ex.span, None, None);
171                }
172                return;
173            }
174            _ => {}
175        };
176        self.with_let_source(LetSource::None, |this| visit::walk_expr(this, ex));
177    }
178
179    fn visit_stmt(&mut self, stmt: &'p Stmt<'tcx>) {
180        match stmt.kind {
181            StmtKind::Let { ref pattern, initializer, else_block, hir_id, span, .. } => {
182                self.with_hir_source(hir_id, |this| {
183                    let let_source =
184                        if else_block.is_some() { LetSource::LetElse } else { LetSource::PlainLet };
185                    let else_span = else_block.map(|bid| this.thir.blocks[bid].span);
186                    this.with_let_source(let_source, |this| {
187                        this.check_let(pattern, initializer, span, else_span)
188                    });
189                    visit::walk_stmt(this, stmt);
190                });
191            }
192            StmtKind::Expr { .. } => {
193                visit::walk_stmt(self, stmt);
194            }
195        }
196    }
197}
198
199impl<'p, 'tcx> MatchVisitor<'p, 'tcx> {
200    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("with_let_source",
                                    "rustc_mir_build::thir::pattern::check_match",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/thir/pattern/check_match.rs"),
                                    ::tracing_core::__macro_support::Option::Some(200u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::thir::pattern::check_match"),
                                    ::tracing_core::field::FieldSet::new(&["let_source"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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(&let_source)
                                                            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 old_let_source = self.let_source;
            self.let_source = let_source;
            ensure_sufficient_stack(|| f(self));
            self.let_source = old_let_source;
        }
    }
}#[instrument(level = "trace", skip(self, f))]
201    fn with_let_source(&mut self, let_source: LetSource, f: impl FnOnce(&mut Self)) {
202        let old_let_source = self.let_source;
203        self.let_source = let_source;
204        ensure_sufficient_stack(|| f(self));
205        self.let_source = old_let_source;
206    }
207
208    fn with_hir_source<T>(&mut self, new_hir_source: HirId, f: impl FnOnce(&mut Self) -> T) -> T {
209        let old_hir_source = self.hir_source;
210        self.hir_source = new_hir_source;
211        let ret = f(self);
212        self.hir_source = old_hir_source;
213        ret
214    }
215
216    /// Visit a nested chain of `&&`. Used for if-let chains. This must call `visit_expr` on the
217    /// subexpressions we are not handling ourselves.
218    fn visit_land(
219        &mut self,
220        ex: &'p Expr<'tcx>,
221        accumulator: &mut Vec<Option<(Span, RefutableFlag)>>,
222    ) -> Result<(), ErrorGuaranteed> {
223        match ex.kind {
224            ExprKind::Scope { value, hir_id, .. } => {
225                self.with_hir_source(hir_id, |this| this.visit_land(&this.thir[value], accumulator))
226            }
227            ExprKind::LogicalOp { op: LogicalOp::And, lhs, rhs } => {
228                // We recurse into the lhs only, because `&&` chains associate to the left.
229                let res_lhs = self.visit_land(&self.thir[lhs], accumulator);
230                let res_rhs = self.visit_land_rhs(&self.thir[rhs])?;
231                accumulator.push(res_rhs);
232                res_lhs
233            }
234            _ => {
235                let res = self.visit_land_rhs(ex)?;
236                accumulator.push(res);
237                Ok(())
238            }
239        }
240    }
241
242    /// Visit the right-hand-side of a `&&`. Used for if-let chains. Returns `Some` if the
243    /// expression was ultimately a `let ... = ...`, and `None` if it was a normal boolean
244    /// expression. This must call `visit_expr` on the subexpressions we are not handling ourselves.
245    fn visit_land_rhs(
246        &mut self,
247        ex: &'p Expr<'tcx>,
248    ) -> Result<Option<(Span, RefutableFlag)>, ErrorGuaranteed> {
249        match ex.kind {
250            ExprKind::Scope { value, hir_id, .. } => {
251                self.with_hir_source(hir_id, |this| this.visit_land_rhs(&this.thir[value]))
252            }
253            ExprKind::Let { ref pat, expr } => {
254                let expr = &self.thir()[expr];
255                self.with_let_source(LetSource::None, |this| {
256                    this.visit_expr(expr);
257                });
258                Ok(Some((ex.span, self.is_let_irrefutable(pat, Some(expr))?)))
259            }
260            _ => {
261                self.with_let_source(LetSource::None, |this| {
262                    this.visit_expr(ex);
263                });
264                Ok(None)
265            }
266        }
267    }
268
269    fn lower_pattern(
270        &mut self,
271        cx: &PatCtxt<'p, 'tcx>,
272        pat: &'p Pat<'tcx>,
273    ) -> Result<&'p DeconstructedPat<'p, 'tcx>, ErrorGuaranteed> {
274        if let Err(err) = pat.pat_error_reported() {
275            self.error = Err(err);
276            Err(err)
277        } else {
278            // Check the pattern for some things unrelated to exhaustiveness.
279            let refutable = if cx.refutable { Refutable } else { Irrefutable };
280            let mut err = Ok(());
281            pat.walk_always(|pat| {
282                check_borrow_conflicts_in_at_patterns(self, pat);
283                check_for_bindings_named_same_as_variants(self, pat, refutable);
284                err = err.and(check_never_pattern(cx, pat));
285            });
286            err?;
287            Ok(self.pattern_arena.alloc(cx.lower_pat(pat)))
288        }
289    }
290
291    /// Inspects the match scrutinee expression to determine whether the place it evaluates to may
292    /// hold invalid data.
293    fn is_known_valid_scrutinee(&self, scrutinee: &Expr<'tcx>) -> bool {
294        use ExprKind::*;
295        match &scrutinee.kind {
296            // Pointers can validly point to a place with invalid data. It is undecided whether
297            // references can too, so we conservatively assume they can.
298            Deref { .. } => false,
299            // Inherit validity of the parent place, unless the parent is an union.
300            Field { lhs, .. } => {
301                let lhs = &self.thir()[*lhs];
302                match lhs.ty.kind() {
303                    ty::Adt(def, _) if def.is_union() => false,
304                    _ => self.is_known_valid_scrutinee(lhs),
305                }
306            }
307            // Essentially a field access.
308            Index { lhs, .. } => {
309                let lhs = &self.thir()[*lhs];
310                self.is_known_valid_scrutinee(lhs)
311            }
312
313            // No-op.
314            Scope { value, .. } => self.is_known_valid_scrutinee(&self.thir()[*value]),
315
316            // Casts don't cause a load.
317            NeverToAny { source }
318            | Cast { source }
319            | Use { source }
320            | PointerCoercion { source, .. }
321            | PlaceTypeAscription { source, .. }
322            | ValueTypeAscription { source, .. }
323            | PlaceUnwrapUnsafeBinder { source }
324            | ValueUnwrapUnsafeBinder { source }
325            | WrapUnsafeBinder { source } => self.is_known_valid_scrutinee(&self.thir()[*source]),
326
327            // These diverge.
328            Become { .. }
329            | Break { .. }
330            | Continue { .. }
331            | ConstContinue { .. }
332            | Return { .. } => true,
333
334            // These are statements that evaluate to `()`.
335            Assign { .. } | AssignOp { .. } | InlineAsm { .. } | Let { .. } => true,
336
337            // These evaluate to a value.
338            RawBorrow { .. }
339            | Adt { .. }
340            | Array { .. }
341            | Binary { .. }
342            | Block { .. }
343            | Borrow { .. }
344            | Call { .. }
345            | ByUse { .. }
346            | Closure { .. }
347            | ConstBlock { .. }
348            | ConstParam { .. }
349            | If { .. }
350            | Literal { .. }
351            | LogicalOp { .. }
352            | Loop { .. }
353            | LoopMatch { .. }
354            | Match { .. }
355            | NamedConst { .. }
356            | NonHirLiteral { .. }
357            | Repeat { .. }
358            | StaticRef { .. }
359            | ThreadLocalRef { .. }
360            | Tuple { .. }
361            | Unary { .. }
362            | UpvarRef { .. }
363            | VarRef { .. }
364            | ZstLiteral { .. }
365            | Yield { .. }
366            | Reborrow { .. } => true,
367        }
368    }
369
370    fn new_cx(
371        &self,
372        refutability: RefutableFlag,
373        whole_match_span: Option<Span>,
374        scrutinee: Option<&Expr<'tcx>>,
375        scrut_span: Span,
376    ) -> PatCtxt<'p, 'tcx> {
377        let refutable = match refutability {
378            Irrefutable => false,
379            Refutable => true,
380        };
381        // If we don't have a scrutinee we're either a function parameter or a `let x;`. Both cases
382        // require validity.
383        let known_valid_scrutinee =
384            scrutinee.map(|scrut| self.is_known_valid_scrutinee(scrut)).unwrap_or(true);
385        PatCtxt {
386            tcx: self.tcx,
387            typeck_results: self.typeck_results,
388            typing_env: self.typing_env,
389            module: self.tcx.parent_module(self.hir_source).to_def_id(),
390            dropless_arena: self.dropless_arena,
391            match_lint_level: self.hir_source,
392            whole_match_span,
393            scrut_span,
394            refutable,
395            known_valid_scrutinee,
396            internal_state: Default::default(),
397        }
398    }
399
400    fn analyze_patterns(
401        &mut self,
402        cx: &PatCtxt<'p, 'tcx>,
403        arms: &[MatchArm<'p, 'tcx>],
404        scrut_ty: Ty<'tcx>,
405    ) -> Result<UsefulnessReport<'p, 'tcx>, ErrorGuaranteed> {
406        let report =
407            rustc_pattern_analysis::rustc::analyze_match(&cx, &arms, scrut_ty).map_err(|err| {
408                self.error = Err(err);
409                err
410            })?;
411
412        // Warn unreachable subpatterns.
413        for (arm, is_useful) in report.arm_usefulness.iter() {
414            if let Usefulness::Useful(redundant_subpats) = is_useful
415                && !redundant_subpats.is_empty()
416            {
417                let mut redundant_subpats = redundant_subpats.clone();
418                // Emit lints in the order in which they occur in the file.
419                redundant_subpats.sort_unstable_by_key(|(pat, _)| pat.data().span);
420                for (pat, explanation) in redundant_subpats {
421                    report_unreachable_pattern(cx, arm.arm_data, pat, &explanation, None)
422                }
423            }
424        }
425        Ok(report)
426    }
427
428    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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_let",
                                    "rustc_mir_build::thir::pattern::check_match",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/thir/pattern/check_match.rs"),
                                    ::tracing_core::__macro_support::Option::Some(428u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::thir::pattern::check_match"),
                                    ::tracing_core::field::FieldSet::new(&["pat", "scrutinee",
                                                    "span", "else_span"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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(&scrutinee)
                                                            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(&span)
                                                            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(&else_span)
                                                            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;
        }
        {
            if !(self.let_source != LetSource::None) {
                ::core::panicking::panic("assertion failed: self.let_source != LetSource::None")
            };
            let scrut = scrutinee.map(|id| &self.thir[id]);
            if let LetSource::PlainLet = self.let_source {
                if let hir::Node::LetStmt(&hir::LetStmt {
                        source: hir::LocalSource::AssignDesugar, .. }) =
                        self.tcx.hir_node(self.hir_source) {
                    self.check_binding_is_irrefutable(pat, "assignment",
                        Some(Inform { descr: "destructuring assignments" }), scrut,
                        None);
                } else {
                    self.check_binding_is_irrefutable(pat, "local binding",
                        Some(Inform { descr: "`let` bindings" }), scrut,
                        Some(span));
                }
            } else if let Ok(Irrefutable) =
                    self.is_let_irrefutable(pat, scrut) {
                if span.from_expansion() {
                    self.lint_single_let(span, None, None);
                    return;
                }
                let let_else_span =
                    self.check_irrefutable_option_some(pat, scrut, span);
                let sm = self.tcx.sess.source_map();
                let next_token_start =
                    sm.span_extend_while_whitespace(span.clone()).hi();
                let line_span =
                    sm.span_extend_to_line(span.clone()).with_lo(next_token_start);
                let else_keyword_span = sm.span_until_whitespace(line_span);
                self.lint_single_let(span, Some(else_keyword_span),
                    let_else_span);
            }
        }
    }
}#[instrument(level = "trace", skip(self))]
429    fn check_let(
430        &mut self,
431        pat: &'p Pat<'tcx>,
432        scrutinee: Option<ExprId>,
433        span: Span,
434        else_span: Option<Span>,
435    ) {
436        assert!(self.let_source != LetSource::None);
437        let scrut = scrutinee.map(|id| &self.thir[id]);
438        if let LetSource::PlainLet = self.let_source {
439            // `lhs = rhs` destructuring assignments are lowered to a `let` tagged
440            // `AssignDesugar`; report them as assignments, not `let` bindings (#157553).
441            if let hir::Node::LetStmt(&hir::LetStmt {
442                source: hir::LocalSource::AssignDesugar,
443                ..
444            }) = self.tcx.hir_node(self.hir_source)
445            {
446                self.check_binding_is_irrefutable(
447                    pat,
448                    "assignment",
449                    Some(Inform { descr: "destructuring assignments" }),
450                    scrut,
451                    None,
452                );
453            } else {
454                self.check_binding_is_irrefutable(
455                    pat,
456                    "local binding",
457                    Some(Inform { descr: "`let` bindings" }),
458                    scrut,
459                    Some(span),
460                );
461            }
462        } else if let Ok(Irrefutable) = self.is_let_irrefutable(pat, scrut) {
463            if span.from_expansion() {
464                self.lint_single_let(span, None, None);
465                return;
466            }
467            let let_else_span = self.check_irrefutable_option_some(pat, scrut, span);
468
469            let sm = self.tcx.sess.source_map();
470            let next_token_start = sm.span_extend_while_whitespace(span.clone()).hi();
471            let line_span = sm.span_extend_to_line(span.clone()).with_lo(next_token_start);
472            let else_keyword_span = sm.span_until_whitespace(line_span);
473            self.lint_single_let(span, Some(else_keyword_span), let_else_span);
474        }
475    }
476
477    /// Check case `let x = Some(y);`, user likely intended to destructure `Option`
478    fn check_irrefutable_option_some(
479        &self,
480        pat: &'p Pat<'tcx>,
481        initializer: Option<&Expr<'tcx>>,
482        span: Span,
483    ) -> Option<LetElseReplacementSuggestion> {
484        if let sm = self.tcx.sess.source_map()
485            && let Some(initializer) = initializer
486            && let Some(s_ty) = initializer.ty.ty_adt_def()
487            && self.tcx.is_diagnostic_item(rustc_span::sym::Option, s_ty.did())
488            && let ExprKind::Scope { value, .. } = initializer.kind
489            && let initializer_expr = &self.thir[value]
490            && let ExprKind::Adt(AdtExpr { fields, .. }) = &initializer_expr.kind
491            && let Some(field) = fields.first()
492            && let inner = &self.thir[field.expr]
493            && let Some(inner_ty) = inner.ty.ty_adt_def()
494            && self.tcx.is_diagnostic_item(rustc_span::sym::Option, inner_ty.did())
495            && let Ok(rhs) = sm.span_to_snippet(inner.span)
496            && let Ok(lhs) = sm.span_to_snippet(pat.span)
497        {
498            let lhs = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("Some({0})", lhs))
    })format!("Some({})", lhs);
499            Some(LetElseReplacementSuggestion { span, lhs, rhs })
500        } else {
501            None
502        }
503    }
504
505    fn check_match(
506        &mut self,
507        scrut: ExprId,
508        arms: &[ArmId],
509        source: hir::MatchSource,
510        expr_span: Span,
511    ) {
512        let scrut = &self.thir[scrut];
513        let cx = self.new_cx(Refutable, Some(expr_span), Some(scrut), scrut.span);
514
515        let mut tarms = Vec::with_capacity(arms.len());
516        for &arm in arms {
517            let arm = &self.thir.arms[arm];
518            let got_error = self.with_hir_source(arm.hir_id, |this| {
519                let Ok(pat) = this.lower_pattern(&cx, &arm.pattern) else { return true };
520                let arm =
521                    MatchArm { pat, arm_data: this.hir_source, has_guard: arm.guard.is_some() };
522                tarms.push(arm);
523                false
524            });
525            if got_error {
526                return;
527            }
528        }
529
530        let Ok(report) = self.analyze_patterns(&cx, &tarms, scrut.ty) else { return };
531
532        match source {
533            // Don't report arm reachability of desugared `match $iter.into_iter() { iter => .. }`
534            // when the iterator is an uninhabited type. unreachable_code will trigger instead.
535            hir::MatchSource::ForLoopDesugar if arms.len() == 1 => {}
536            hir::MatchSource::ForLoopDesugar
537            | hir::MatchSource::Postfix
538            | hir::MatchSource::Normal
539            | hir::MatchSource::FormatArgs => {
540                let is_match_arm =
541                    #[allow(non_exhaustive_omitted_patterns)] match source {
    hir::MatchSource::Postfix | hir::MatchSource::Normal => true,
    _ => false,
}matches!(source, hir::MatchSource::Postfix | hir::MatchSource::Normal);
542                report_arm_reachability(&cx, &report, is_match_arm);
543            }
544            // Unreachable patterns in try and await expressions occur when one of
545            // the arms are an uninhabited type. Which is OK.
546            hir::MatchSource::AwaitDesugar | hir::MatchSource::TryDesugar(_) => {}
547        }
548
549        // Check if the match is exhaustive.
550        let witnesses = report.non_exhaustiveness_witnesses;
551        if !witnesses.is_empty() {
552            if source == hir::MatchSource::ForLoopDesugar
553                && let [_, snd_arm] = *arms
554            {
555                // the for loop pattern is not irrefutable
556                let pat = &self.thir[snd_arm].pattern;
557                // `pat` should be `Some(<pat_field>)` from a desugared for loop.
558                if true {
    match (&pat.span.desugaring_kind(), &Some(DesugaringKind::ForLoop)) {
        (left_val, right_val) => {
            if !(*left_val == *right_val) {
                let kind = ::core::panicking::AssertKind::Eq;
                ::core::panicking::assert_failed(kind, &*left_val,
                    &*right_val, ::core::option::Option::None);
            }
        }
    };
};debug_assert_eq!(pat.span.desugaring_kind(), Some(DesugaringKind::ForLoop));
559                let PatKind::Variant { ref subpatterns, .. } = pat.kind else { ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!() };
560                let [pat_field] = &subpatterns[..] else { ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!() };
561                self.check_binding_is_irrefutable(
562                    &pat_field.pattern,
563                    "`for` loop binding",
564                    None,
565                    None,
566                    None,
567                );
568            } else {
569                // span after scrutinee, or after `.match`. That is, the braces, arms,
570                // and any whitespace preceding the braces.
571                let braces_span = match source {
572                    hir::MatchSource::Normal => scrut
573                        .span
574                        .find_ancestor_in_same_ctxt(expr_span)
575                        .map(|scrut_span| scrut_span.shrink_to_hi().with_hi(expr_span.hi())),
576                    hir::MatchSource::Postfix => {
577                        // This is horrendous, and we should deal with it by just
578                        // stashing the span of the braces somewhere (like in the match source).
579                        scrut.span.find_ancestor_in_same_ctxt(expr_span).and_then(|scrut_span| {
580                            let sm = self.tcx.sess.source_map();
581                            let brace_span = sm.span_extend_to_next_char(scrut_span, '{', true);
582                            if sm.span_to_snippet(sm.next_point(brace_span)).as_deref() == Ok("{") {
583                                let sp = brace_span.shrink_to_hi().with_hi(expr_span.hi());
584                                // We also need to extend backwards for whitespace
585                                sm.span_extend_prev_while(sp, |c| c.is_whitespace()).ok()
586                            } else {
587                                None
588                            }
589                        })
590                    }
591                    hir::MatchSource::ForLoopDesugar
592                    | hir::MatchSource::TryDesugar(_)
593                    | hir::MatchSource::AwaitDesugar
594                    | hir::MatchSource::FormatArgs => None,
595                };
596
597                // Check if the match would be exhaustive if all guards were removed.
598                // If so, we leave a note that guards don't count towards exhaustivity.
599                let would_be_exhaustive_without_guards = {
600                    let any_arm_has_guard = tarms.iter().any(|arm| arm.has_guard);
601                    any_arm_has_guard && {
602                        let guardless_arms: Vec<_> =
603                            tarms.iter().map(|arm| MatchArm { has_guard: false, ..*arm }).collect();
604                        rustc_pattern_analysis::rustc::analyze_match(&cx, &guardless_arms, scrut.ty)
605                            .is_ok_and(|report| report.non_exhaustiveness_witnesses.is_empty())
606                    }
607                };
608                self.error = Err(report_non_exhaustive_match(
609                    &cx,
610                    self.thir,
611                    scrut.ty,
612                    scrut.span,
613                    witnesses,
614                    arms,
615                    braces_span,
616                    would_be_exhaustive_without_guards,
617                ));
618            }
619        }
620    }
621
622    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("lint_single_let",
                                    "rustc_mir_build::thir::pattern::check_match",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/thir/pattern/check_match.rs"),
                                    ::tracing_core::__macro_support::Option::Some(622u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::thir::pattern::check_match"),
                                    ::tracing_core::field::FieldSet::new(&["let_span",
                                                    "else_keyword_span", "let_else_span"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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(&let_span)
                                                            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(&else_keyword_span)
                                                            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(&let_else_span)
                                                            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;
        }
        {
            report_irrefutable_let_patterns(self.tcx, self.hir_source,
                self.let_source, 1, let_span, else_keyword_span,
                let_else_span);
        }
    }
}#[instrument(level = "trace", skip(self))]
623    fn lint_single_let(
624        &mut self,
625        let_span: Span,
626        else_keyword_span: Option<Span>,
627        let_else_span: Option<LetElseReplacementSuggestion>,
628    ) {
629        report_irrefutable_let_patterns(
630            self.tcx,
631            self.hir_source,
632            self.let_source,
633            1,
634            let_span,
635            else_keyword_span,
636            let_else_span,
637        );
638    }
639
640    fn analyze_binding(
641        &mut self,
642        pat: &'p Pat<'tcx>,
643        refutability: RefutableFlag,
644        scrut: Option<&Expr<'tcx>>,
645    ) -> Result<(PatCtxt<'p, 'tcx>, UsefulnessReport<'p, 'tcx>), ErrorGuaranteed> {
646        let cx = self.new_cx(refutability, None, scrut, pat.span);
647        let pat = self.lower_pattern(&cx, pat)?;
648        let arms = [MatchArm { pat, arm_data: self.hir_source, has_guard: false }];
649        let report = self.analyze_patterns(&cx, &arms, pat.ty().inner())?;
650        Ok((cx, report))
651    }
652
653    fn is_let_irrefutable(
654        &mut self,
655        pat: &'p Pat<'tcx>,
656        scrut: Option<&Expr<'tcx>>,
657    ) -> Result<RefutableFlag, ErrorGuaranteed> {
658        let (cx, report) = self.analyze_binding(pat, Refutable, scrut)?;
659        // Report if the pattern is unreachable, which can only occur when the type is uninhabited.
660        report_arm_reachability(&cx, &report, false);
661        // If the list of witnesses is empty, the match is exhaustive, i.e. the `if let` pattern is
662        // irrefutable.
663        Ok(if report.non_exhaustiveness_witnesses.is_empty() { Irrefutable } else { Refutable })
664    }
665
666    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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_binding_is_irrefutable",
                                    "rustc_mir_build::thir::pattern::check_match",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_mir_build/src/thir/pattern/check_match.rs"),
                                    ::tracing_core::__macro_support::Option::Some(666u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_mir_build::thir::pattern::check_match"),
                                    ::tracing_core::field::FieldSet::new(&["pat", "origin",
                                                    "inform", "scrut", "sp"],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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(&origin 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(&inform)
                                                            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(&scrut)
                                                            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(&sp)
                                                            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 pattern_ty = pat.ty;
            let Ok((cx, report)) =
                self.analyze_binding(pat, Irrefutable, scrut) else { return };
            let witnesses = report.non_exhaustiveness_witnesses;
            if witnesses.is_empty() { return; }
            let mut let_suggestion = None;
            let mut misc_suggestion = None;
            let mut interpreted_as_const = None;
            let mut interpreted_as_const_sugg = None;
            if let Some(def_id) =
                    is_const_pat_that_looks_like_binding(self.tcx, pat) {
                let span = self.tcx.def_span(def_id);
                let variable = self.tcx.item_name(def_id).to_string();
                interpreted_as_const =
                    Some(InterpretedAsConst {
                            span,
                            variable: variable.clone(),
                        });
                interpreted_as_const_sugg =
                    Some(InterpretedAsConstSugg { span: pat.span, variable });
            } else if let PatKind::Constant { .. } = pat.kind &&
                    let Ok(snippet) =
                        self.tcx.sess.source_map().span_to_snippet(pat.span) {
                if snippet.chars().all(|c| c.is_digit(10)) {
                    misc_suggestion =
                        Some(MiscPatternSuggestion::AttemptedIntegerLiteral {
                                start_span: pat.span.shrink_to_lo(),
                            });
                }
            }
            if let Some(span) = sp &&
                            self.tcx.sess.source_map().is_span_accessible(span) &&
                        interpreted_as_const.is_none() && scrut.is_some() {
                let mut bindings = ::alloc::vec::Vec::new();
                pat.each_binding(|name, _, _, _| bindings.push(name));
                let semi_span = span.shrink_to_hi();
                let start_span = span.shrink_to_lo();
                let end_span = semi_span.shrink_to_lo();
                let count = witnesses.len();
                let_suggestion =
                    Some(if bindings.is_empty() {
                            SuggestLet::If { start_span, semi_span, count }
                        } else { SuggestLet::Else { end_span, count } });
            };
            let adt_defined_here =
                report_adt_defined_here(self.tcx, pattern_ty, &witnesses,
                    false);
            let witness_1_is_privately_uninhabited =
                if let Some(witness_1) = witnesses.get(0) &&
                                let ty::Adt(adt, args) = witness_1.ty().kind() &&
                            adt.is_enum() &&
                        let Constructor::Variant(variant_index) = witness_1.ctor() {
                    let variant_inhabited =
                        adt.variant(*variant_index).inhabited_predicate(self.tcx,
                                *adt).instantiate(self.tcx, args);
                    variant_inhabited.apply(self.tcx, cx.typing_env, cx.module)
                        &&
                        !variant_inhabited.apply_ignore_module(self.tcx,
                                cx.typing_env)
                } else { false };
            let witness_1 = cx.print_witness_pat(witnesses.get(0).unwrap());
            self.error =
                Err(self.tcx.dcx().emit_err(PatternNotCovered {
                            span: pat.span,
                            origin,
                            uncovered: Uncovered::new(pat.span, &cx, witnesses),
                            inform,
                            interpreted_as_const,
                            interpreted_as_const_sugg,
                            witness_1_is_privately_uninhabited,
                            witness_1,
                            _p: (),
                            pattern_ty,
                            let_suggestion,
                            misc_suggestion,
                            adt_defined_here,
                        }));
        }
    }
}#[instrument(level = "trace", skip(self))]
667    fn check_binding_is_irrefutable(
668        &mut self,
669        pat: &'p Pat<'tcx>,
670        origin: &str,
671        inform: Option<Inform>,
672        scrut: Option<&Expr<'tcx>>,
673        sp: Option<Span>,
674    ) {
675        let pattern_ty = pat.ty;
676
677        let Ok((cx, report)) = self.analyze_binding(pat, Irrefutable, scrut) else { return };
678        let witnesses = report.non_exhaustiveness_witnesses;
679        if witnesses.is_empty() {
680            // The pattern is irrefutable.
681            return;
682        }
683
684        let mut let_suggestion = None;
685        let mut misc_suggestion = None;
686        let mut interpreted_as_const = None;
687        let mut interpreted_as_const_sugg = None;
688
689        if let Some(def_id) = is_const_pat_that_looks_like_binding(self.tcx, pat) {
690            let span = self.tcx.def_span(def_id);
691            let variable = self.tcx.item_name(def_id).to_string();
692            // When we encounter a constant as the binding name, point at the `const` definition.
693            interpreted_as_const = Some(InterpretedAsConst { span, variable: variable.clone() });
694            interpreted_as_const_sugg = Some(InterpretedAsConstSugg { span: pat.span, variable });
695        } else if let PatKind::Constant { .. } = pat.kind
696            && let Ok(snippet) = self.tcx.sess.source_map().span_to_snippet(pat.span)
697        {
698            // If the pattern to match is an integer literal:
699            if snippet.chars().all(|c| c.is_digit(10)) {
700                // Then give a suggestion, the user might've meant to create a binding instead.
701                misc_suggestion = Some(MiscPatternSuggestion::AttemptedIntegerLiteral {
702                    start_span: pat.span.shrink_to_lo(),
703                });
704            }
705        }
706
707        if let Some(span) = sp
708            && self.tcx.sess.source_map().is_span_accessible(span)
709            && interpreted_as_const.is_none()
710            && scrut.is_some()
711        {
712            let mut bindings = vec![];
713            pat.each_binding(|name, _, _, _| bindings.push(name));
714
715            let semi_span = span.shrink_to_hi();
716            let start_span = span.shrink_to_lo();
717            let end_span = semi_span.shrink_to_lo();
718            let count = witnesses.len();
719
720            let_suggestion = Some(if bindings.is_empty() {
721                SuggestLet::If { start_span, semi_span, count }
722            } else {
723                SuggestLet::Else { end_span, count }
724            });
725        };
726
727        let adt_defined_here = report_adt_defined_here(self.tcx, pattern_ty, &witnesses, false);
728
729        // Emit an extra note if the first uncovered witness would be uninhabited
730        // if we disregard visibility.
731        let witness_1_is_privately_uninhabited = if let Some(witness_1) = witnesses.get(0)
732            && let ty::Adt(adt, args) = witness_1.ty().kind()
733            && adt.is_enum()
734            && let Constructor::Variant(variant_index) = witness_1.ctor()
735        {
736            let variant_inhabited = adt
737                .variant(*variant_index)
738                .inhabited_predicate(self.tcx, *adt)
739                .instantiate(self.tcx, args);
740            variant_inhabited.apply(self.tcx, cx.typing_env, cx.module)
741                && !variant_inhabited.apply_ignore_module(self.tcx, cx.typing_env)
742        } else {
743            false
744        };
745
746        let witness_1 = cx.print_witness_pat(witnesses.get(0).unwrap());
747
748        self.error = Err(self.tcx.dcx().emit_err(PatternNotCovered {
749            span: pat.span,
750            origin,
751            uncovered: Uncovered::new(pat.span, &cx, witnesses),
752            inform,
753            interpreted_as_const,
754            interpreted_as_const_sugg,
755            witness_1_is_privately_uninhabited,
756            witness_1,
757            _p: (),
758            pattern_ty,
759            let_suggestion,
760            misc_suggestion,
761            adt_defined_here,
762        }));
763    }
764}
765
766/// Check if a by-value binding is by-value. That is, check if the binding's type is not `Copy`.
767/// Check that there are no borrow or move conflicts in `binding @ subpat` patterns.
768///
769/// For example, this would reject:
770/// - `ref x @ Some(ref mut y)`,
771/// - `ref mut x @ Some(ref y)`,
772/// - `ref mut x @ Some(ref mut y)`,
773/// - `ref mut? x @ Some(y)`, and
774/// - `x @ Some(ref mut? y)`.
775///
776/// This analysis is *not* subsumed by NLL.
777fn check_borrow_conflicts_in_at_patterns<'tcx>(cx: &MatchVisitor<'_, 'tcx>, pat: &Pat<'tcx>) {
778    // Extract `sub` in `binding @ sub`.
779    let PatKind::Binding { name, mode, ty, subpattern: Some(ref sub), .. } = pat.kind else {
780        return;
781    };
782
783    let is_binding_by_move = |ty: Ty<'tcx>| !cx.tcx.type_is_copy_modulo_regions(cx.typing_env, ty);
784
785    let sess = cx.tcx.sess;
786
787    // Get the binding move, extract the mutability if by-ref.
788    let mut_outer = match mode.0 {
789        ByRef::No if is_binding_by_move(ty) => {
790            // We have `x @ pat` where `x` is by-move. Reject all borrows in `pat`.
791            let mut conflicts_ref = Vec::new();
792            sub.each_binding(|_, mode, _, span| {
793                if #[allow(non_exhaustive_omitted_patterns)] match mode {
    ByRef::Yes(..) => true,
    _ => false,
}matches!(mode, ByRef::Yes(..)) {
794                    conflicts_ref.push(span)
795                }
796            });
797            if !conflicts_ref.is_empty() {
798                sess.dcx().emit_err(BorrowOfMovedValue {
799                    binding_span: pat.span,
800                    conflicts_ref,
801                    name: Ident::new(name, pat.span),
802                    ty,
803                    suggest_borrowing: Some(pat.span.shrink_to_lo()),
804                });
805            }
806            return;
807        }
808        ByRef::No => return,
809        ByRef::Yes(_, m) => m,
810    };
811
812    // We now have `ref $mut_outer binding @ sub` (semantically).
813    // Recurse into each binding in `sub` and find mutability or move conflicts.
814    let mut conflicts_move = Vec::new();
815    let mut conflicts_mut_mut = Vec::new();
816    let mut conflicts_mut_ref = Vec::new();
817    sub.each_binding(|name, mode, ty, span| {
818        match mode {
819            ByRef::Yes(_, mut_inner) => match (mut_outer, mut_inner) {
820                // Both sides are `ref`.
821                (Mutability::Not, Mutability::Not) => {}
822                // 2x `ref mut`.
823                (Mutability::Mut, Mutability::Mut) => {
824                    conflicts_mut_mut.push(Conflict::Mut { span, name })
825                }
826                (Mutability::Not, Mutability::Mut) => {
827                    conflicts_mut_ref.push(Conflict::Mut { span, name })
828                }
829                (Mutability::Mut, Mutability::Not) => {
830                    conflicts_mut_ref.push(Conflict::Ref { span, name })
831                }
832            },
833            ByRef::No if is_binding_by_move(ty) => {
834                conflicts_move.push(Conflict::Moved { span, name }) // `ref mut?` + by-move conflict.
835            }
836            ByRef::No => {} // `ref mut?` + by-copy is fine.
837        }
838    });
839
840    let report_mut_mut = !conflicts_mut_mut.is_empty();
841    let report_mut_ref = !conflicts_mut_ref.is_empty();
842    let report_move_conflict = !conflicts_move.is_empty();
843
844    let mut occurrences = match mut_outer {
845        Mutability::Mut => ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [Conflict::Mut { span: pat.span, name }]))vec![Conflict::Mut { span: pat.span, name }],
846        Mutability::Not => ::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
        [Conflict::Ref { span: pat.span, name }]))vec![Conflict::Ref { span: pat.span, name }],
847    };
848    occurrences.extend(conflicts_mut_mut);
849    occurrences.extend(conflicts_mut_ref);
850    occurrences.extend(conflicts_move);
851
852    // Report errors if any.
853    if report_mut_mut {
854        // Report mutability conflicts for e.g. `ref mut x @ Some(ref mut y)`.
855        sess.dcx().emit_err(MultipleMutBorrows { span: pat.span, occurrences });
856    } else if report_mut_ref {
857        // Report mutability conflicts for e.g. `ref x @ Some(ref mut y)` or the converse.
858        match mut_outer {
859            Mutability::Mut => {
860                sess.dcx().emit_err(AlreadyMutBorrowed { span: pat.span, occurrences });
861            }
862            Mutability::Not => {
863                sess.dcx().emit_err(AlreadyBorrowed { span: pat.span, occurrences });
864            }
865        };
866    } else if report_move_conflict {
867        // Report by-ref and by-move conflicts, e.g. `ref x @ y`.
868        sess.dcx().emit_err(MovedWhileBorrowed { span: pat.span, occurrences });
869    }
870}
871
872fn check_for_bindings_named_same_as_variants(
873    cx: &MatchVisitor<'_, '_>,
874    pat: &Pat<'_>,
875    rf: RefutableFlag,
876) {
877    if let PatKind::Binding {
878        name,
879        mode: BindingMode(ByRef::No, Mutability::Not),
880        subpattern: None,
881        ty,
882        ..
883    } = pat.kind
884        && let ty::Adt(edef, _) = ty.peel_refs().kind()
885        && edef.is_enum()
886        && edef
887            .variants()
888            .iter()
889            .any(|variant| variant.name == name && variant.ctor_kind() == Some(CtorKind::Const))
890    {
891        let variant_count = edef.variants().len();
892        let ty_path = { let _guard = NoTrimmedGuard::new(); cx.tcx.def_path_str(edef.did()) }with_no_trimmed_paths!(cx.tcx.def_path_str(edef.did()));
893        cx.tcx.emit_node_span_lint(
894            BINDINGS_WITH_VARIANT_NAME,
895            cx.hir_source,
896            pat.span,
897            BindingsWithVariantName {
898                // If this is an irrefutable pattern, and there's > 1 variant,
899                // then we can't actually match on this. Applying the below
900                // suggestion would produce code that breaks on `check_binding_is_irrefutable`.
901                suggestion: if rf == Refutable || variant_count == 1 {
902                    Some(pat.span)
903                } else {
904                    None
905                },
906                ty_path,
907                name: Ident::new(name, pat.span),
908            },
909        )
910    }
911}
912
913/// Check that never patterns are only used on inhabited types.
914fn check_never_pattern<'tcx>(
915    cx: &PatCtxt<'_, 'tcx>,
916    pat: &Pat<'tcx>,
917) -> Result<(), ErrorGuaranteed> {
918    if let PatKind::Never = pat.kind {
919        if !cx.is_uninhabited(pat.ty) {
920            return Err(cx.tcx.dcx().emit_err(NonEmptyNeverPattern { span: pat.span, ty: pat.ty }));
921        }
922    }
923    Ok(())
924}
925
926fn report_irrefutable_let_patterns(
927    tcx: TyCtxt<'_>,
928    id: HirId,
929    source: LetSource,
930    count: usize,
931    span: Span,
932    else_keyword_span: Option<Span>,
933    let_else_span: Option<LetElseReplacementSuggestion>,
934) {
935    macro_rules! emit_diag {
936        ($lint:tt) => {{
937            tcx.emit_node_span_lint(IRREFUTABLE_LET_PATTERNS, id, span, $lint { count });
938        }};
939    }
940
941    match source {
942        LetSource::None | LetSource::PlainLet | LetSource::Else => ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!(),
943        LetSource::IfLet | LetSource::ElseIfLet => {
    tcx.emit_node_span_lint(IRREFUTABLE_LET_PATTERNS, id, span,
        IrrefutableLetPatternsIfLet { count });
}emit_diag!(IrrefutableLetPatternsIfLet),
944        LetSource::IfLetGuard => {
    tcx.emit_node_span_lint(IRREFUTABLE_LET_PATTERNS, id, span,
        IrrefutableLetPatternsIfLetGuard { count });
}emit_diag!(IrrefutableLetPatternsIfLetGuard),
945        LetSource::LetElse => {
946            let spans = match else_keyword_span {
947                Some(else_keyword_span) => {
948                    let mut spans = MultiSpan::from_span(else_keyword_span);
949                    spans.push_span_label(
950                        span,
951                        rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("assigning to binding pattern will always succeed"))msg!("assigning to binding pattern will always succeed"),
952                    );
953                    spans
954                }
955                None => span.into(),
956            };
957
958            tcx.emit_node_span_lint(
959                IRREFUTABLE_LET_PATTERNS,
960                id,
961                spans,
962                IrrefutableLetPatternsLetElse { be_replaced: let_else_span },
963            );
964        }
965        LetSource::WhileLet => {
    tcx.emit_node_span_lint(IRREFUTABLE_LET_PATTERNS, id, span,
        IrrefutableLetPatternsWhileLet { count });
}emit_diag!(IrrefutableLetPatternsWhileLet),
966    }
967}
968
969/// Report unreachable arms, if any.
970fn report_unreachable_pattern<'p, 'tcx>(
971    cx: &PatCtxt<'p, 'tcx>,
972    hir_id: HirId,
973    pat: &DeconstructedPat<'p, 'tcx>,
974    explanation: &RedundancyExplanation<'p, 'tcx>,
975    whole_arm_span: Option<Span>,
976) {
977    static CAP_COVERED_BY_MANY: usize = 4;
978    let pat_span = pat.data().span;
979    let mut lint = UnreachablePatternInner {
980        span: Some(pat_span),
981        matches_no_values: None,
982        matches_no_values_ty: **pat.ty(),
983        uninhabited_note: None,
984        covered_by_catchall: None,
985        covered_by_one: None,
986        covered_by_many: None,
987        wanted_constant: None,
988        accessible_constant: None,
989        inaccessible_constant: None,
990        pattern_let_binding: None,
991        suggest_remove: None,
992    };
993    let mut covered_by_many_n_more_count = None;
994    match explanation.covered_by.as_slice() {
995        [] => {
996            // Empty pattern; we report the uninhabited type that caused the emptiness.
997            lint.span = None; // Don't label the pattern itself
998            lint.uninhabited_note = Some(()); // Give a link about empty types
999            lint.matches_no_values = Some(pat_span);
1000            lint.suggest_remove = whole_arm_span; // Suggest to remove the match arm
1001            pat.walk(&mut |subpat| {
1002                let ty = **subpat.ty();
1003                if cx.is_uninhabited(ty) {
1004                    lint.matches_no_values_ty = ty;
1005                    false // No need to dig further.
1006                } else if #[allow(non_exhaustive_omitted_patterns)] match subpat.ctor() {
    Constructor::Ref | Constructor::UnionField => true,
    _ => false,
}matches!(subpat.ctor(), Constructor::Ref | Constructor::UnionField) {
1007                    false // Don't explore further since they are not by-value.
1008                } else {
1009                    true
1010                }
1011            });
1012        }
1013        [covering_pat] if pat_is_catchall(covering_pat) => {
1014            // A binding pattern that matches all, a single binding name.
1015            let pat = covering_pat.data();
1016            lint.covered_by_catchall = Some(pat.span);
1017            find_fallback_pattern_typo(cx, hir_id, pat, &mut lint);
1018        }
1019        [covering_pat] => {
1020            lint.covered_by_one = Some(covering_pat.data().span);
1021        }
1022        covering_pats => {
1023            let mut iter = covering_pats.iter();
1024            let mut multispan = MultiSpan::from_span(pat_span);
1025            for p in iter.by_ref().take(CAP_COVERED_BY_MANY) {
1026                multispan.push_span_label(p.data().span, rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("matches some of the same values"))msg!("matches some of the same values"));
1027            }
1028            let remain = iter.count();
1029            if remain == 0 {
1030                multispan.push_span_label(pat_span, rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("collectively making this unreachable"))msg!("collectively making this unreachable"));
1031            } else {
1032                covered_by_many_n_more_count = Some(remain);
1033                multispan.push_span_label(
1034                    pat_span,
1035                    rustc_errors::DiagMessage::Inline(std::borrow::Cow::Borrowed("...and {$covered_by_many_n_more_count} other patterns collectively make this unreachable"))msg!("...and {$covered_by_many_n_more_count} other patterns collectively make this unreachable"),
1036                );
1037            }
1038            lint.covered_by_many = Some(multispan);
1039        }
1040    }
1041    cx.tcx.emit_node_span_lint(
1042        UNREACHABLE_PATTERNS,
1043        hir_id,
1044        pat_span,
1045        UnreachablePattern { inner: lint, covered_by_many_n_more_count },
1046    );
1047}
1048
1049/// Detect typos that were meant to be a `const` but were interpreted as a new pattern binding.
1050fn find_fallback_pattern_typo<'tcx>(
1051    cx: &PatCtxt<'_, 'tcx>,
1052    hir_id: HirId,
1053    pat: &Pat<'tcx>,
1054    lint: &mut UnreachablePatternInner<'_>,
1055) {
1056    if cx.tcx.lint_level_spec_at_node(UNREACHABLE_PATTERNS, hir_id).is_allow() {
1057        // This is because we use `with_no_trimmed_paths` later, so if we never emit the lint we'd
1058        // ICE. At the same time, we don't really need to do all of this if we won't emit anything.
1059        return;
1060    }
1061    if let PatKind::Binding { name, subpattern: None, ty, .. } = pat.kind {
1062        // See if the binding might have been a `const` that was mistyped or out of scope.
1063        let mut accessible = ::alloc::vec::Vec::new()vec![];
1064        let mut accessible_path = ::alloc::vec::Vec::new()vec![];
1065        let mut inaccessible = ::alloc::vec::Vec::new()vec![];
1066        let mut imported = ::alloc::vec::Vec::new()vec![];
1067        let mut imported_spans = ::alloc::vec::Vec::new()vec![];
1068        let (infcx, param_env) = cx.tcx.infer_ctxt().build_with_typing_env(cx.typing_env);
1069        let parent = cx.tcx.hir_get_parent_item(hir_id);
1070
1071        for item in cx.tcx.hir_crate_items(()).free_items() {
1072            if let DefKind::Use = cx.tcx.def_kind(item.owner_id) {
1073                // Look for consts being re-exported.
1074                let item = cx.tcx.hir_expect_item(item.owner_id.def_id);
1075                let hir::ItemKind::Use(path, _) = item.kind else {
1076                    continue;
1077                };
1078                if let Some(value_ns) = path.res.value_ns
1079                    && let Res::Def(DefKind::Const { .. }, id) = value_ns
1080                    && infcx.can_eq(
1081                        param_env,
1082                        ty,
1083                        cx.tcx.type_of(id).instantiate_identity().skip_norm_wip(),
1084                    )
1085                {
1086                    if cx.tcx.visibility(id).is_accessible_from(parent, cx.tcx) {
1087                        // The original const is accessible, suggest using it directly.
1088                        let item_name = cx.tcx.item_name(id);
1089                        accessible.push(item_name);
1090                        accessible_path.push({ let _guard = NoTrimmedGuard::new(); cx.tcx.def_path_str(id) }with_no_trimmed_paths!(cx.tcx.def_path_str(id)));
1091                    } else if cx.tcx.visibility(item.owner_id).is_accessible_from(parent, cx.tcx) {
1092                        // The const is accessible only through the re-export, point at
1093                        // the `use`.
1094                        let ident = item.kind.ident().unwrap();
1095                        imported.push(ident.name);
1096                        imported_spans.push(ident.span);
1097                    }
1098                }
1099            }
1100            if let DefKind::Const { .. } = cx.tcx.def_kind(item.owner_id)
1101                && infcx.can_eq(
1102                    param_env,
1103                    ty,
1104                    cx.tcx.type_of(item.owner_id).instantiate_identity().skip_norm_wip(),
1105                )
1106            {
1107                // Look for local consts.
1108                let item_name = cx.tcx.item_name(item.owner_id);
1109                let vis = cx.tcx.visibility(item.owner_id);
1110                if vis.is_accessible_from(parent, cx.tcx) {
1111                    accessible.push(item_name);
1112                    // FIXME: the line below from PR #135310 is a workaround for the ICE in issue
1113                    // #135289, where a macro in a dependency can create unreachable patterns in the
1114                    // current crate. Path trimming expects diagnostics for a typoed const, but no
1115                    // diagnostics are emitted and we ICE. See
1116                    // `tests/ui/resolve/const-with-typo-in-pattern-binding-ice-135289.rs` for a
1117                    // test that reproduces the ICE if we don't use `with_no_trimmed_paths!`.
1118                    let path = { let _guard = NoTrimmedGuard::new(); cx.tcx.def_path_str(item.owner_id) }with_no_trimmed_paths!(cx.tcx.def_path_str(item.owner_id));
1119                    accessible_path.push(path);
1120                } else if name == item_name {
1121                    // The const exists somewhere in this crate, but it can't be imported
1122                    // from this pattern's scope. We'll just point at its definition.
1123                    inaccessible.push(cx.tcx.def_span(item.owner_id));
1124                }
1125            }
1126        }
1127        if let Some((i, &const_name)) =
1128            accessible.iter().enumerate().find(|&(_, &const_name)| const_name == name)
1129        {
1130            // The pattern name is an exact match, so the pattern needed to be imported.
1131            lint.wanted_constant = Some(WantedConstant {
1132                span: pat.span,
1133                is_typo: false,
1134                const_name: const_name.to_string(),
1135                const_path: accessible_path[i].clone(),
1136            });
1137        } else if let Some(name) = find_best_match_for_name(&accessible, name, None) {
1138            // The pattern name is likely a typo.
1139            lint.wanted_constant = Some(WantedConstant {
1140                span: pat.span,
1141                is_typo: true,
1142                const_name: name.to_string(),
1143                const_path: name.to_string(),
1144            });
1145        } else if let Some(i) =
1146            imported.iter().enumerate().find(|&(_, &const_name)| const_name == name).map(|(i, _)| i)
1147        {
1148            // The const with the exact name wasn't re-exported from an import in this
1149            // crate, we point at the import.
1150            lint.accessible_constant = Some(imported_spans[i]);
1151        } else if let Some(name) = find_best_match_for_name(&imported, name, None) {
1152            // The typoed const wasn't re-exported by an import in this crate, we suggest
1153            // the right name (which will likely require another follow up suggestion).
1154            lint.wanted_constant = Some(WantedConstant {
1155                span: pat.span,
1156                is_typo: true,
1157                const_path: name.to_string(),
1158                const_name: name.to_string(),
1159            });
1160        } else if !inaccessible.is_empty() {
1161            for span in inaccessible {
1162                // The const with the exact name match isn't accessible, we just point at it.
1163                lint.inaccessible_constant = Some(span);
1164            }
1165        } else {
1166            // Look for local bindings for people that might have gotten confused with how
1167            // `let` and `const` works.
1168            for (_, node) in cx.tcx.hir_parent_iter(hir_id) {
1169                match node {
1170                    hir::Node::Stmt(hir::Stmt { kind: hir::StmtKind::Let(let_stmt), .. }) => {
1171                        if let hir::PatKind::Binding(_, _, binding_name, _) = let_stmt.pat.kind {
1172                            if name == binding_name.name {
1173                                lint.pattern_let_binding = Some(binding_name.span);
1174                            }
1175                        }
1176                    }
1177                    hir::Node::Block(hir::Block { stmts, .. }) => {
1178                        for stmt in *stmts {
1179                            if let hir::StmtKind::Let(let_stmt) = stmt.kind
1180                                && let hir::PatKind::Binding(_, _, binding_name, _) =
1181                                    let_stmt.pat.kind
1182                                && name == binding_name.name
1183                            {
1184                                lint.pattern_let_binding = Some(binding_name.span);
1185                            }
1186                        }
1187                    }
1188                    hir::Node::Item(_) => break,
1189                    _ => {}
1190                }
1191            }
1192        }
1193    }
1194}
1195
1196/// Report unreachable arms, if any.
1197fn report_arm_reachability<'p, 'tcx>(
1198    cx: &PatCtxt<'p, 'tcx>,
1199    report: &UsefulnessReport<'p, 'tcx>,
1200    is_match_arm: bool,
1201) {
1202    let sm = cx.tcx.sess.source_map();
1203    for (arm, is_useful) in report.arm_usefulness.iter() {
1204        if let Usefulness::Redundant(explanation) = is_useful {
1205            let hir_id = arm.arm_data;
1206            let arm_span = cx.tcx.hir_span(hir_id);
1207            let whole_arm_span = if is_match_arm {
1208                // If the arm is followed by a comma, extend the span to include it.
1209                if let Some(comma) = sm.span_followed_by(arm_span, ",") {
1210                    Some(arm_span.to(comma))
1211                } else {
1212                    Some(arm_span)
1213                }
1214            } else {
1215                None
1216            };
1217            report_unreachable_pattern(cx, hir_id, arm.pat, explanation, whole_arm_span)
1218        }
1219    }
1220}
1221
1222/// Checks for common cases of "catchall" patterns that may not be intended as such.
1223fn pat_is_catchall(pat: &DeconstructedPat<'_, '_>) -> bool {
1224    match pat.ctor() {
1225        Constructor::Wildcard => true,
1226        Constructor::Struct | Constructor::Ref => {
1227            pat.iter_fields().all(|ipat| pat_is_catchall(&ipat.pat))
1228        }
1229        _ => false,
1230    }
1231}
1232
1233/// If the given pattern is a named constant that looks like it could have been
1234/// intended to be a binding, returns the `DefId` of the named constant.
1235///
1236/// Diagnostics use this to give more detailed suggestions for non-exhaustive
1237/// matches.
1238fn is_const_pat_that_looks_like_binding<'tcx>(tcx: TyCtxt<'tcx>, pat: &Pat<'tcx>) -> Option<DefId> {
1239    // The pattern must be a named constant, and the name that appears in
1240    // the pattern's source text must resemble a plain identifier without any
1241    // `::` namespace separators or other non-identifier characters.
1242    if let Some(def_id) = try { pat.extra.as_deref()?.expanded_const? }
1243        && #[allow(non_exhaustive_omitted_patterns)] match tcx.def_kind(def_id) {
    DefKind::Const { .. } => true,
    _ => false,
}matches!(tcx.def_kind(def_id), DefKind::Const { .. })
1244        && let Ok(snippet) = tcx.sess.source_map().span_to_snippet(pat.span)
1245        && snippet.chars().all(|c| c.is_alphanumeric() || c == '_')
1246    {
1247        Some(def_id)
1248    } else {
1249        None
1250    }
1251}
1252
1253/// Report that a match is not exhaustive.
1254fn report_non_exhaustive_match<'p, 'tcx>(
1255    cx: &PatCtxt<'p, 'tcx>,
1256    thir: &Thir<'tcx>,
1257    scrut_ty: Ty<'tcx>,
1258    sp: Span,
1259    witnesses: Vec<WitnessPat<'p, 'tcx>>,
1260    arms: &[ArmId],
1261    braces_span: Option<Span>,
1262    would_be_exhaustive_without_guards: bool,
1263) -> ErrorGuaranteed {
1264    let is_empty_match = arms.is_empty();
1265    let non_empty_enum = match scrut_ty.kind() {
1266        ty::Adt(def, _) => def.is_enum() && !def.variants().is_empty(),
1267        _ => false,
1268    };
1269    // In the case of an empty match, replace the '`_` not covered' diagnostic with something more
1270    // informative.
1271    if is_empty_match && !non_empty_enum {
1272        return cx.tcx.dcx().emit_err(NonExhaustivePatternsTypeNotEmpty {
1273            cx,
1274            scrut_span: sp,
1275            braces_span,
1276            ty: scrut_ty,
1277        });
1278    }
1279
1280    // FIXME: migration of this diagnostic will require list support
1281    let joined_patterns = joined_uncovered_patterns(cx, &witnesses);
1282    let mut err = {
    cx.tcx.dcx().struct_span_err(sp,
            ::alloc::__export::must_use({
                    ::alloc::fmt::format(format_args!("non-exhaustive patterns: {0} not covered",
                            joined_patterns))
                })).with_code(E0004)
}struct_span_code_err!(
1283        cx.tcx.dcx(),
1284        sp,
1285        E0004,
1286        "non-exhaustive patterns: {joined_patterns} not covered"
1287    );
1288    err.span_label(
1289        sp,
1290        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("pattern{0} {1} not covered",
                if witnesses.len() == 1 { "" } else { "s" }, joined_patterns))
    })format!(
1291            "pattern{} {} not covered",
1292            rustc_errors::pluralize!(witnesses.len()),
1293            joined_patterns
1294        ),
1295    );
1296
1297    // Point at the definition of non-covered `enum` variants.
1298    if let Some(AdtDefinedHere { adt_def_span, ty, variants }) =
1299        report_adt_defined_here(cx.tcx, scrut_ty, &witnesses, true)
1300    {
1301        let mut multi_span = MultiSpan::from_span(adt_def_span);
1302        multi_span.push_span_label(adt_def_span, "");
1303        for Variant { span } in variants {
1304            multi_span.push_span_label(span, "not covered");
1305        }
1306        err.span_note(multi_span, ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` defined here", ty))
    })format!("`{ty}` defined here"));
1307    }
1308    err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("the matched value is of type `{0}`",
                scrut_ty))
    })format!("the matched value is of type `{}`", scrut_ty));
1309
1310    if !is_empty_match {
1311        let mut special_tys = FxIndexSet::default();
1312        // Look at the first witness.
1313        collect_special_tys(cx, &witnesses[0], &mut special_tys);
1314
1315        for ty in special_tys {
1316            if ty.is_ptr_sized_integral() {
1317                if ty.inner() == cx.tcx.types.usize {
1318                    err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}::MAX` is not treated as exhaustive, so half-open ranges are necessary to match exhaustively",
                ty))
    })format!(
1319                        "`{ty}::MAX` is not treated as exhaustive, \
1320                        so half-open ranges are necessary to match exhaustively",
1321                    ));
1322                } else if ty.inner() == cx.tcx.types.isize {
1323                    err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}::MIN` and `{0}::MAX` are not treated as exhaustive, so half-open ranges are necessary to match exhaustively",
                ty))
    })format!(
1324                        "`{ty}::MIN` and `{ty}::MAX` are not treated as exhaustive, \
1325                        so half-open ranges are necessary to match exhaustively",
1326                    ));
1327                }
1328            } else if ty.inner() == cx.tcx.types.str_ {
1329                err.note("`&str` cannot be matched exhaustively, so a wildcard `_` is necessary");
1330            } else if cx.is_foreign_non_exhaustive_enum(ty) {
1331                err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` is marked as non-exhaustive, so a wildcard `_` is necessary to match exhaustively",
                ty))
    })format!("`{ty}` is marked as non-exhaustive, so a wildcard `_` is necessary to match exhaustively"));
1332            } else if cx.is_uninhabited(ty.inner()) {
1333                // The type is uninhabited yet there is a witness: we must be in the `MaybeInvalid`
1334                // case.
1335                err.note(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` is uninhabited but is not being matched by value, so a wildcard `_` is required",
                ty))
    })format!("`{ty}` is uninhabited but is not being matched by value, so a wildcard `_` is required"));
1336            }
1337        }
1338    }
1339
1340    if let ty::Ref(_, sub_ty, _) = scrut_ty.kind() {
1341        if !sub_ty.is_inhabited_from(cx.tcx, cx.module, cx.typing_env) {
1342            err.note("references are always considered inhabited");
1343        }
1344    }
1345
1346    for &arm in arms {
1347        let arm = &thir.arms[arm];
1348        if let Some(def_id) = is_const_pat_that_looks_like_binding(cx.tcx, &arm.pattern) {
1349            let const_name = cx.tcx.item_name(def_id);
1350            err.span_label(
1351                arm.pattern.span,
1352                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("this pattern doesn\'t introduce a new catch-all binding, but rather pattern matches against the value of constant `{0}`",
                const_name))
    })format!(
1353                    "this pattern doesn't introduce a new catch-all binding, but rather pattern \
1354                     matches against the value of constant `{const_name}`",
1355                ),
1356            );
1357            err.span_note(cx.tcx.def_span(def_id), ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("constant `{0}` defined here",
                const_name))
    })format!("constant `{const_name}` defined here"));
1358            err.span_suggestion_verbose(
1359                arm.pattern.span.shrink_to_hi(),
1360                "if you meant to introduce a binding, use a different name",
1361                "_var".to_string(),
1362                Applicability::MaybeIncorrect,
1363            );
1364        }
1365    }
1366
1367    // Whether we suggest the actual missing patterns or `_`.
1368    let suggest_the_witnesses = witnesses.len() < 4;
1369    let suggested_arm = if suggest_the_witnesses {
1370        let pattern = witnesses
1371            .iter()
1372            .map(|witness| cx.print_witness_pat(witness))
1373            .collect::<Vec<String>>()
1374            .join(" | ");
1375        if witnesses.iter().all(|p| p.is_never_pattern()) && cx.tcx.features().never_patterns() {
1376            // Arms with a never pattern don't take a body.
1377            pattern
1378        } else {
1379            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0} => todo!()", pattern))
    })format!("{pattern} => todo!()")
1380        }
1381    } else {
1382        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("_ => todo!()"))
    })format!("_ => todo!()")
1383    };
1384    let mut suggestion = None;
1385    let sm = cx.tcx.sess.source_map();
1386    match arms {
1387        [] if let Some(braces_span) = braces_span => {
1388            // Get the span for the empty match body `{}`.
1389            let (indentation, more) = if let Some(snippet) = sm.indentation_before(sp) {
1390                (::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("\n{0}", snippet))
    })format!("\n{snippet}"), "    ")
1391            } else {
1392                (" ".to_string(), "")
1393            };
1394            suggestion = Some((
1395                braces_span,
1396                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!(" {{{0}{1}{2},{0}}}", indentation,
                more, suggested_arm))
    })format!(" {{{indentation}{more}{suggested_arm},{indentation}}}",),
1397            ));
1398        }
1399        [only] => {
1400            let only = &thir[*only];
1401            let (pre_indentation, is_multiline) = if let Some(snippet) =
1402                sm.indentation_before(only.span)
1403                && let Ok(with_trailing) =
1404                    sm.span_extend_while(only.span, |c| c.is_whitespace() || c == ',')
1405                && sm.is_multiline(with_trailing)
1406            {
1407                (::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("\n{0}", snippet))
    })format!("\n{snippet}"), true)
1408            } else {
1409                (" ".to_string(), false)
1410            };
1411            let only_body = &thir[only.body];
1412            let comma = if #[allow(non_exhaustive_omitted_patterns)] match only_body.kind {
    ExprKind::Block { .. } => true,
    _ => false,
}matches!(only_body.kind, ExprKind::Block { .. })
1413                && only.span.eq_ctxt(only_body.span)
1414                && is_multiline
1415            {
1416                ""
1417            } else {
1418                ","
1419            };
1420            suggestion = Some((
1421                only.span.shrink_to_hi(),
1422                ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}{1}{2}", comma, pre_indentation,
                suggested_arm))
    })format!("{comma}{pre_indentation}{suggested_arm}"),
1423            ));
1424        }
1425        [.., prev, last] => {
1426            let prev = &thir[*prev];
1427            let last = &thir[*last];
1428            if prev.span.eq_ctxt(last.span) {
1429                let last_body = &thir[last.body];
1430                let comma = if #[allow(non_exhaustive_omitted_patterns)] match last_body.kind {
    ExprKind::Block { .. } => true,
    _ => false,
}matches!(last_body.kind, ExprKind::Block { .. })
1431                    && last.span.eq_ctxt(last_body.span)
1432                {
1433                    ""
1434                } else {
1435                    ","
1436                };
1437                let spacing = if sm.is_multiline(prev.span.between(last.span)) {
1438                    sm.indentation_before(last.span).map(|indent| ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("\n{0}", indent))
    })format!("\n{indent}"))
1439                } else {
1440                    Some(" ".to_string())
1441                };
1442                if let Some(spacing) = spacing {
1443                    suggestion = Some((
1444                        last.span.shrink_to_hi(),
1445                        ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("{0}{1}{2}", comma, spacing,
                suggested_arm))
    })format!("{comma}{spacing}{suggested_arm}"),
1446                    ));
1447                }
1448            }
1449        }
1450        _ => {}
1451    }
1452
1453    let msg = ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("ensure that all possible cases are being handled by adding a match arm with a wildcard pattern{0}{1}",
                if witnesses.len() > 1 && suggest_the_witnesses &&
                        suggestion.is_some() {
                    ", a match arm with multiple or-patterns"
                } else { "" },
                match witnesses.len() {
                    0 if suggestion.is_some() => " as shown",
                    0 => "",
                    1 if suggestion.is_some() =>
                        " or an explicit pattern as shown",
                    1 => " or an explicit pattern",
                    _ if suggestion.is_some() =>
                        " as shown, or multiple match arms",
                    _ => " or multiple match arms",
                }))
    })format!(
1454        "ensure that all possible cases are being handled by adding a match arm with a wildcard \
1455         pattern{}{}",
1456        if witnesses.len() > 1 && suggest_the_witnesses && suggestion.is_some() {
1457            ", a match arm with multiple or-patterns"
1458        } else {
1459            // we are either not suggesting anything, or suggesting `_`
1460            ""
1461        },
1462        match witnesses.len() {
1463            // non-exhaustive enum case
1464            0 if suggestion.is_some() => " as shown",
1465            0 => "",
1466            1 if suggestion.is_some() => " or an explicit pattern as shown",
1467            1 => " or an explicit pattern",
1468            _ if suggestion.is_some() => " as shown, or multiple match arms",
1469            _ => " or multiple match arms",
1470        },
1471    );
1472
1473    if would_be_exhaustive_without_guards {
1474        err.subdiagnostic(NonExhaustiveMatchAllArmsGuarded);
1475    }
1476    if let Some((span, sugg)) = suggestion {
1477        err.span_suggestion_verbose(span, msg, sugg, Applicability::HasPlaceholders);
1478    } else {
1479        err.help(msg);
1480    }
1481    err.emit()
1482}
1483
1484fn joined_uncovered_patterns<'p, 'tcx>(
1485    cx: &PatCtxt<'p, 'tcx>,
1486    witnesses: &[WitnessPat<'p, 'tcx>],
1487) -> String {
1488    const LIMIT: usize = 3;
1489    let pat_to_str = |pat: &WitnessPat<'p, 'tcx>| cx.print_witness_pat(pat);
1490    match witnesses {
1491        [] => ::rustc_middle::util::bug::bug_fmt(format_args!("impossible case reached"))bug!(),
1492        [witness] => ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}`",
                cx.print_witness_pat(witness)))
    })format!("`{}`", cx.print_witness_pat(witness)),
1493        [head @ .., tail] if head.len() < LIMIT => {
1494            let head: Vec<_> = head.iter().map(pat_to_str).collect();
1495            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` and `{1}`",
                head.join("`, `"), cx.print_witness_pat(tail)))
    })format!("`{}` and `{}`", head.join("`, `"), cx.print_witness_pat(tail))
1496        }
1497        _ => {
1498            let (head, tail) = witnesses.split_at(LIMIT);
1499            let head: Vec<_> = head.iter().map(pat_to_str).collect();
1500            ::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("`{0}` and {1} more",
                head.join("`, `"), tail.len()))
    })format!("`{}` and {} more", head.join("`, `"), tail.len())
1501        }
1502    }
1503}
1504
1505/// Collect types that require specific explanations when they show up in witnesses.
1506fn collect_special_tys<'tcx>(
1507    cx: &PatCtxt<'_, 'tcx>,
1508    pat: &WitnessPat<'_, 'tcx>,
1509    special_tys: &mut FxIndexSet<RevealedTy<'tcx>>,
1510) {
1511    if #[allow(non_exhaustive_omitted_patterns)] match pat.ctor() {
    Constructor::NonExhaustive | Constructor::Never => true,
    _ => false,
}matches!(pat.ctor(), Constructor::NonExhaustive | Constructor::Never) {
1512        special_tys.insert(*pat.ty());
1513    }
1514    if let Constructor::IntRange(range) = pat.ctor() {
1515        if cx.is_range_beyond_boundaries(range, *pat.ty()) {
1516            // The range denotes the values before `isize::MIN` or the values after `usize::MAX`/`isize::MAX`.
1517            special_tys.insert(*pat.ty());
1518        }
1519    }
1520    pat.iter_fields().for_each(|field_pat| collect_special_tys(cx, field_pat, special_tys))
1521}
1522
1523fn report_adt_defined_here<'tcx>(
1524    tcx: TyCtxt<'tcx>,
1525    ty: Ty<'tcx>,
1526    witnesses: &[WitnessPat<'_, 'tcx>],
1527    point_at_non_local_ty: bool,
1528) -> Option<AdtDefinedHere<'tcx>> {
1529    let ty = ty.peel_refs();
1530    let ty::Adt(def, _) = ty.kind() else {
1531        return None;
1532    };
1533    let adt_def_span =
1534        tcx.hir_get_if_local(def.did()).and_then(|node| node.ident()).map(|ident| ident.span);
1535    let adt_def_span = if point_at_non_local_ty {
1536        adt_def_span.unwrap_or_else(|| tcx.def_span(def.did()))
1537    } else {
1538        adt_def_span?
1539    };
1540
1541    let mut variants = ::alloc::vec::Vec::new()vec![];
1542    for span in maybe_point_at_variant(tcx, *def, witnesses.iter().take(5)) {
1543        variants.push(Variant { span });
1544    }
1545    Some(AdtDefinedHere { adt_def_span, ty, variants })
1546}
1547
1548fn maybe_point_at_variant<'a, 'p: 'a, 'tcx: 'p>(
1549    tcx: TyCtxt<'tcx>,
1550    def: AdtDef<'tcx>,
1551    patterns: impl Iterator<Item = &'a WitnessPat<'p, 'tcx>>,
1552) -> Vec<Span> {
1553    let mut covered = ::alloc::vec::Vec::new()vec![];
1554    for pattern in patterns {
1555        if let Constructor::Variant(variant_index) = pattern.ctor() {
1556            if let ty::Adt(this_def, _) = pattern.ty().kind()
1557                && this_def.did() != def.did()
1558            {
1559                continue;
1560            }
1561            let sp = def.variant(*variant_index).ident(tcx).span;
1562            if covered.contains(&sp) {
1563                // Don't point at variants that have already been covered due to other patterns to avoid
1564                // visual clutter.
1565                continue;
1566            }
1567            covered.push(sp);
1568        }
1569        covered.extend(maybe_point_at_variant(tcx, def, pattern.iter_fields()));
1570    }
1571    covered
1572}