Skip to content

Commit 2487765

Browse files
committed
Detect const in pattern with typo
When writing a constant name incorrectly in a pattern, the pattern will be identified as a new binding. We look for consts in the current crate, consts that where imported in the current crate and for local `let` bindings in case someone got them confused with `const`s. ``` error: unreachable pattern --> $DIR/const-with-typo-in-pattern-binding.rs:30:9 | LL | GOOOD => {} | ----- matches any value LL | LL | _ => {} | ^ no value can reach this | help: you might have meant to pattern match against the value of similarly named constant `GOOD` instead of introducing a new catch-all binding | LL | GOOD => {} | ~~~~ ``` Fix #132582.
1 parent bfe809d commit 2487765

File tree

5 files changed

+315
-1
lines changed

5 files changed

+315
-1
lines changed

compiler/rustc_mir_build/messages.ftl

+7
Original file line numberDiff line numberDiff line change
@@ -338,6 +338,13 @@ mir_build_unreachable_pattern = unreachable pattern
338338
.unreachable_covered_by_catchall = matches any value
339339
.unreachable_covered_by_one = matches all the relevant values
340340
.unreachable_covered_by_many = multiple earlier patterns match some of the same values
341+
.unreachable_pattern_const_reexport_accessible = there is a constant of the same name imported in another scope, which could have been used to pattern match against its value instead of introducing a new catch-all binding, but it needs to be imported in the pattern's scope
342+
.unreachable_pattern_wanted_const = you might have meant to pattern match against the value of {$is_typo ->
343+
[true] similarly named constant
344+
*[false] constant
345+
} `{$const_name}` instead of introducing a new catch-all binding
346+
.unreachable_pattern_const_inaccessible = there is a constant of the same name, which could have been used to pattern match against its value instead of introducing a new catch-all binding, but it is not accessible from this scope
347+
.unreachable_pattern_let_binding = there is a binding of the same name; if you meant to pattern match against the value of that binding, that is a feature of constants that is not available for `let` bindings
341348
.suggestion = remove the match arm
342349
343350
mir_build_unsafe_fn_safe_body = an unsafe function restricts its caller, but its body is safe by default

compiler/rustc_mir_build/src/errors.rs

+22
Original file line numberDiff line numberDiff line change
@@ -593,6 +593,14 @@ pub(crate) struct UnreachablePattern<'tcx> {
593593
pub(crate) uninhabited_note: Option<()>,
594594
#[label(mir_build_unreachable_covered_by_catchall)]
595595
pub(crate) covered_by_catchall: Option<Span>,
596+
#[subdiagnostic]
597+
pub(crate) wanted_constant: Option<WantedConstant>,
598+
#[note(mir_build_unreachable_pattern_const_reexport_accessible)]
599+
pub(crate) accessible_constant: Option<Span>,
600+
#[note(mir_build_unreachable_pattern_const_inaccessible)]
601+
pub(crate) inaccessible_constant: Option<Span>,
602+
#[note(mir_build_unreachable_pattern_let_binding)]
603+
pub(crate) pattern_let_binding: Option<Span>,
596604
#[label(mir_build_unreachable_covered_by_one)]
597605
pub(crate) covered_by_one: Option<Span>,
598606
#[note(mir_build_unreachable_covered_by_many)]
@@ -602,6 +610,20 @@ pub(crate) struct UnreachablePattern<'tcx> {
602610
pub(crate) suggest_remove: Option<Span>,
603611
}
604612

613+
#[derive(Subdiagnostic)]
614+
#[suggestion(
615+
mir_build_unreachable_pattern_wanted_const,
616+
code = "{const_path}",
617+
applicability = "machine-applicable"
618+
)]
619+
pub(crate) struct WantedConstant {
620+
#[primary_span]
621+
pub(crate) span: Span,
622+
pub(crate) is_typo: bool,
623+
pub(crate) const_name: String,
624+
pub(crate) const_path: String,
625+
}
626+
605627
#[derive(Diagnostic)]
606628
#[diag(mir_build_const_pattern_depends_on_generic_parameter, code = E0158)]
607629
pub(crate) struct ConstPatternDependsOnGenericParameter {

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

+163-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,9 @@ use rustc_errors::{Applicability, ErrorGuaranteed, MultiSpan, struct_span_code_e
77
use rustc_hir::def::*;
88
use rustc_hir::def_id::LocalDefId;
99
use rustc_hir::{self as hir, BindingMode, ByRef, HirId};
10+
use rustc_infer::infer::TyCtxtInferExt;
1011
use rustc_infer::traits::Reveal;
12+
use rustc_lint::Level;
1113
use rustc_middle::bug;
1214
use rustc_middle::middle::limits::get_limit_size;
1315
use rustc_middle::thir::visit::Visitor;
@@ -22,8 +24,10 @@ use rustc_pattern_analysis::rustc::{
2224
use rustc_session::lint::builtin::{
2325
BINDINGS_WITH_VARIANT_NAME, IRREFUTABLE_LET_PATTERNS, UNREACHABLE_PATTERNS,
2426
};
27+
use rustc_span::edit_distance::find_best_match_for_name;
2528
use rustc_span::hygiene::DesugaringKind;
2629
use rustc_span::{Span, sym};
30+
use rustc_trait_selection::infer::InferCtxtExt;
2731
use tracing::instrument;
2832

2933
use crate::errors::*;
@@ -954,6 +958,10 @@ fn report_unreachable_pattern<'p, 'tcx>(
954958
covered_by_one: None,
955959
covered_by_many: None,
956960
covered_by_many_n_more_count: 0,
961+
wanted_constant: None,
962+
accessible_constant: None,
963+
inaccessible_constant: None,
964+
pattern_let_binding: None,
957965
suggest_remove: None,
958966
};
959967
match explanation.covered_by.as_slice() {
@@ -976,7 +984,10 @@ fn report_unreachable_pattern<'p, 'tcx>(
976984
});
977985
}
978986
[covering_pat] if pat_is_catchall(covering_pat) => {
979-
lint.covered_by_catchall = Some(covering_pat.data().span);
987+
// A binding pattern that matches all, a single binding name.
988+
let pat = covering_pat.data();
989+
lint.covered_by_catchall = Some(pat.span);
990+
find_fallback_pattern_typo(cx, hir_id, pat, &mut lint);
980991
}
981992
[covering_pat] => {
982993
lint.covered_by_one = Some(covering_pat.data().span);
@@ -1009,6 +1020,157 @@ fn report_unreachable_pattern<'p, 'tcx>(
10091020
cx.tcx.emit_node_span_lint(UNREACHABLE_PATTERNS, hir_id, pat_span, lint);
10101021
}
10111022

1023+
/// Detect typos that were meant to be a `const` but were interpreted as a new pattern binding.
1024+
fn find_fallback_pattern_typo<'tcx>(
1025+
cx: &PatCtxt<'_, 'tcx>,
1026+
hir_id: HirId,
1027+
pat: &Pat<'tcx>,
1028+
lint: &mut UnreachablePattern<'_>,
1029+
) {
1030+
if let (Level::Allow, _) = cx.tcx.lint_level_at_node(UNREACHABLE_PATTERNS, hir_id) {
1031+
// This is because we use `with_no_trimmed_paths` later, so if we never emit the lint we'd
1032+
// ICE. At the same time, we don't really need to do all of this if we won't emit anything.
1033+
return;
1034+
}
1035+
if let PatKind::Binding { name, subpattern: None, ty, .. } = pat.kind {
1036+
// See if the binding might have been a `const` that was mistyped or out of scope.
1037+
let mut accessible = vec![];
1038+
let mut accessible_path = vec![];
1039+
let mut inaccessible = vec![];
1040+
let mut imported = vec![];
1041+
let mut imported_spans = vec![];
1042+
let infcx = cx.tcx.infer_ctxt().build(ty::TypingMode::non_body_analysis());
1043+
let parent = cx.tcx.hir().get_parent_item(hir_id);
1044+
1045+
for item in cx.tcx.hir_crate_items(()).free_items() {
1046+
if let DefKind::Use = cx.tcx.def_kind(item.owner_id) {
1047+
// Look for consts being re-exported.
1048+
let item = cx.tcx.hir().expect_item(item.owner_id.def_id);
1049+
let use_name = item.ident.name;
1050+
let hir::ItemKind::Use(path, _) = item.kind else {
1051+
continue;
1052+
};
1053+
for res in &path.res {
1054+
if let Res::Def(DefKind::Const, id) = res
1055+
&& infcx.can_eq(cx.param_env, ty, cx.tcx.type_of(id).instantiate_identity())
1056+
{
1057+
if cx.tcx.visibility(id).is_accessible_from(parent, cx.tcx) {
1058+
// The original const is accessible, suggest using it directly.
1059+
let item_name = cx.tcx.item_name(*id);
1060+
accessible.push(item_name);
1061+
accessible_path.push(with_no_trimmed_paths!(cx.tcx.def_path_str(id)));
1062+
} else if cx
1063+
.tcx
1064+
.visibility(item.owner_id)
1065+
.is_accessible_from(parent, cx.tcx)
1066+
{
1067+
// The const is accessible only through the re-export, point at
1068+
// the `use`.
1069+
imported.push(use_name);
1070+
imported_spans.push(item.ident.span);
1071+
}
1072+
}
1073+
}
1074+
}
1075+
if let DefKind::Const = cx.tcx.def_kind(item.owner_id)
1076+
&& infcx.can_eq(
1077+
cx.param_env,
1078+
ty,
1079+
cx.tcx.type_of(item.owner_id).instantiate_identity(),
1080+
)
1081+
{
1082+
// Look for local consts.
1083+
let item_name = cx.tcx.item_name(item.owner_id.into());
1084+
let vis = cx.tcx.visibility(item.owner_id);
1085+
if vis.is_accessible_from(parent, cx.tcx) {
1086+
accessible.push(item_name);
1087+
let path = if item_name == name {
1088+
// We know that the const wasn't in scope because it has the exact
1089+
// same name, so we suggest the full path.
1090+
with_no_trimmed_paths!(cx.tcx.def_path_str(item.owner_id))
1091+
} else {
1092+
// The const is likely just typoed, and nothing else.
1093+
cx.tcx.def_path_str(item.owner_id)
1094+
};
1095+
accessible_path.push(path);
1096+
} else if name == item_name {
1097+
// The const exists somewhere in this crate, but it can't be imported
1098+
// from this pattern's scope. We'll just point at its definition.
1099+
inaccessible.push(cx.tcx.def_span(item.owner_id));
1100+
}
1101+
}
1102+
}
1103+
if let Some((i, &const_name)) =
1104+
accessible.iter().enumerate().find(|(_, &const_name)| const_name == name)
1105+
{
1106+
// The pattern name is an exact match, so the pattern needed to be imported.
1107+
lint.wanted_constant = Some(WantedConstant {
1108+
span: pat.span,
1109+
is_typo: false,
1110+
const_name: const_name.to_string(),
1111+
const_path: accessible_path[i].clone(),
1112+
});
1113+
} else if let Some(name) = find_best_match_for_name(&accessible, name, None) {
1114+
// The pattern name is likely a typo.
1115+
lint.wanted_constant = Some(WantedConstant {
1116+
span: pat.span,
1117+
is_typo: true,
1118+
const_name: name.to_string(),
1119+
const_path: name.to_string(),
1120+
});
1121+
} else if let Some(i) =
1122+
imported.iter().enumerate().find(|(_, &const_name)| const_name == name).map(|(i, _)| i)
1123+
{
1124+
// The const with the exact name wasn't re-exported from an import in this
1125+
// crate, we point at the import.
1126+
lint.accessible_constant = Some(imported_spans[i]);
1127+
} else if let Some(name) = find_best_match_for_name(&imported, name, None) {
1128+
// The typoed const wasn't re-exported by an import in this crate, we suggest
1129+
// the right name (which will likely require another follow up suggestion).
1130+
lint.wanted_constant = Some(WantedConstant {
1131+
span: pat.span,
1132+
is_typo: true,
1133+
const_path: name.to_string(),
1134+
const_name: name.to_string(),
1135+
});
1136+
} else if !inaccessible.is_empty() {
1137+
for span in inaccessible {
1138+
// The const with the exact name match isn't accessible, we just point at it.
1139+
lint.inaccessible_constant = Some(span);
1140+
}
1141+
} else {
1142+
// Look for local bindings for people that might have gotten confused with how
1143+
// `let` and `const` works.
1144+
for (_, node) in cx.tcx.hir().parent_iter(hir_id) {
1145+
match node {
1146+
hir::Node::Stmt(hir::Stmt { kind: hir::StmtKind::Let(let_stmt), .. }) => {
1147+
if let hir::PatKind::Binding(_, _, binding_name, _) = let_stmt.pat.kind {
1148+
if name == binding_name.name {
1149+
lint.pattern_let_binding = Some(binding_name.span);
1150+
}
1151+
}
1152+
}
1153+
hir::Node::Block(hir::Block { stmts, .. }) => {
1154+
for stmt in *stmts {
1155+
if let hir::StmtKind::Let(let_stmt) = stmt.kind {
1156+
if let hir::PatKind::Binding(_, _, binding_name, _) =
1157+
let_stmt.pat.kind
1158+
{
1159+
if name == binding_name.name {
1160+
lint.pattern_let_binding = Some(binding_name.span);
1161+
}
1162+
}
1163+
}
1164+
}
1165+
}
1166+
hir::Node::Item(_) => break,
1167+
_ => {}
1168+
}
1169+
}
1170+
}
1171+
}
1172+
}
1173+
10121174
/// Report unreachable arms, if any.
10131175
fn report_arm_reachability<'p, 'tcx>(
10141176
cx: &PatCtxt<'p, 'tcx>,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
#![deny(unreachable_patterns)] //~ NOTE the lint level is defined here
2+
#![allow(non_snake_case, non_upper_case_globals)]
3+
mod x {
4+
pub use std::env::consts::ARCH;
5+
const X: i32 = 0; //~ NOTE there is a constant of the same name
6+
}
7+
fn main() {
8+
let input: i32 = 42;
9+
10+
const god: i32 = 1;
11+
const GOOD: i32 = 1;
12+
const BAD: i32 = 2;
13+
14+
let name: i32 = 42; //~ NOTE there is a binding of the same name
15+
16+
match input {
17+
X => {} //~ NOTE matches any value
18+
_ => {} //~ ERROR unreachable pattern
19+
//~^ NOTE no value can reach this
20+
}
21+
match input {
22+
GOD => {} //~ HELP you might have meant to pattern match against the value of similarly named constant `god`
23+
//~^ NOTE matches any value
24+
_ => {} //~ ERROR unreachable pattern
25+
//~^ NOTE no value can reach this
26+
}
27+
match input {
28+
GOOOD => {} //~ HELP you might have meant to pattern match against the value of similarly named constant `GOOD`
29+
//~^ NOTE matches any value
30+
_ => {} //~ ERROR unreachable pattern
31+
//~^ NOTE no value can reach this
32+
}
33+
match input {
34+
name => {}
35+
//~^ NOTE matches any value
36+
_ => {} //~ ERROR unreachable pattern
37+
//~^ NOTE no value can reach this
38+
}
39+
match "" {
40+
ARCH => {} //~ HELP you might have meant to pattern match against the value of constant `ARCH`
41+
//~^ NOTE matches any value
42+
_ => {} //~ ERROR unreachable pattern
43+
//~^ NOTE no value can reach this
44+
}
45+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
error: unreachable pattern
2+
--> $DIR/const-with-typo-in-pattern-binding.rs:18:9
3+
|
4+
LL | X => {}
5+
| - matches any value
6+
LL | _ => {}
7+
| ^ no value can reach this
8+
|
9+
note: there is a constant of the same name, which could have been used to pattern match against its value instead of introducing a new catch-all binding, but it is not accessible from this scope
10+
--> $DIR/const-with-typo-in-pattern-binding.rs:5:5
11+
|
12+
LL | const X: i32 = 0;
13+
| ^^^^^^^^^^^^
14+
note: the lint level is defined here
15+
--> $DIR/const-with-typo-in-pattern-binding.rs:1:9
16+
|
17+
LL | #![deny(unreachable_patterns)]
18+
| ^^^^^^^^^^^^^^^^^^^^
19+
20+
error: unreachable pattern
21+
--> $DIR/const-with-typo-in-pattern-binding.rs:24:9
22+
|
23+
LL | GOD => {}
24+
| --- matches any value
25+
LL |
26+
LL | _ => {}
27+
| ^ no value can reach this
28+
|
29+
help: you might have meant to pattern match against the value of similarly named constant `god` instead of introducing a new catch-all binding
30+
|
31+
LL | god => {}
32+
| ~~~
33+
34+
error: unreachable pattern
35+
--> $DIR/const-with-typo-in-pattern-binding.rs:30:9
36+
|
37+
LL | GOOOD => {}
38+
| ----- matches any value
39+
LL |
40+
LL | _ => {}
41+
| ^ no value can reach this
42+
|
43+
help: you might have meant to pattern match against the value of similarly named constant `GOOD` instead of introducing a new catch-all binding
44+
|
45+
LL | GOOD => {}
46+
| ~~~~
47+
48+
error: unreachable pattern
49+
--> $DIR/const-with-typo-in-pattern-binding.rs:36:9
50+
|
51+
LL | name => {}
52+
| ---- matches any value
53+
LL |
54+
LL | _ => {}
55+
| ^ no value can reach this
56+
|
57+
note: there is a binding of the same name; if you meant to pattern match against the value of that binding, that is a feature of constants that is not available for `let` bindings
58+
--> $DIR/const-with-typo-in-pattern-binding.rs:14:9
59+
|
60+
LL | let name: i32 = 42;
61+
| ^^^^
62+
63+
error: unreachable pattern
64+
--> $DIR/const-with-typo-in-pattern-binding.rs:42:9
65+
|
66+
LL | ARCH => {}
67+
| ---- matches any value
68+
LL |
69+
LL | _ => {}
70+
| ^ no value can reach this
71+
|
72+
help: you might have meant to pattern match against the value of constant `ARCH` instead of introducing a new catch-all binding
73+
|
74+
LL | std::env::consts::ARCH => {}
75+
| ~~~~~~~~~~~~~~~~~~~~~~
76+
77+
error: aborting due to 5 previous errors
78+

0 commit comments

Comments
 (0)