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 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 _ 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 error: Result<(), ErrorGuaranteed>,
101}
102
103impl<'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 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 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 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 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 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 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 fn is_known_valid_scrutinee(&self, scrutinee: &Expr<'tcx>) -> bool {
294 use ExprKind::*;
295 match &scrutinee.kind {
296 Deref { .. } => false,
299 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 Index { lhs, .. } => {
309 let lhs = &self.thir()[*lhs];
310 self.is_known_valid_scrutinee(lhs)
311 }
312
313 Scope { value, .. } => self.is_known_valid_scrutinee(&self.thir()[*value]),
315
316 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 Become { .. }
329 | Break { .. }
330 | Continue { .. }
331 | ConstContinue { .. }
332 | Return { .. } => true,
333
334 Assign { .. } | AssignOp { .. } | InlineAsm { .. } | Let { .. } => true,
336
337 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 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 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 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 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 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 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 hir::MatchSource::AwaitDesugar | hir::MatchSource::TryDesugar(_) => {}
547 }
548
549 let witnesses = report.non_exhaustiveness_witnesses;
551 if !witnesses.is_empty() {
552 if source == hir::MatchSource::ForLoopDesugar
553 && let [_, snd_arm] = *arms
554 {
555 let pat = &self.thir[snd_arm].pattern;
557 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 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 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 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 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_arm_reachability(&cx, &report, false);
661 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 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 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 snippet.chars().all(|c| c.is_digit(10)) {
700 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 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
766fn check_borrow_conflicts_in_at_patterns<'tcx>(cx: &MatchVisitor<'_, 'tcx>, pat: &Pat<'tcx>) {
778 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 let mut_outer = match mode.0 {
789 ByRef::No if is_binding_by_move(ty) => {
790 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 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 (Mutability::Not, Mutability::Not) => {}
822 (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 }) }
836 ByRef::No => {} }
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 if report_mut_mut {
854 sess.dcx().emit_err(MultipleMutBorrows { span: pat.span, occurrences });
856 } else if report_mut_ref {
857 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 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 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
913fn 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
969fn 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 lint.span = None; lint.uninhabited_note = Some(()); lint.matches_no_values = Some(pat_span);
1000 lint.suggest_remove = whole_arm_span; pat.walk(&mut |subpat| {
1002 let ty = **subpat.ty();
1003 if cx.is_uninhabited(ty) {
1004 lint.matches_no_values_ty = ty;
1005 false } 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 } else {
1009 true
1010 }
1011 });
1012 }
1013 [covering_pat] if pat_is_catchall(covering_pat) => {
1014 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
1049fn 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 return;
1060 }
1061 if let PatKind::Binding { name, subpattern: None, ty, .. } = pat.kind {
1062 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 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 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 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 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 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 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 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 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 lint.accessible_constant = Some(imported_spans[i]);
1151 } else if let Some(name) = find_best_match_for_name(&imported, name, None) {
1152 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 lint.inaccessible_constant = Some(span);
1164 }
1165 } else {
1166 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
1196fn 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 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
1222fn 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
1233fn is_const_pat_that_looks_like_binding<'tcx>(tcx: TyCtxt<'tcx>, pat: &Pat<'tcx>) -> Option<DefId> {
1239 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
1253fn 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 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 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 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 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 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 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 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 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 ""
1461 },
1462 match witnesses.len() {
1463 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
1505fn 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 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 continue;
1566 }
1567 covered.push(sp);
1568 }
1569 covered.extend(maybe_point_at_variant(tcx, def, pattern.iter_fields()));
1570 }
1571 covered
1572}