rust/compiler/rustc_pattern_analysis/src/lints.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

285 lines
12 KiB
Rust
Raw Normal View History

2023-12-11 09:40:31 +00:00
use smallvec::SmallVec;
use rustc_data_structures::captures::Captures;
use rustc_middle::ty;
2023-12-11 09:40:31 +00:00
use rustc_session::lint;
use rustc_session::lint::builtin::NON_EXHAUSTIVE_OMITTED_PATTERNS;
2024-01-07 20:20:16 +00:00
use rustc_span::{ErrorGuaranteed, Span};
2023-12-11 09:40:31 +00:00
2023-12-11 19:01:02 +00:00
use crate::constructor::{IntRange, MaybeInfiniteInt};
2023-12-11 09:40:31 +00:00
use crate::errors::{
NonExhaustiveOmittedPattern, NonExhaustiveOmittedPatternLintOnArm, Overlap,
OverlappingRangeEndpoints, Uncovered,
};
use crate::pat::PatOrWild;
2023-12-11 19:15:52 +00:00
use crate::rustc::{
Constructor, DeconstructedPat, MatchArm, MatchCtxt, PlaceCtxt, RevealedTy, RustcMatchCheckCtxt,
2023-12-15 15:53:29 +00:00
SplitConstructorSet, WitnessPat,
2023-12-11 19:15:52 +00:00
};
2023-12-11 09:40:31 +00:00
/// A column of patterns in the matrix, where a column is the intuitive notion of "subpatterns that
/// inspect the same subvalue/place".
2023-12-11 11:53:01 +00:00
/// This is used to traverse patterns column-by-column for lints. Despite similarities with the
/// algorithm in [`crate::usefulness`], this does a different traversal. Notably this is linear in
/// the depth of patterns, whereas `compute_exhaustiveness_and_usefulness` is worst-case exponential
/// (exhaustiveness is NP-complete). The core difference is that we treat sub-columns separately.
2023-12-11 09:40:31 +00:00
///
/// This must not contain an or-pattern. `expand_and_push` takes care to expand them.
2023-12-11 09:40:31 +00:00
///
/// This is not used in the usefulness algorithm; only in lints.
2023-12-11 09:40:31 +00:00
#[derive(Debug)]
2023-12-26 01:59:18 +00:00
pub(crate) struct PatternColumn<'p, 'tcx> {
patterns: Vec<&'p DeconstructedPat<'p, 'tcx>>,
2023-12-11 09:40:31 +00:00
}
2023-12-26 01:59:18 +00:00
impl<'p, 'tcx> PatternColumn<'p, 'tcx> {
2023-12-11 09:40:31 +00:00
pub(crate) fn new(arms: &[MatchArm<'p, 'tcx>]) -> Self {
let patterns = Vec::with_capacity(arms.len());
let mut column = PatternColumn { patterns };
2023-12-11 09:40:31 +00:00
for arm in arms {
column.expand_and_push(PatOrWild::Pat(arm.pat));
}
column
}
/// Pushes a pattern onto the column, expanding any or-patterns into its subpatterns.
/// Internal method, prefer [`PatternColumn::new`].
fn expand_and_push(&mut self, pat: PatOrWild<'p, RustcMatchCheckCtxt<'p, 'tcx>>) {
// We flatten or-patterns and skip algorithm-generated wildcards.
if pat.is_or_pat() {
self.patterns.extend(
pat.flatten_or_pat().into_iter().filter_map(|pat_or_wild| pat_or_wild.as_pat()),
)
} else if let Some(pat) = pat.as_pat() {
self.patterns.push(pat)
2023-12-11 09:40:31 +00:00
}
}
fn head_ty(&self) -> Option<RevealedTy<'tcx>> {
self.patterns.first().map(|pat| pat.ty())
2023-12-11 09:40:31 +00:00
}
/// Do constructor splitting on the constructors of the column.
2024-01-01 22:54:20 +00:00
fn analyze_ctors(
&self,
pcx: &PlaceCtxt<'_, 'p, 'tcx>,
) -> Result<SplitConstructorSet<'p, 'tcx>, ErrorGuaranteed> {
2023-12-11 09:40:31 +00:00
let column_ctors = self.patterns.iter().map(|p| p.ctor());
2024-01-01 22:54:20 +00:00
let ctors_for_ty = &pcx.ctors_for_ty()?;
Ok(ctors_for_ty.split(pcx, column_ctors))
2023-12-11 09:40:31 +00:00
}
2023-12-26 02:12:33 +00:00
fn iter(&self) -> impl Iterator<Item = &'p DeconstructedPat<'p, 'tcx>> + Captures<'_> {
2023-12-11 09:40:31 +00:00
self.patterns.iter().copied()
}
/// Does specialization: given a constructor, this takes the patterns from the column that match
/// the constructor, and outputs their fields.
/// This returns one column per field of the constructor. They usually all have the same length
/// (the number of patterns in `self` that matched `ctor`), except that we expand or-patterns
/// which may change the lengths.
2023-12-14 16:58:33 +00:00
fn specialize(
&self,
2023-12-26 01:59:18 +00:00
pcx: &PlaceCtxt<'_, 'p, 'tcx>,
2023-12-11 19:01:02 +00:00
ctor: &Constructor<'p, 'tcx>,
2023-12-26 01:59:18 +00:00
) -> Vec<PatternColumn<'p, 'tcx>> {
2023-12-11 09:40:31 +00:00
let arity = ctor.arity(pcx);
if arity == 0 {
return Vec::new();
}
// We specialize the column by `ctor`. This gives us `arity`-many columns of patterns. These
// columns may have different lengths in the presence of or-patterns (this is why we can't
// reuse `Matrix`).
let mut specialized_columns: Vec<_> =
(0..arity).map(|_| Self { patterns: Vec::new() }).collect();
let relevant_patterns =
self.patterns.iter().filter(|pat| ctor.is_covered_by(pcx, pat.ctor()));
for pat in relevant_patterns {
let specialized = pat.specialize(ctor, arity);
for (subpat, column) in specialized.into_iter().zip(&mut specialized_columns) {
column.expand_and_push(subpat);
2023-12-11 09:40:31 +00:00
}
}
specialized_columns
}
}
/// Traverse the patterns to collect any variants of a non_exhaustive enum that fail to be mentioned
/// in a given column.
2023-12-15 15:53:29 +00:00
#[instrument(level = "debug", skip(cx), ret)]
fn collect_nonexhaustive_missing_variants<'a, 'p, 'tcx>(
2023-12-15 15:53:29 +00:00
cx: MatchCtxt<'a, 'p, 'tcx>,
2023-12-26 01:59:18 +00:00
column: &PatternColumn<'p, 'tcx>,
2024-01-07 20:20:16 +00:00
) -> Result<Vec<WitnessPat<'p, 'tcx>>, ErrorGuaranteed> {
let Some(ty) = column.head_ty() else {
2024-01-07 20:20:16 +00:00
return Ok(Vec::new());
2023-12-11 09:40:31 +00:00
};
2023-12-15 15:53:29 +00:00
let pcx = &PlaceCtxt::new_dummy(cx, ty);
2023-12-11 09:40:31 +00:00
2024-01-01 22:54:20 +00:00
let set = column.analyze_ctors(pcx)?;
2023-12-11 09:40:31 +00:00
if set.present.is_empty() {
// We can't consistently handle the case where no constructors are present (since this would
// require digging deep through any type in case there's a non_exhaustive enum somewhere),
// so for consistency we refuse to handle the top-level case, where we could handle it.
2024-01-07 20:20:16 +00:00
return Ok(Vec::new());
2023-12-11 09:40:31 +00:00
}
let mut witnesses = Vec::new();
2023-12-15 15:53:29 +00:00
if cx.tycx.is_foreign_non_exhaustive_enum(ty) {
2023-12-11 09:40:31 +00:00
witnesses.extend(
set.missing
.into_iter()
// This will list missing visible variants.
.filter(|c| !matches!(c, Constructor::Hidden | Constructor::NonExhaustive))
.map(|missing_ctor| WitnessPat::wild_from_ctor(pcx, missing_ctor)),
)
}
// Recurse into the fields.
for ctor in set.present {
let specialized_columns = column.specialize(pcx, &ctor);
let wild_pat = WitnessPat::wild_from_ctor(pcx, ctor);
for (i, col_i) in specialized_columns.iter().enumerate() {
// Compute witnesses for each column.
2024-01-07 20:20:16 +00:00
let wits_for_col_i = collect_nonexhaustive_missing_variants(cx, col_i)?;
2023-12-11 09:40:31 +00:00
// For each witness, we build a new pattern in the shape of `ctor(_, _, wit, _, _)`,
// adding enough wildcards to match `arity`.
for wit in wits_for_col_i {
let mut pat = wild_pat.clone();
pat.fields[i] = wit;
witnesses.push(pat);
}
}
}
2024-01-07 20:20:16 +00:00
Ok(witnesses)
2023-12-11 09:40:31 +00:00
}
pub(crate) fn lint_nonexhaustive_missing_variants<'a, 'p, 'tcx>(
2023-12-15 15:53:29 +00:00
cx: MatchCtxt<'a, 'p, 'tcx>,
2023-12-11 09:40:31 +00:00
arms: &[MatchArm<'p, 'tcx>],
2023-12-26 01:59:18 +00:00
pat_column: &PatternColumn<'p, 'tcx>,
scrut_ty: RevealedTy<'tcx>,
2024-01-07 20:20:16 +00:00
) -> Result<(), ErrorGuaranteed> {
2023-12-15 15:53:29 +00:00
let rcx: &RustcMatchCheckCtxt<'_, '_> = cx.tycx;
2023-12-11 09:40:31 +00:00
if !matches!(
2023-12-15 15:53:29 +00:00
rcx.tcx.lint_level_at_node(NON_EXHAUSTIVE_OMITTED_PATTERNS, rcx.match_lint_level).0,
2023-12-11 09:40:31 +00:00
rustc_session::lint::Level::Allow
) {
2024-01-07 20:20:16 +00:00
let witnesses = collect_nonexhaustive_missing_variants(cx, pat_column)?;
2023-12-11 09:40:31 +00:00
if !witnesses.is_empty() {
// Report that a match of a `non_exhaustive` enum marked with `non_exhaustive_omitted_patterns`
// is not exhaustive enough.
//
// NB: The partner lint for structs lives in `compiler/rustc_hir_analysis/src/check/pat.rs`.
2023-12-15 15:53:29 +00:00
rcx.tcx.emit_spanned_lint(
2023-12-11 09:40:31 +00:00
NON_EXHAUSTIVE_OMITTED_PATTERNS,
2023-12-15 15:53:29 +00:00
rcx.match_lint_level,
rcx.scrut_span,
2023-12-11 09:40:31 +00:00
NonExhaustiveOmittedPattern {
scrut_ty: scrut_ty.inner(),
2023-12-15 15:53:29 +00:00
uncovered: Uncovered::new(rcx.scrut_span, rcx, witnesses),
2023-12-11 09:40:31 +00:00
},
);
}
} else {
// We used to allow putting the `#[allow(non_exhaustive_omitted_patterns)]` on a match
// arm. This no longer makes sense so we warn users, to avoid silently breaking their
// usage of the lint.
for arm in arms {
let (lint_level, lint_level_source) =
2023-12-15 15:53:29 +00:00
rcx.tcx.lint_level_at_node(NON_EXHAUSTIVE_OMITTED_PATTERNS, arm.arm_data);
2023-12-11 09:40:31 +00:00
if !matches!(lint_level, rustc_session::lint::Level::Allow) {
let decorator = NonExhaustiveOmittedPatternLintOnArm {
lint_span: lint_level_source.span(),
2023-12-15 15:53:29 +00:00
suggest_lint_on_match: rcx.whole_match_span.map(|span| span.shrink_to_lo()),
2023-12-11 09:40:31 +00:00
lint_level: lint_level.as_str(),
lint_name: "non_exhaustive_omitted_patterns",
};
use rustc_errors::DecorateLint;
let mut err = rcx.tcx.dcx().struct_span_warn(arm.pat.data().unwrap().span, "");
err.primary_message(decorator.msg());
2023-12-11 09:40:31 +00:00
decorator.decorate_lint(&mut err);
err.emit();
}
}
}
2024-01-07 20:20:16 +00:00
Ok(())
2023-12-11 09:40:31 +00:00
}
/// Traverse the patterns to warn the user about ranges that overlap on their endpoints.
2023-12-15 15:53:29 +00:00
#[instrument(level = "debug", skip(cx))]
pub(crate) fn lint_overlapping_range_endpoints<'a, 'p, 'tcx>(
2023-12-15 15:53:29 +00:00
cx: MatchCtxt<'a, 'p, 'tcx>,
2023-12-26 01:59:18 +00:00
column: &PatternColumn<'p, 'tcx>,
2024-01-07 20:20:16 +00:00
) -> Result<(), ErrorGuaranteed> {
let Some(ty) = column.head_ty() else {
2024-01-07 20:20:16 +00:00
return Ok(());
2023-12-11 09:40:31 +00:00
};
2023-12-15 15:53:29 +00:00
let pcx = &PlaceCtxt::new_dummy(cx, ty);
let rcx: &RustcMatchCheckCtxt<'_, '_> = cx.tycx;
2023-12-11 09:40:31 +00:00
2024-01-01 22:54:20 +00:00
let set = column.analyze_ctors(pcx)?;
2023-12-11 09:40:31 +00:00
if matches!(ty.kind(), ty::Char | ty::Int(_) | ty::Uint(_)) {
let emit_lint = |overlap: &IntRange, this_span: Span, overlapped_spans: &[Span]| {
2023-12-15 15:53:29 +00:00
let overlap_as_pat = rcx.hoist_pat_range(overlap, ty);
2023-12-11 09:40:31 +00:00
let overlaps: Vec<_> = overlapped_spans
.iter()
.copied()
.map(|span| Overlap { range: overlap_as_pat.clone(), span })
.collect();
2023-12-15 15:53:29 +00:00
rcx.tcx.emit_spanned_lint(
2023-12-11 09:40:31 +00:00
lint::builtin::OVERLAPPING_RANGE_ENDPOINTS,
2023-12-15 15:53:29 +00:00
rcx.match_lint_level,
2023-12-11 09:40:31 +00:00
this_span,
OverlappingRangeEndpoints { overlap: overlaps, range: this_span },
);
};
// If two ranges overlapped, the split set will contain their intersection as a singleton.
let split_int_ranges = set.present.iter().filter_map(|c| c.as_int_range());
for overlap_range in split_int_ranges.clone() {
if overlap_range.is_singleton() {
let overlap: MaybeInfiniteInt = overlap_range.lo;
// Ranges that look like `lo..=overlap`.
let mut prefixes: SmallVec<[_; 1]> = Default::default();
// Ranges that look like `overlap..=hi`.
let mut suffixes: SmallVec<[_; 1]> = Default::default();
// Iterate on patterns that contained `overlap`.
for pat in column.iter() {
let Constructor::IntRange(this_range) = pat.ctor() else { continue };
let this_span = pat.data().unwrap().span;
2023-12-11 09:40:31 +00:00
if this_range.is_singleton() {
// Don't lint when one of the ranges is a singleton.
continue;
}
if this_range.lo == overlap {
// `this_range` looks like `overlap..=this_range.hi`; it overlaps with any
// ranges that look like `lo..=overlap`.
if !prefixes.is_empty() {
emit_lint(overlap_range, this_span, &prefixes);
}
suffixes.push(this_span)
} else if this_range.hi == overlap.plus_one() {
// `this_range` looks like `this_range.lo..=overlap`; it overlaps with any
// ranges that look like `overlap..=hi`.
if !suffixes.is_empty() {
emit_lint(overlap_range, this_span, &suffixes);
}
prefixes.push(this_span)
}
}
}
}
} else {
// Recurse into the fields.
for ctor in set.present {
for col in column.specialize(pcx, &ctor) {
2024-01-07 20:20:16 +00:00
lint_overlapping_range_endpoints(cx, &col)?;
2023-12-11 09:40:31 +00:00
}
}
}
2024-01-07 20:20:16 +00:00
Ok(())
2023-12-11 09:40:31 +00:00
}