Skip to content

Commit 894e87c

Browse files
authored
Fix type suggestion for manual_is_ascii_check (#13916)
Fixes #13913 . changelog: [`manual_is_ascii_check`]: fix type suggestions for references Previously it only derived `char` and `u8` types, now it should always annotate the lambda parameter with the correct type (e.g. `&char`). I'm quite new to Rust and this is my first contact with clippy, so I'm open for suggetions :)
2 parents f5ca68f + 8461d3f commit 894e87c

4 files changed

+43
-16
lines changed

clippy_lints/src/manual_is_ascii_check.rs

+10-15
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use rustc_ast::ast::RangeLimits;
99
use rustc_errors::Applicability;
1010
use rustc_hir::{Expr, ExprKind, Node, Param, PatKind, RangeEnd};
1111
use rustc_lint::{LateContext, LateLintPass};
12-
use rustc_middle::ty;
12+
use rustc_middle::ty::{self, Ty};
1313
use rustc_session::impl_lint_pass;
1414
use rustc_span::{Span, sym};
1515

@@ -114,7 +114,7 @@ impl<'tcx> LateLintPass<'tcx> for ManualIsAsciiCheck {
114114
&& !matches!(cx.typeck_results().expr_ty(arg).peel_refs().kind(), ty::Param(_))
115115
{
116116
let arg = peel_ref_operators(cx, arg);
117-
let ty_sugg = get_ty_sugg(cx, arg, start);
117+
let ty_sugg = get_ty_sugg(cx, arg);
118118
let range = check_range(start, end);
119119
check_is_ascii(cx, expr.span, arg, &range, ty_sugg);
120120
}
@@ -123,19 +123,14 @@ impl<'tcx> LateLintPass<'tcx> for ManualIsAsciiCheck {
123123
extract_msrv_attr!(LateContext);
124124
}
125125

126-
fn get_ty_sugg(cx: &LateContext<'_>, arg: &Expr<'_>, bound_expr: &Expr<'_>) -> Option<(Span, &'static str)> {
127-
if let ExprKind::Lit(lit) = bound_expr.kind
128-
&& let local_hid = path_to_local(arg)?
129-
&& let Node::Param(Param { ty_span, span, .. }) = cx.tcx.parent_hir_node(local_hid)
126+
fn get_ty_sugg<'tcx>(cx: &LateContext<'tcx>, arg: &Expr<'_>) -> Option<(Span, Ty<'tcx>)> {
127+
let local_hid = path_to_local(arg)?;
128+
if let Node::Param(Param { ty_span, span, .. }) = cx.tcx.parent_hir_node(local_hid)
130129
// `ty_span` and `span` are the same for inferred type, thus a type suggestion must be given
131130
&& ty_span == span
132131
{
133-
let ty_str = match lit.node {
134-
Char(_) => "char",
135-
Byte(_) => "u8",
136-
_ => return None,
137-
};
138-
return Some((*ty_span, ty_str));
132+
let arg_type = cx.typeck_results().expr_ty(arg);
133+
return Some((*ty_span, arg_type));
139134
}
140135
None
141136
}
@@ -145,7 +140,7 @@ fn check_is_ascii(
145140
span: Span,
146141
recv: &Expr<'_>,
147142
range: &CharRange,
148-
ty_sugg: Option<(Span, &'_ str)>,
143+
ty_sugg: Option<(Span, Ty<'_>)>,
149144
) {
150145
let sugg = match range {
151146
CharRange::UpperChar => "is_ascii_uppercase",
@@ -159,8 +154,8 @@ fn check_is_ascii(
159154
let mut app = Applicability::MachineApplicable;
160155
let recv = Sugg::hir_with_context(cx, recv, span.ctxt(), default_snip, &mut app).maybe_par();
161156
let mut suggestion = vec![(span, format!("{recv}.{sugg}()"))];
162-
if let Some((ty_span, ty_str)) = ty_sugg {
163-
suggestion.push((ty_span, format!("{recv}: {ty_str}")));
157+
if let Some((ty_span, ty)) = ty_sugg {
158+
suggestion.push((ty_span, format!("{recv}: {ty}")));
164159
}
165160

166161
span_lint_and_then(

tests/ui/manual_is_ascii_check.fixed

+5
Original file line numberDiff line numberDiff line change
@@ -82,3 +82,8 @@ fn generics() {
8282
take_while(|c: u8| c.is_ascii_uppercase());
8383
take_while(|c: char| c.is_ascii_uppercase());
8484
}
85+
86+
fn adds_type_reference() {
87+
let digits: Vec<&char> = ['1', 'A'].iter().take_while(|c: &&char| c.is_ascii_digit()).collect();
88+
let digits: Vec<&mut char> = ['1', 'A'].iter_mut().take_while(|c: &&mut char| c.is_ascii_digit()).collect();
89+
}

tests/ui/manual_is_ascii_check.rs

+5
Original file line numberDiff line numberDiff line change
@@ -82,3 +82,8 @@ fn generics() {
8282
take_while(|c| (b'A'..=b'Z').contains(&c));
8383
take_while(|c: char| ('A'..='Z').contains(&c));
8484
}
85+
86+
fn adds_type_reference() {
87+
let digits: Vec<&char> = ['1', 'A'].iter().take_while(|c| ('0'..='9').contains(c)).collect();
88+
let digits: Vec<&mut char> = ['1', 'A'].iter_mut().take_while(|c| ('0'..='9').contains(c)).collect();
89+
}

tests/ui/manual_is_ascii_check.stderr

+23-1
Original file line numberDiff line numberDiff line change
@@ -173,5 +173,27 @@ error: manual check for common ascii range
173173
LL | take_while(|c: char| ('A'..='Z').contains(&c));
174174
| ^^^^^^^^^^^^^^^^^^^^^^^^ help: try: `c.is_ascii_uppercase()`
175175

176-
error: aborting due to 27 previous errors
176+
error: manual check for common ascii range
177+
--> tests/ui/manual_is_ascii_check.rs:87:63
178+
|
179+
LL | let digits: Vec<&char> = ['1', 'A'].iter().take_while(|c| ('0'..='9').contains(c)).collect();
180+
| ^^^^^^^^^^^^^^^^^^^^^^^
181+
|
182+
help: try
183+
|
184+
LL | let digits: Vec<&char> = ['1', 'A'].iter().take_while(|c: &&char| c.is_ascii_digit()).collect();
185+
| ~~~~~~~~~ ~~~~~~~~~~~~~~~~~~
186+
187+
error: manual check for common ascii range
188+
--> tests/ui/manual_is_ascii_check.rs:88:71
189+
|
190+
LL | let digits: Vec<&mut char> = ['1', 'A'].iter_mut().take_while(|c| ('0'..='9').contains(c)).collect();
191+
| ^^^^^^^^^^^^^^^^^^^^^^^
192+
|
193+
help: try
194+
|
195+
LL | let digits: Vec<&mut char> = ['1', 'A'].iter_mut().take_while(|c: &&mut char| c.is_ascii_digit()).collect();
196+
| ~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~
197+
198+
error: aborting due to 29 previous errors
177199

0 commit comments

Comments
 (0)