Skip to content

Commit 355efac

Browse files
committed
Auto merge of rust-lang#128034 - Nadrieril:explain-unreachable, r=compiler-errors
exhaustiveness: Explain why a given pattern is considered unreachable This PR tells the user why a given pattern is considered unreachable. I reused the intersection information we were already computing; even though it's incomplete I convinced myself that it is sufficient to always get a set of patterns that cover the unreachable one. I'm not a fan of the diagnostic messages I came up with, I'm open to suggestions. Fixes rust-lang#127870. This is also the other one of the two diagnostic improvements I wanted to do before rust-lang#122792. Note: the first commit is an unrelated drive-by tweak. r? `@compiler-errors`
2 parents 83d6768 + 940769a commit 355efac

File tree

53 files changed

+1526
-308
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

53 files changed

+1526
-308
lines changed

compiler/rustc_mir_build/messages.ftl

+8-1
Original file line numberDiff line numberDiff line change
@@ -325,9 +325,16 @@ mir_build_union_field_requires_unsafe_unsafe_op_in_unsafe_fn_allowed =
325325
326326
mir_build_union_pattern = cannot use unions in constant patterns
327327
328+
mir_build_unreachable_making_this_unreachable = collectively making this unreachable
329+
330+
mir_build_unreachable_matches_same_values = matches some of the same values
331+
328332
mir_build_unreachable_pattern = unreachable pattern
329333
.label = unreachable pattern
330-
.catchall_label = matches any value
334+
.unreachable_matches_no_values = this pattern matches no values because `{$ty}` is uninhabited
335+
.unreachable_covered_by_catchall = matches any value
336+
.unreachable_covered_by_one = matches all the values already
337+
.unreachable_covered_by_many = these patterns collectively make the last one unreachable
331338
332339
mir_build_unsafe_fn_safe_body = an unsafe function restricts its caller, but its body is safe by default
333340
mir_build_unsafe_not_inherited = items do not inherit unsafety from separate enclosing items

compiler/rustc_mir_build/src/errors.rs

+15-3
Original file line numberDiff line numberDiff line change
@@ -582,11 +582,23 @@ pub(crate) struct NonConstPath {
582582

583583
#[derive(LintDiagnostic)]
584584
#[diag(mir_build_unreachable_pattern)]
585-
pub(crate) struct UnreachablePattern {
585+
pub(crate) struct UnreachablePattern<'tcx> {
586586
#[label]
587587
pub(crate) span: Option<Span>,
588-
#[label(mir_build_catchall_label)]
589-
pub(crate) catchall: Option<Span>,
588+
#[subdiagnostic]
589+
pub(crate) matches_no_values: Option<UnreachableMatchesNoValues<'tcx>>,
590+
#[label(mir_build_unreachable_covered_by_catchall)]
591+
pub(crate) covered_by_catchall: Option<Span>,
592+
#[label(mir_build_unreachable_covered_by_one)]
593+
pub(crate) covered_by_one: Option<Span>,
594+
#[note(mir_build_unreachable_covered_by_many)]
595+
pub(crate) covered_by_many: Option<MultiSpan>,
596+
}
597+
598+
#[derive(Subdiagnostic)]
599+
#[note(mir_build_unreachable_matches_no_values)]
600+
pub(crate) struct UnreachableMatchesNoValues<'tcx> {
601+
pub(crate) ty: Ty<'tcx>,
590602
}
591603

592604
#[derive(Diagnostic)]

compiler/rustc_mir_build/src/thir/pattern/check_match.rs

+64-25
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
use crate::errors::*;
2+
use crate::fluent_generated as fluent;
23

34
use rustc_arena::{DroplessArena, TypedArena};
45
use rustc_ast::Mutability;
@@ -16,8 +17,8 @@ use rustc_middle::ty::print::with_no_trimmed_paths;
1617
use rustc_middle::ty::{self, AdtDef, Ty, TyCtxt};
1718
use rustc_pattern_analysis::errors::Uncovered;
1819
use rustc_pattern_analysis::rustc::{
19-
Constructor, DeconstructedPat, MatchArm, RevealedTy, RustcPatCtxt as PatCtxt, Usefulness,
20-
UsefulnessReport, WitnessPat,
20+
Constructor, DeconstructedPat, MatchArm, RedundancyExplanation, RevealedTy,
21+
RustcPatCtxt as PatCtxt, Usefulness, UsefulnessReport, WitnessPat,
2122
};
2223
use rustc_session::lint::builtin::{
2324
BINDINGS_WITH_VARIANT_NAME, IRREFUTABLE_LET_PATTERNS, UNREACHABLE_PATTERNS,
@@ -391,12 +392,16 @@ impl<'p, 'tcx> MatchVisitor<'p, 'tcx> {
391392
) -> Result<UsefulnessReport<'p, 'tcx>, ErrorGuaranteed> {
392393
let pattern_complexity_limit =
393394
get_limit_size(cx.tcx.hir().krate_attrs(), cx.tcx.sess, sym::pattern_complexity);
394-
let report =
395-
rustc_pattern_analysis::analyze_match(&cx, &arms, scrut_ty, pattern_complexity_limit)
396-
.map_err(|err| {
397-
self.error = Err(err);
398-
err
399-
})?;
395+
let report = rustc_pattern_analysis::rustc::analyze_match(
396+
&cx,
397+
&arms,
398+
scrut_ty,
399+
pattern_complexity_limit,
400+
)
401+
.map_err(|err| {
402+
self.error = Err(err);
403+
err
404+
})?;
400405

401406
// Warn unreachable subpatterns.
402407
for (arm, is_useful) in report.arm_usefulness.iter() {
@@ -405,9 +410,9 @@ impl<'p, 'tcx> MatchVisitor<'p, 'tcx> {
405410
{
406411
let mut redundant_subpats = redundant_subpats.clone();
407412
// Emit lints in the order in which they occur in the file.
408-
redundant_subpats.sort_unstable_by_key(|pat| pat.data().span);
409-
for pat in redundant_subpats {
410-
report_unreachable_pattern(cx, arm.arm_data, pat.data().span, None)
413+
redundant_subpats.sort_unstable_by_key(|(pat, _)| pat.data().span);
414+
for (pat, explanation) in redundant_subpats {
415+
report_unreachable_pattern(cx, arm.arm_data, pat, &explanation)
411416
}
412417
}
413418
}
@@ -906,26 +911,60 @@ fn report_irrefutable_let_patterns(
906911
fn report_unreachable_pattern<'p, 'tcx>(
907912
cx: &PatCtxt<'p, 'tcx>,
908913
hir_id: HirId,
909-
span: Span,
910-
catchall: Option<Span>,
914+
pat: &DeconstructedPat<'p, 'tcx>,
915+
explanation: &RedundancyExplanation<'p, 'tcx>,
911916
) {
912-
cx.tcx.emit_node_span_lint(
913-
UNREACHABLE_PATTERNS,
914-
hir_id,
915-
span,
916-
UnreachablePattern { span: if catchall.is_some() { Some(span) } else { None }, catchall },
917-
);
917+
let pat_span = pat.data().span;
918+
let mut lint = UnreachablePattern {
919+
span: Some(pat_span),
920+
matches_no_values: None,
921+
covered_by_catchall: None,
922+
covered_by_one: None,
923+
covered_by_many: None,
924+
};
925+
match explanation.covered_by.as_slice() {
926+
[] => {
927+
// Empty pattern; we report the uninhabited type that caused the emptiness.
928+
lint.span = None; // Don't label the pattern itself
929+
pat.walk(&mut |subpat| {
930+
let ty = **subpat.ty();
931+
if cx.is_uninhabited(ty) {
932+
lint.matches_no_values = Some(UnreachableMatchesNoValues { ty });
933+
false // No need to dig further.
934+
} else if matches!(subpat.ctor(), Constructor::Ref | Constructor::UnionField) {
935+
false // Don't explore further since they are not by-value.
936+
} else {
937+
true
938+
}
939+
});
940+
}
941+
[covering_pat] if pat_is_catchall(covering_pat) => {
942+
lint.covered_by_catchall = Some(covering_pat.data().span);
943+
}
944+
[covering_pat] => {
945+
lint.covered_by_one = Some(covering_pat.data().span);
946+
}
947+
covering_pats => {
948+
let mut multispan = MultiSpan::from_span(pat_span);
949+
for p in covering_pats {
950+
multispan.push_span_label(
951+
p.data().span,
952+
fluent::mir_build_unreachable_matches_same_values,
953+
);
954+
}
955+
multispan
956+
.push_span_label(pat_span, fluent::mir_build_unreachable_making_this_unreachable);
957+
lint.covered_by_many = Some(multispan);
958+
}
959+
}
960+
cx.tcx.emit_node_span_lint(UNREACHABLE_PATTERNS, hir_id, pat_span, lint);
918961
}
919962

920963
/// Report unreachable arms, if any.
921964
fn report_arm_reachability<'p, 'tcx>(cx: &PatCtxt<'p, 'tcx>, report: &UsefulnessReport<'p, 'tcx>) {
922-
let mut catchall = None;
923965
for (arm, is_useful) in report.arm_usefulness.iter() {
924-
if matches!(is_useful, Usefulness::Redundant) {
925-
report_unreachable_pattern(cx, arm.arm_data, arm.pat.data().span, catchall)
926-
}
927-
if !arm.has_guard && catchall.is_none() && pat_is_catchall(arm.pat) {
928-
catchall = Some(arm.pat.data().span);
966+
if let Usefulness::Redundant(explanation) = is_useful {
967+
report_unreachable_pattern(cx, arm.arm_data, arm.pat, explanation)
929968
}
930969
}
931970
}

compiler/rustc_pattern_analysis/src/lib.rs

+3-34
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
1-
//! Analysis of patterns, notably match exhaustiveness checking.
1+
//! Analysis of patterns, notably match exhaustiveness checking. The main entrypoint for this crate
2+
//! is [`usefulness::compute_match_usefulness`]. For rustc-specific types and entrypoints, see the
3+
//! [`rustc`] module.
24
35
// tidy-alphabetical-start
46
#![allow(rustc::diagnostic_outside_of_impl)]
@@ -23,14 +25,8 @@ use std::fmt;
2325

2426
pub use rustc_index::{Idx, IndexVec}; // re-exported to avoid rustc_index version issues
2527

26-
#[cfg(feature = "rustc")]
27-
use rustc_middle::ty::Ty;
28-
#[cfg(feature = "rustc")]
29-
use rustc_span::ErrorGuaranteed;
30-
3128
use crate::constructor::{Constructor, ConstructorSet, IntRange};
3229
use crate::pat::DeconstructedPat;
33-
use crate::pat_column::PatternColumn;
3430

3531
pub trait Captures<'a> {}
3632
impl<'a, T: ?Sized> Captures<'a> for T {}
@@ -128,30 +124,3 @@ impl<'p, Cx: PatCx> Clone for MatchArm<'p, Cx> {
128124
}
129125

130126
impl<'p, Cx: PatCx> Copy for MatchArm<'p, Cx> {}
131-
132-
/// The entrypoint for this crate. Computes whether a match is exhaustive and which of its arms are
133-
/// useful, and runs some lints.
134-
#[cfg(feature = "rustc")]
135-
pub fn analyze_match<'p, 'tcx>(
136-
tycx: &rustc::RustcPatCtxt<'p, 'tcx>,
137-
arms: &[rustc::MatchArm<'p, 'tcx>],
138-
scrut_ty: Ty<'tcx>,
139-
pattern_complexity_limit: Option<usize>,
140-
) -> Result<rustc::UsefulnessReport<'p, 'tcx>, ErrorGuaranteed> {
141-
use lints::lint_nonexhaustive_missing_variants;
142-
use usefulness::{compute_match_usefulness, PlaceValidity};
143-
144-
let scrut_ty = tycx.reveal_opaque_ty(scrut_ty);
145-
let scrut_validity = PlaceValidity::from_bool(tycx.known_valid_scrutinee);
146-
let report =
147-
compute_match_usefulness(tycx, arms, scrut_ty, scrut_validity, pattern_complexity_limit)?;
148-
149-
// Run the non_exhaustive_omitted_patterns lint. Only run on refutable patterns to avoid hitting
150-
// `if let`s. Only run if the match is exhaustive otherwise the error is redundant.
151-
if tycx.refutable && report.non_exhaustiveness_witnesses.is_empty() {
152-
let pat_column = PatternColumn::new(arms);
153-
lint_nonexhaustive_missing_variants(tycx, arms, &pat_column, scrut_ty)?;
154-
}
155-
156-
Ok(report)
157-
}

compiler/rustc_pattern_analysis/src/pat.rs

+16-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ use crate::{PatCx, PrivateUninhabitedField};
1111
use self::Constructor::*;
1212

1313
/// A globally unique id to distinguish patterns.
14-
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
14+
#[derive(Copy, Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
1515
pub(crate) struct PatId(u32);
1616
impl PatId {
1717
fn new() -> Self {
@@ -147,6 +147,21 @@ impl<Cx: PatCx> fmt::Debug for DeconstructedPat<Cx> {
147147
}
148148
}
149149

150+
/// Delegate to `uid`.
151+
impl<Cx: PatCx> PartialEq for DeconstructedPat<Cx> {
152+
fn eq(&self, other: &Self) -> bool {
153+
self.uid == other.uid
154+
}
155+
}
156+
/// Delegate to `uid`.
157+
impl<Cx: PatCx> Eq for DeconstructedPat<Cx> {}
158+
/// Delegate to `uid`.
159+
impl<Cx: PatCx> std::hash::Hash for DeconstructedPat<Cx> {
160+
fn hash<H: std::hash::Hasher>(&self, state: &mut H) {
161+
self.uid.hash(state);
162+
}
163+
}
164+
150165
/// Represents either a pattern obtained from user input or a wildcard constructed during the
151166
/// algorithm. Do not use `Wild` to represent a wildcard pattern comping from user input.
152167
///

compiler/rustc_pattern_analysis/src/rustc.rs

+28
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,9 @@ use rustc_target::abi::{FieldIdx, Integer, VariantIdx, FIRST_VARIANT};
2020
use crate::constructor::{
2121
IntRange, MaybeInfiniteInt, OpaqueId, RangeEnd, Slice, SliceKind, VariantVisibility,
2222
};
23+
use crate::lints::lint_nonexhaustive_missing_variants;
24+
use crate::pat_column::PatternColumn;
25+
use crate::usefulness::{compute_match_usefulness, PlaceValidity};
2326
use crate::{errors, Captures, PatCx, PrivateUninhabitedField};
2427

2528
use crate::constructor::Constructor::*;
@@ -29,6 +32,8 @@ pub type Constructor<'p, 'tcx> = crate::constructor::Constructor<RustcPatCtxt<'p
2932
pub type ConstructorSet<'p, 'tcx> = crate::constructor::ConstructorSet<RustcPatCtxt<'p, 'tcx>>;
3033
pub type DeconstructedPat<'p, 'tcx> = crate::pat::DeconstructedPat<RustcPatCtxt<'p, 'tcx>>;
3134
pub type MatchArm<'p, 'tcx> = crate::MatchArm<'p, RustcPatCtxt<'p, 'tcx>>;
35+
pub type RedundancyExplanation<'p, 'tcx> =
36+
crate::usefulness::RedundancyExplanation<'p, RustcPatCtxt<'p, 'tcx>>;
3237
pub type Usefulness<'p, 'tcx> = crate::usefulness::Usefulness<'p, RustcPatCtxt<'p, 'tcx>>;
3338
pub type UsefulnessReport<'p, 'tcx> =
3439
crate::usefulness::UsefulnessReport<'p, RustcPatCtxt<'p, 'tcx>>;
@@ -1058,3 +1063,26 @@ fn expand_or_pat<'p, 'tcx>(pat: &'p Pat<'tcx>) -> Vec<&'p Pat<'tcx>> {
10581063
expand(pat, &mut pats);
10591064
pats
10601065
}
1066+
1067+
/// The entrypoint for this crate. Computes whether a match is exhaustive and which of its arms are
1068+
/// useful, and runs some lints.
1069+
pub fn analyze_match<'p, 'tcx>(
1070+
tycx: &RustcPatCtxt<'p, 'tcx>,
1071+
arms: &[MatchArm<'p, 'tcx>],
1072+
scrut_ty: Ty<'tcx>,
1073+
pattern_complexity_limit: Option<usize>,
1074+
) -> Result<UsefulnessReport<'p, 'tcx>, ErrorGuaranteed> {
1075+
let scrut_ty = tycx.reveal_opaque_ty(scrut_ty);
1076+
let scrut_validity = PlaceValidity::from_bool(tycx.known_valid_scrutinee);
1077+
let report =
1078+
compute_match_usefulness(tycx, arms, scrut_ty, scrut_validity, pattern_complexity_limit)?;
1079+
1080+
// Run the non_exhaustive_omitted_patterns lint. Only run on refutable patterns to avoid hitting
1081+
// `if let`s. Only run if the match is exhaustive otherwise the error is redundant.
1082+
if tycx.refutable && report.non_exhaustiveness_witnesses.is_empty() {
1083+
let pat_column = PatternColumn::new(arms);
1084+
lint_nonexhaustive_missing_variants(tycx, arms, &pat_column, scrut_ty)?;
1085+
}
1086+
1087+
Ok(report)
1088+
}

0 commit comments

Comments
 (0)