Skip to content

Commit 493111e

Browse files
committed
Lint non_exhaustive_omitted_patterns per column
1 parent 6c346d1 commit 493111e

10 files changed

+262
-201
lines changed

compiler/rustc_lint_defs/src/builtin.rs

+21-13
Original file line numberDiff line numberDiff line change
@@ -3993,8 +3993,13 @@ declare_lint! {
39933993
}
39943994

39953995
declare_lint! {
3996-
/// The `non_exhaustive_omitted_patterns` lint detects when a wildcard (`_` or `..`) in a
3997-
/// pattern for a `#[non_exhaustive]` struct or enum is reachable.
3996+
/// The `non_exhaustive_omitted_patterns` lint aims to help consumers of a `#[non_exhaustive]`
3997+
/// struct or enum who want to match all of its fields/variants explicitly.
3998+
///
3999+
/// The `#[non_exhaustive]` annotation forces matches to use wildcards, so exhaustiveness
4000+
/// checking cannot be used to ensure that all fields/variants are matched explicitly. To remedy
4001+
/// this, this allow-by-default lint warns the user when a match mentions some but not all of
4002+
/// the fields/variants of a `#[non_exhaustive]` struct or enum.
39984003
///
39994004
/// ### Example
40004005
///
@@ -4008,39 +4013,42 @@ declare_lint! {
40084013
///
40094014
/// // in crate B
40104015
/// #![feature(non_exhaustive_omitted_patterns_lint)]
4016+
/// #[warn(non_exhaustive_omitted_patterns)]
40114017
/// match Bar::A {
40124018
/// Bar::A => {},
4013-
/// #[warn(non_exhaustive_omitted_patterns)]
40144019
/// _ => {},
40154020
/// }
40164021
/// ```
40174022
///
40184023
/// This will produce:
40194024
///
40204025
/// ```text
4021-
/// warning: reachable patterns not covered of non exhaustive enum
4026+
/// warning: some variants are not matched explicitly
40224027
/// --> $DIR/reachable-patterns.rs:70:9
40234028
/// |
4024-
/// LL | _ => {}
4025-
/// | ^ pattern `B` not covered
4029+
/// LL | match Bar::A {
4030+
/// | ^ pattern `Bar::B` not covered
40264031
/// |
40274032
/// note: the lint level is defined here
40284033
/// --> $DIR/reachable-patterns.rs:69:16
40294034
/// |
40304035
/// LL | #[warn(non_exhaustive_omitted_patterns)]
40314036
/// | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
4032-
/// = help: ensure that all possible cases are being handled by adding the suggested match arms
4037+
/// = help: ensure that all variants are matched explicitly by adding the suggested match arms
40334038
/// = note: the matched value is of type `Bar` and the `non_exhaustive_omitted_patterns` attribute was found
40344039
/// ```
40354040
///
4041+
/// Warning: setting this to `deny` will make upstream non-breaking changes (adding fields or
4042+
/// variants to a `#[non_exhaustive]` struct or enum) break your crate. This goes against
4043+
/// expected semver behavior.
4044+
///
40364045
/// ### Explanation
40374046
///
4038-
/// Structs and enums tagged with `#[non_exhaustive]` force the user to add a
4039-
/// (potentially redundant) wildcard when pattern-matching, to allow for future
4040-
/// addition of fields or variants. The `non_exhaustive_omitted_patterns` lint
4041-
/// detects when such a wildcard happens to actually catch some fields/variants.
4042-
/// In other words, when the match without the wildcard would not be exhaustive.
4043-
/// This lets the user be informed if new fields/variants were added.
4047+
/// Structs and enums tagged with `#[non_exhaustive]` force the user to add a (potentially
4048+
/// redundant) wildcard when pattern-matching, to allow for future addition of fields or
4049+
/// variants. The `non_exhaustive_omitted_patterns` lint detects when such a wildcard happens to
4050+
/// actually catch some fields/variants. In other words, when the match without the wildcard
4051+
/// would not be exhaustive. This lets the user be informed if new fields/variants were added.
40444052
pub NON_EXHAUSTIVE_OMITTED_PATTERNS,
40454053
Allow,
40464054
"detect when patterns of types marked `non_exhaustive` are missed",

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

+3-3
Original file line numberDiff line numberDiff line change
@@ -269,7 +269,7 @@ impl<'p, 'tcx> MatchVisitor<'_, 'p, 'tcx> {
269269

270270
let scrut = &self.thir[scrut];
271271
let scrut_ty = scrut.ty;
272-
let report = compute_match_usefulness(&cx, &tarms, self.lint_level, scrut_ty);
272+
let report = compute_match_usefulness(&cx, &tarms, self.lint_level, scrut_ty, scrut.span);
273273

274274
match source {
275275
// Don't report arm reachability of desugared `match $iter.into_iter() { iter => .. }`
@@ -431,7 +431,7 @@ impl<'p, 'tcx> MatchVisitor<'_, 'p, 'tcx> {
431431
let pattern = self.lower_pattern(&mut cx, pat);
432432
let pattern_ty = pattern.ty();
433433
let arm = MatchArm { pat: pattern, hir_id: self.lint_level, has_guard: false };
434-
let report = compute_match_usefulness(&cx, &[arm], self.lint_level, pattern_ty);
434+
let report = compute_match_usefulness(&cx, &[arm], self.lint_level, pattern_ty, pattern.span());
435435

436436
// Note: we ignore whether the pattern is unreachable (i.e. whether the type is empty). We
437437
// only care about exhaustiveness here.
@@ -622,7 +622,7 @@ fn is_let_irrefutable<'p, 'tcx>(
622622
pat: &'p DeconstructedPat<'p, 'tcx>,
623623
) -> bool {
624624
let arms = [MatchArm { pat, hir_id: pat_id, has_guard: false }];
625-
let report = compute_match_usefulness(&cx, &arms, pat_id, pat.ty());
625+
let report = compute_match_usefulness(&cx, &arms, pat_id, pat.ty(), pat.span());
626626

627627
// Report if the pattern is unreachable, which can only occur when the type is uninhabited.
628628
// This also reports unreachable sub-patterns though, so we can't just replace it with an

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

+18-28
Original file line numberDiff line numberDiff line change
@@ -629,18 +629,11 @@ pub(super) enum Constructor<'tcx> {
629629
/// `#[doc(hidden)]` ones.
630630
Hidden,
631631
/// Fake extra constructor for constructors that are not seen in the matrix, as explained in the
632-
/// code for [`Constructor::split`]. The carried `bool` is used for the
633-
/// `non_exhaustive_omitted_patterns` lint.
634-
Missing {
635-
nonexhaustive_enum_missing_visible_variants: bool,
636-
},
632+
/// code for [`Constructor::split`].
633+
Missing,
637634
}
638635

639636
impl<'tcx> Constructor<'tcx> {
640-
pub(super) fn is_wildcard(&self) -> bool {
641-
matches!(self, Wildcard)
642-
}
643-
644637
pub(super) fn is_non_exhaustive(&self) -> bool {
645638
matches!(self, NonExhaustive)
646639
}
@@ -778,14 +771,8 @@ impl<'tcx> Constructor<'tcx> {
778771
let all_missing = split_set.present.is_empty();
779772
let report_when_all_missing =
780773
pcx.is_top_level && !IntRange::is_integral(pcx.ty);
781-
let ctor = if all_missing && !report_when_all_missing {
782-
Wildcard
783-
} else {
784-
Missing {
785-
nonexhaustive_enum_missing_visible_variants: split_set
786-
.nonexhaustive_enum_missing_visible_variants,
787-
}
788-
};
774+
let ctor =
775+
if all_missing && !report_when_all_missing { Wildcard } else { Missing };
789776
smallvec![ctor]
790777
} else {
791778
split_set.present
@@ -905,11 +892,9 @@ pub(super) enum ConstructorSet {
905892
/// either fully included in or disjoint from each constructor in the column. This avoids
906893
/// non-trivial intersections like between `0..10` and `5..15`.
907894
#[derive(Debug)]
908-
struct SplitConstructorSet<'tcx> {
909-
present: SmallVec<[Constructor<'tcx>; 1]>,
910-
missing: Vec<Constructor<'tcx>>,
911-
/// For the `non_exhaustive_omitted_patterns` lint.
912-
nonexhaustive_enum_missing_visible_variants: bool,
895+
pub(super) struct SplitConstructorSet<'tcx> {
896+
pub(super) present: SmallVec<[Constructor<'tcx>; 1]>,
897+
pub(super) missing: Vec<Constructor<'tcx>>,
913898
}
914899

915900
impl ConstructorSet {
@@ -1039,7 +1024,7 @@ impl ConstructorSet {
10391024
/// constructors to 1/ determine which constructors of the type (if any) are missing; 2/ split
10401025
/// constructors to handle non-trivial intersections e.g. on ranges or slices.
10411026
#[instrument(level = "debug", skip(self, pcx, ctors), ret)]
1042-
fn split<'a, 'tcx>(
1027+
pub(super) fn split<'a, 'tcx>(
10431028
&self,
10441029
pcx: &PatCtxt<'_, '_, 'tcx>,
10451030
ctors: impl Iterator<Item = &'a Constructor<'tcx>> + Clone,
@@ -1051,7 +1036,6 @@ impl ConstructorSet {
10511036
let mut missing = Vec::new();
10521037
// Constructors in `ctors`, except wildcards.
10531038
let mut seen = ctors.filter(|c| !(matches!(c, Opaque | Wildcard)));
1054-
let mut nonexhaustive_enum_missing_visible_variants = false;
10551039
match self {
10561040
ConstructorSet::Single => {
10571041
if seen.next().is_none() {
@@ -1063,6 +1047,7 @@ impl ConstructorSet {
10631047
ConstructorSet::Variants { visible_variants, hidden_variants, non_exhaustive } => {
10641048
let seen_set: FxHashSet<_> = seen.map(|c| c.as_variant().unwrap()).collect();
10651049
let mut skipped_a_hidden_variant = false;
1050+
10661051
for variant in visible_variants {
10671052
let ctor = Variant(*variant);
10681053
if seen_set.contains(&variant) {
@@ -1071,8 +1056,6 @@ impl ConstructorSet {
10711056
missing.push(ctor);
10721057
}
10731058
}
1074-
nonexhaustive_enum_missing_visible_variants =
1075-
*non_exhaustive && !missing.is_empty();
10761059

10771060
for variant in hidden_variants {
10781061
let ctor = Variant(*variant);
@@ -1159,7 +1142,7 @@ impl ConstructorSet {
11591142
ConstructorSet::Uninhabited => {}
11601143
}
11611144

1162-
SplitConstructorSet { present, missing, nonexhaustive_enum_missing_visible_variants }
1145+
SplitConstructorSet { present, missing }
11631146
}
11641147

11651148
/// Compute the set of constructors missing from this column.
@@ -1519,6 +1502,13 @@ impl<'p, 'tcx> DeconstructedPat<'p, 'tcx> {
15191502
pub(super) fn is_or_pat(&self) -> bool {
15201503
matches!(self.ctor, Or)
15211504
}
1505+
pub(super) fn flatten_or_pat(&'p self) -> SmallVec<[&'p Self; 1]> {
1506+
if self.is_or_pat() {
1507+
self.iter_fields().flat_map(|p| p.flatten_or_pat()).collect()
1508+
} else {
1509+
smallvec![self]
1510+
}
1511+
}
15221512

15231513
pub(super) fn ctor(&self) -> &Constructor<'tcx> {
15241514
&self.ctor
@@ -1704,7 +1694,7 @@ impl<'p, 'tcx> fmt::Debug for DeconstructedPat<'p, 'tcx> {
17041694
#[derive(Debug, Clone)]
17051695
pub(crate) struct WitnessPat<'tcx> {
17061696
ctor: Constructor<'tcx>,
1707-
fields: Vec<WitnessPat<'tcx>>,
1697+
pub(crate) fields: Vec<WitnessPat<'tcx>>,
17081698
ty: Ty<'tcx>,
17091699
}
17101700

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

+100-47
Original file line numberDiff line numberDiff line change
@@ -844,8 +844,6 @@ fn is_useful<'p, 'tcx>(
844844
}
845845
// We split the head constructor of `v`.
846846
let split_ctors = v_ctor.split(pcx, matrix.heads().map(DeconstructedPat::ctor));
847-
let is_non_exhaustive_and_wild =
848-
cx.is_foreign_non_exhaustive_enum(ty) && v_ctor.is_wildcard();
849847
// For each constructor, we compute whether there's a value that starts with it that would
850848
// witness the usefulness of `v`.
851849
let start_matrix = &matrix;
@@ -866,50 +864,6 @@ fn is_useful<'p, 'tcx>(
866864
)
867865
});
868866
let usefulness = usefulness.apply_constructor(pcx, start_matrix, &ctor);
869-
870-
// When all the conditions are met we have a match with a `non_exhaustive` enum
871-
// that has the potential to trigger the `non_exhaustive_omitted_patterns` lint.
872-
// To understand the workings checkout `Constructor::split` and `SplitWildcard::new/into_ctors`
873-
if is_non_exhaustive_and_wild
874-
// Only emit a lint on refutable patterns.
875-
&& cx.refutable
876-
// We check that the match has a wildcard pattern and that wildcard is useful,
877-
// meaning there are variants that are covered by the wildcard. Without the check
878-
// for `witness_preference` the lint would trigger on `if let NonExhaustiveEnum::A = foo {}`
879-
&& usefulness.is_useful() && matches!(witness_preference, RealArm)
880-
&& matches!(
881-
&ctor,
882-
Constructor::Missing { nonexhaustive_enum_missing_visible_variants: true }
883-
)
884-
{
885-
let missing = ConstructorSet::for_ty(pcx.cx, pcx.ty)
886-
.compute_missing(pcx, matrix.heads().map(DeconstructedPat::ctor));
887-
// Construct for each missing constructor a "wild" version of this constructor, that
888-
// matches everything that can be built with it. For example, if `ctor` is a
889-
// `Constructor::Variant` for `Option::Some`, we get the pattern `Some(_)`.
890-
let patterns = missing
891-
.into_iter()
892-
// Because of how we computed `nonexhaustive_enum_missing_visible_variants`,
893-
// this will not return an empty `Vec`.
894-
.filter(|c| !(matches!(c, Constructor::NonExhaustive | Constructor::Hidden)))
895-
.map(|missing_ctor| WitnessPat::wild_from_ctor(pcx, missing_ctor))
896-
.collect::<Vec<_>>();
897-
898-
// Report that a match of a `non_exhaustive` enum marked with `non_exhaustive_omitted_patterns`
899-
// is not exhaustive enough.
900-
//
901-
// NB: The partner lint for structs lives in `compiler/rustc_hir_analysis/src/check/pat.rs`.
902-
cx.tcx.emit_spanned_lint(
903-
NON_EXHAUSTIVE_OMITTED_PATTERNS,
904-
lint_root,
905-
pcx.span,
906-
NonExhaustiveOmittedPattern {
907-
scrut_ty: pcx.ty,
908-
uncovered: Uncovered::new(pcx.span, pcx.cx, patterns),
909-
},
910-
);
911-
}
912-
913867
ret.extend(usefulness);
914868
}
915869
}
@@ -921,6 +875,74 @@ fn is_useful<'p, 'tcx>(
921875
ret
922876
}
923877

878+
/// Traverse the patterns to collect any variants of a non_exhaustive enum that fail to be mentioned
879+
/// in a given column. This traverses patterns column-by-column, where a column is the intuitive
880+
/// notion of "subpatterns that inspect the same subvalue".
881+
/// Despite similarities with `is_useful`, this traversal is different. Notably this is linear in the
882+
/// depth of patterns, whereas `is_useful` is worst-case exponential (exhaustiveness is NP-complete).
883+
fn collect_nonexhaustive_missing_variants<'p, 'tcx>(
884+
cx: &MatchCheckCtxt<'p, 'tcx>,
885+
column: &[&DeconstructedPat<'p, 'tcx>],
886+
) -> Vec<WitnessPat<'tcx>> {
887+
let ty = column[0].ty();
888+
let pcx = &PatCtxt { cx, ty, span: DUMMY_SP, is_top_level: false };
889+
890+
let mut witnesses = Vec::new();
891+
892+
let set = ConstructorSet::for_ty(pcx.cx, pcx.ty).split(pcx, column.iter().map(|p| p.ctor()));
893+
if cx.is_foreign_non_exhaustive_enum(ty) {
894+
witnesses.extend(
895+
set.missing
896+
.into_iter()
897+
// This will list missing visible variants.
898+
.filter(|c| !matches!(c, Constructor::Hidden | Constructor::NonExhaustive))
899+
.map(|missing_ctor| WitnessPat::wild_from_ctor(pcx, missing_ctor)),
900+
)
901+
}
902+
903+
// Recurse into the fields.
904+
for ctor in set.present {
905+
let arity = ctor.arity(pcx);
906+
if arity == 0 {
907+
continue;
908+
}
909+
910+
// We specialize the column by `ctor`. This gives us `arity`-many columns of patterns. These
911+
// columns may have different lengths in the presence of or-patterns (this is why we can't
912+
// reuse `Matrix`).
913+
let mut specialized_columns: Vec<Vec<_>> = (0..arity).map(|_| Vec::new()).collect();
914+
let relevant_patterns = column.iter().filter(|pat| ctor.is_covered_by(pcx, pat.ctor()));
915+
for pat in relevant_patterns {
916+
let specialized = pat.specialize(pcx, &ctor);
917+
for (subpat, sub_column) in specialized.iter().zip(&mut specialized_columns) {
918+
if subpat.is_or_pat() {
919+
sub_column.extend(subpat.iter_fields())
920+
} else {
921+
sub_column.push(subpat)
922+
}
923+
}
924+
}
925+
debug_assert!(
926+
!specialized_columns[0].is_empty(),
927+
"ctor {ctor:?} was listed as present but isn't"
928+
);
929+
930+
let wild_pat = WitnessPat::wild_from_ctor(pcx, ctor);
931+
for (i, col_i) in specialized_columns.iter().enumerate() {
932+
// Compute witnesses for each column.
933+
let wits_for_col_i = collect_nonexhaustive_missing_variants(cx, col_i.as_slice());
934+
// For each witness, we build a new pattern in the shape of `ctor(_, _, wit, _, _)`,
935+
// adding enough wildcards to match `arity`.
936+
for wit in wits_for_col_i {
937+
let mut pat = wild_pat.clone();
938+
pat.fields[i] = wit;
939+
witnesses.push(pat);
940+
}
941+
}
942+
}
943+
witnesses
944+
}
945+
924946
/// The arm of a match expression.
925947
#[derive(Clone, Copy, Debug)]
926948
pub(crate) struct MatchArm<'p, 'tcx> {
@@ -961,6 +983,7 @@ pub(crate) fn compute_match_usefulness<'p, 'tcx>(
961983
arms: &[MatchArm<'p, 'tcx>],
962984
lint_root: HirId,
963985
scrut_ty: Ty<'tcx>,
986+
scrut_span: Span,
964987
) -> UsefulnessReport<'p, 'tcx> {
965988
let mut matrix = Matrix::empty();
966989
let arm_usefulness: Vec<_> = arms
@@ -985,9 +1008,39 @@ pub(crate) fn compute_match_usefulness<'p, 'tcx>(
9851008
let wild_pattern = cx.pattern_arena.alloc(DeconstructedPat::wildcard(scrut_ty, DUMMY_SP));
9861009
let v = PatStack::from_pattern(wild_pattern);
9871010
let usefulness = is_useful(cx, &matrix, &v, FakeExtraWildcard, lint_root, false, true);
988-
let non_exhaustiveness_witnesses = match usefulness {
1011+
let non_exhaustiveness_witnesses: Vec<_> = match usefulness {
9891012
WithWitnesses(pats) => pats.into_iter().map(|w| w.single_pattern()).collect(),
9901013
NoWitnesses { .. } => bug!(),
9911014
};
1015+
1016+
// Run the non_exhaustive_omitted_patterns lint. Only run on refutable patterns to avoid hitting
1017+
// `if let`s. Only run if the match is exhaustive otherwise the error is redundant.
1018+
if cx.refutable
1019+
&& non_exhaustiveness_witnesses.is_empty()
1020+
&& !matches!(
1021+
cx.tcx.lint_level_at_node(NON_EXHAUSTIVE_OMITTED_PATTERNS, lint_root).0,
1022+
rustc_session::lint::Level::Allow
1023+
)
1024+
{
1025+
let pat_column = arms.iter().flat_map(|arm| arm.pat.flatten_or_pat()).collect::<Vec<_>>();
1026+
let witnesses = collect_nonexhaustive_missing_variants(cx, &pat_column);
1027+
1028+
if !witnesses.is_empty() {
1029+
// Report that a match of a `non_exhaustive` enum marked with `non_exhaustive_omitted_patterns`
1030+
// is not exhaustive enough.
1031+
//
1032+
// NB: The partner lint for structs lives in `compiler/rustc_hir_analysis/src/check/pat.rs`.
1033+
cx.tcx.emit_spanned_lint(
1034+
NON_EXHAUSTIVE_OMITTED_PATTERNS,
1035+
lint_root,
1036+
scrut_span,
1037+
NonExhaustiveOmittedPattern {
1038+
scrut_ty,
1039+
uncovered: Uncovered::new(scrut_span, cx, witnesses),
1040+
},
1041+
);
1042+
}
1043+
}
1044+
9921045
UsefulnessReport { arm_usefulness, non_exhaustiveness_witnesses }
9931046
}

0 commit comments

Comments
 (0)