handle non matching enum pattern types

This commit is contained in:
Josh Mcguigan 2020-04-05 10:16:34 -07:00
parent b87b7a088f
commit 43dfd89493
2 changed files with 57 additions and 41 deletions

View File

@ -250,17 +250,13 @@ pub enum Usefulness {
pub struct MatchCheckCtx<'a> {
pub body: Arc<Body>,
pub match_expr: &'a Expr,
pub infer: Arc<InferenceResult>,
pub db: &'a dyn HirDatabase,
}
// see src/librustc_mir_build/hair/pattern/_match.rs
// It seems the rustc version of this method is able to assume that all the match arm
// patterns are valid (they are valid given a particular match expression), but I
// don't think we can make that assumption here. How should that be handled?
//
// Perhaps check that validity before passing the patterns into this method?
/// Given a set of patterns `matrix`, and pattern to consider `v`, determines
/// whether `v` is useful. A pattern is useful if it covers cases which were
/// not previously covered.
pub(crate) fn is_useful(
cx: &MatchCheckCtx,
matrix: &Matrix,
@ -517,6 +513,19 @@ mod tests {
check_diagnostic_with_no_fix(content);
}
#[test]
fn empty_tuple_wild() {
let content = r"
fn test_fn() {
match () {
_ => {}
}
}
";
check_no_diagnostic(content);
}
#[test]
fn empty_tuple_no_diagnostic() {
let content = r"
@ -976,21 +985,6 @@ mod tests {
check_no_diagnostic(content);
}
}
#[cfg(test)]
mod false_negatives {
//! The implementation of match checking here is a work in progress. As we roll this out, we
//! prefer false negatives to false positives (ideally there would be no false positives). This
//! test module should document known false negatives. Eventually we will have a complete
//! implementation of match checking and this module will be empty.
//!
//! The reasons for documenting known false negatives:
//!
//! 1. It acts as a backlog of work that can be done to improve the behavior of the system.
//! 2. It ensures the code doesn't panic when handling these cases.
use super::tests::*;
#[test]
fn mismatched_types() {
@ -1011,10 +1005,8 @@ mod false_negatives {
}
";
// This is a false negative.
// We don't currently check that the match arms actually
// match the type of the match expression.
check_no_diagnostic(content);
// Match arms with the incorrect type are filtered out.
check_diagnostic_with_no_fix(content);
}
#[test]
@ -1028,14 +1020,24 @@ mod false_negatives {
}
";
// This is a false negative.
// We don't currently check that the match arms actually
// match the type of the match expression. This test
// checks to ensure we don't panic when the code we are
// checking is malformed in such a way that the arity of the
// constructors doesn't match.
check_no_diagnostic(content);
// Match arms with the incorrect type are filtered out.
check_diagnostic_with_no_fix(content);
}
}
#[cfg(test)]
mod false_negatives {
//! The implementation of match checking here is a work in progress. As we roll this out, we
//! prefer false negatives to false positives (ideally there would be no false positives). This
//! test module should document known false negatives. Eventually we will have a complete
//! implementation of match checking and this module will be empty.
//!
//! The reasons for documenting known false negatives:
//!
//! 1. It acts as a backlog of work that can be done to improve the behavior of the system.
//! 2. It ensures the code doesn't panic when handling these cases.
use super::tests::*;
#[test]
fn integers() {

View File

@ -67,7 +67,7 @@ impl<'a, 'b> ExprValidator<'a, 'b> {
fn validate_match(
&mut self,
id: ExprId,
expr: ExprId,
match_expr: ExprId,
arms: &[MatchArm],
db: &dyn HirDatabase,
infer: Arc<InferenceResult>,
@ -75,18 +75,32 @@ impl<'a, 'b> ExprValidator<'a, 'b> {
let (body, source_map): (Arc<Body>, Arc<BodySourceMap>) =
db.body_with_source_map(self.func.into());
let match_expr: &hir_def::expr::Expr = &body[expr];
let match_expr_ty = match infer.type_of_expr.get(match_expr) {
Some(ty) => ty,
// If we can't resolve the type of the match expression
// we cannot perform exhaustiveness checks.
None => return,
};
let cx = MatchCheckCtx { body: body.clone(), match_expr, infer, db };
let cx = MatchCheckCtx { body, infer: infer.clone(), db };
let pats = arms.iter().map(|arm| arm.pat);
let mut seen = Matrix::empty();
for pat in pats {
// If we had a NotUsefulMatchArm diagnostic, we could
// check the usefulness of each pattern as we added it
// to the matrix here.
let v = PatStack::from_pattern(pat);
seen.push(&cx, v);
// We skip any patterns whose type we cannot resolve.
if let Some(pat_ty) = infer.type_of_pat.get(pat) {
// We only include patterns whose type matches the type
// of the match expression. If we had a InvalidMatchArmPattern
// diagnostic or similar we could raise that in an else
// block here.
if pat_ty == match_expr_ty {
// If we had a NotUsefulMatchArm diagnostic, we could
// check the usefulness of each pattern as we added it
// to the matrix here.
let v = PatStack::from_pattern(pat);
seen.push(&cx, v);
}
}
}
match is_useful(&cx, &seen, &PatStack::from_wild()) {