Skip to content

Commit a2c1d56

Browse files
committed
Auto merge of rust-lang#12259 - GuillaumeGomez:multiple-bound-locations, r=llogiq
Add new `multiple_bound_locations` lint Fixes rust-lang#7181. r? `@llogiq` changelog: Add new `multiple_bound_locations` lint
2 parents 6405469 + 762448b commit a2c1d56

12 files changed

+259
-49
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -5430,6 +5430,7 @@ Released 2018-09-13
54305430
[`modulo_arithmetic`]: https://rust-lang.github.io/rust-clippy/master/index.html#modulo_arithmetic
54315431
[`modulo_one`]: https://rust-lang.github.io/rust-clippy/master/index.html#modulo_one
54325432
[`multi_assignments`]: https://rust-lang.github.io/rust-clippy/master/index.html#multi_assignments
5433+
[`multiple_bound_locations`]: https://rust-lang.github.io/rust-clippy/master/index.html#multiple_bound_locations
54335434
[`multiple_crate_versions`]: https://rust-lang.github.io/rust-clippy/master/index.html#multiple_crate_versions
54345435
[`multiple_inherent_impl`]: https://rust-lang.github.io/rust-clippy/master/index.html#multiple_inherent_impl
54355436
[`multiple_unsafe_ops_per_block`]: https://rust-lang.github.io/rust-clippy/master/index.html#multiple_unsafe_ops_per_block

clippy_lints/src/declared_lints.rs

+1
Original file line numberDiff line numberDiff line change
@@ -499,6 +499,7 @@ pub(crate) static LINTS: &[&crate::LintInfo] = &[
499499
crate::module_style::MOD_MODULE_FILES_INFO,
500500
crate::module_style::SELF_NAMED_MODULE_FILES_INFO,
501501
crate::multi_assignments::MULTI_ASSIGNMENTS_INFO,
502+
crate::multiple_bound_locations::MULTIPLE_BOUND_LOCATIONS_INFO,
502503
crate::multiple_unsafe_ops_per_block::MULTIPLE_UNSAFE_OPS_PER_BLOCK_INFO,
503504
crate::mut_key::MUTABLE_KEY_TYPE_INFO,
504505
crate::mut_mut::MUT_MUT_INFO,

clippy_lints/src/lib.rs

+2
Original file line numberDiff line numberDiff line change
@@ -231,6 +231,7 @@ mod missing_trait_methods;
231231
mod mixed_read_write_in_expression;
232232
mod module_style;
233233
mod multi_assignments;
234+
mod multiple_bound_locations;
234235
mod multiple_unsafe_ops_per_block;
235236
mod mut_key;
236237
mod mut_mut;
@@ -1116,6 +1117,7 @@ pub fn register_lints(store: &mut rustc_lint::LintStore, conf: &'static Conf) {
11161117
});
11171118
store.register_late_pass(move |_| Box::new(incompatible_msrv::IncompatibleMsrv::new(msrv())));
11181119
store.register_late_pass(|_| Box::new(to_string_trait_impl::ToStringTraitImpl));
1120+
store.register_early_pass(|| Box::new(multiple_bound_locations::MultipleBoundLocations));
11191121
// add lints here, do not remove this comment, it's used in `new_lint`
11201122
}
11211123

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
use rustc_ast::visit::FnKind;
2+
use rustc_ast::{NodeId, WherePredicate};
3+
use rustc_data_structures::fx::FxHashMap;
4+
use rustc_lint::{EarlyContext, EarlyLintPass};
5+
use rustc_session::declare_lint_pass;
6+
use rustc_span::Span;
7+
8+
use clippy_utils::diagnostics::span_lint;
9+
use clippy_utils::source::snippet_opt;
10+
11+
declare_clippy_lint! {
12+
/// ### What it does
13+
/// Check if a generic is defined both in the bound predicate and in the `where` clause.
14+
///
15+
/// ### Why is this bad?
16+
/// It can be confusing for developers when seeing bounds for a generic in multiple places.
17+
///
18+
/// ### Example
19+
/// ```no_run
20+
/// fn ty<F: std::fmt::Debug>(a: F)
21+
/// where
22+
/// F: Sized,
23+
/// {}
24+
/// ```
25+
/// Use instead:
26+
/// ```no_run
27+
/// fn ty<F>(a: F)
28+
/// where
29+
/// F: Sized + std::fmt::Debug,
30+
/// {}
31+
/// ```
32+
#[clippy::version = "1.77.0"]
33+
pub MULTIPLE_BOUND_LOCATIONS,
34+
suspicious,
35+
"defining generic bounds in multiple locations"
36+
}
37+
38+
declare_lint_pass!(MultipleBoundLocations => [MULTIPLE_BOUND_LOCATIONS]);
39+
40+
impl EarlyLintPass for MultipleBoundLocations {
41+
fn check_fn(&mut self, cx: &EarlyContext<'_>, kind: FnKind<'_>, _: Span, _: NodeId) {
42+
if let FnKind::Fn(_, _, _, _, generics, _) = kind
43+
&& !generics.params.is_empty()
44+
&& !generics.where_clause.predicates.is_empty()
45+
{
46+
let mut generic_params_with_bounds = FxHashMap::default();
47+
48+
for param in &generics.params {
49+
if !param.bounds.is_empty() {
50+
generic_params_with_bounds.insert(param.ident.name.as_str(), param.ident.span);
51+
}
52+
}
53+
for clause in &generics.where_clause.predicates {
54+
match clause {
55+
WherePredicate::BoundPredicate(pred) => {
56+
if (!pred.bound_generic_params.is_empty() || !pred.bounds.is_empty())
57+
&& let Some(name) = snippet_opt(cx, pred.bounded_ty.span)
58+
&& let Some(bound_span) = generic_params_with_bounds.get(name.as_str())
59+
{
60+
emit_lint(cx, *bound_span, pred.bounded_ty.span);
61+
}
62+
},
63+
WherePredicate::RegionPredicate(pred) => {
64+
if !pred.bounds.is_empty()
65+
&& let Some(bound_span) = generic_params_with_bounds.get(&pred.lifetime.ident.name.as_str())
66+
{
67+
emit_lint(cx, *bound_span, pred.lifetime.ident.span);
68+
}
69+
},
70+
WherePredicate::EqPredicate(_) => {},
71+
}
72+
}
73+
}
74+
}
75+
}
76+
77+
fn emit_lint(cx: &EarlyContext<'_>, bound_span: Span, where_span: Span) {
78+
span_lint(
79+
cx,
80+
MULTIPLE_BOUND_LOCATIONS,
81+
vec![bound_span, where_span],
82+
"bound is defined in more than one place",
83+
);
84+
}

tests/ui/multiple_bound_locations.rs

+60
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
#![warn(clippy::multiple_bound_locations)]
2+
3+
fn ty<F: std::fmt::Debug>(a: F)
4+
//~^ ERROR: bound is defined in more than one place
5+
where
6+
F: Sized,
7+
{
8+
}
9+
10+
fn lifetime<'a, 'b: 'a, 'c>(a: &'b str, b: &'a str, c: &'c str)
11+
//~^ ERROR: bound is defined in more than one place
12+
where
13+
'b: 'c,
14+
{
15+
}
16+
17+
fn ty_pred<F: Sized>()
18+
//~^ ERROR: bound is defined in more than one place
19+
where
20+
for<'a> F: Send + 'a,
21+
{
22+
}
23+
24+
struct B;
25+
26+
impl B {
27+
fn ty<F: std::fmt::Debug>(a: F)
28+
//~^ ERROR: bound is defined in more than one place
29+
where
30+
F: Sized,
31+
{
32+
}
33+
34+
fn lifetime<'a, 'b: 'a, 'c>(a: &'b str, b: &'a str, c: &'c str)
35+
//~^ ERROR: bound is defined in more than one place
36+
where
37+
'b: 'c,
38+
{
39+
}
40+
41+
fn ty_pred<F: Sized>()
42+
//~^ ERROR: bound is defined in more than one place
43+
where
44+
for<'a> F: Send + 'a,
45+
{
46+
}
47+
}
48+
49+
struct C<F>(F);
50+
51+
impl<F> C<F> {
52+
fn foo(_f: F) -> Self
53+
where
54+
F: std::fmt::Display,
55+
{
56+
todo!()
57+
}
58+
}
59+
60+
fn main() {}
+59
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
error: bound is defined in more than one place
2+
--> tests/ui/multiple_bound_locations.rs:3:7
3+
|
4+
LL | fn ty<F: std::fmt::Debug>(a: F)
5+
| ^
6+
...
7+
LL | F: Sized,
8+
| ^
9+
|
10+
= note: `-D clippy::multiple-bound-locations` implied by `-D warnings`
11+
= help: to override `-D warnings` add `#[allow(clippy::multiple_bound_locations)]`
12+
13+
error: bound is defined in more than one place
14+
--> tests/ui/multiple_bound_locations.rs:10:17
15+
|
16+
LL | fn lifetime<'a, 'b: 'a, 'c>(a: &'b str, b: &'a str, c: &'c str)
17+
| ^^
18+
...
19+
LL | 'b: 'c,
20+
| ^^
21+
22+
error: bound is defined in more than one place
23+
--> tests/ui/multiple_bound_locations.rs:17:12
24+
|
25+
LL | fn ty_pred<F: Sized>()
26+
| ^
27+
...
28+
LL | for<'a> F: Send + 'a,
29+
| ^
30+
31+
error: bound is defined in more than one place
32+
--> tests/ui/multiple_bound_locations.rs:27:11
33+
|
34+
LL | fn ty<F: std::fmt::Debug>(a: F)
35+
| ^
36+
...
37+
LL | F: Sized,
38+
| ^
39+
40+
error: bound is defined in more than one place
41+
--> tests/ui/multiple_bound_locations.rs:34:21
42+
|
43+
LL | fn lifetime<'a, 'b: 'a, 'c>(a: &'b str, b: &'a str, c: &'c str)
44+
| ^^
45+
...
46+
LL | 'b: 'c,
47+
| ^^
48+
49+
error: bound is defined in more than one place
50+
--> tests/ui/multiple_bound_locations.rs:41:16
51+
|
52+
LL | fn ty_pred<F: Sized>()
53+
| ^
54+
...
55+
LL | for<'a> F: Send + 'a,
56+
| ^
57+
58+
error: aborting due to 6 previous errors
59+

tests/ui/trait_duplication_in_bounds_unfixable.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
#![deny(clippy::trait_duplication_in_bounds)]
2+
#![allow(clippy::multiple_bound_locations)]
23

34
use std::collections::BTreeMap;
45
use std::ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Sub, SubAssign};

tests/ui/trait_duplication_in_bounds_unfixable.stderr

+8-8
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
error: this trait bound is already specified in the where clause
2-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:6:15
2+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:7:15
33
|
44
LL | fn bad_foo<T: Clone + Default, Z: Copy>(arg0: T, arg1: Z)
55
| ^^^^^
@@ -12,55 +12,55 @@ LL | #![deny(clippy::trait_duplication_in_bounds)]
1212
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
1313

1414
error: this trait bound is already specified in the where clause
15-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:6:23
15+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:7:23
1616
|
1717
LL | fn bad_foo<T: Clone + Default, Z: Copy>(arg0: T, arg1: Z)
1818
| ^^^^^^^
1919
|
2020
= help: consider removing this trait bound
2121

2222
error: this trait bound is already specified in trait declaration
23-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:37:15
23+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:38:15
2424
|
2525
LL | Self: Default;
2626
| ^^^^^^^
2727
|
2828
= help: consider removing this trait bound
2929

3030
error: this trait bound is already specified in trait declaration
31-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:52:15
31+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:53:15
3232
|
3333
LL | Self: Default + Clone;
3434
| ^^^^^^^
3535
|
3636
= help: consider removing this trait bound
3737

3838
error: this trait bound is already specified in trait declaration
39-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:59:15
39+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:60:15
4040
|
4141
LL | Self: Default + Clone;
4242
| ^^^^^^^
4343
|
4444
= help: consider removing this trait bound
4545

4646
error: this trait bound is already specified in trait declaration
47-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:59:25
47+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:60:25
4848
|
4949
LL | Self: Default + Clone;
5050
| ^^^^^
5151
|
5252
= help: consider removing this trait bound
5353

5454
error: this trait bound is already specified in trait declaration
55-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:64:15
55+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:65:15
5656
|
5757
LL | Self: Default;
5858
| ^^^^^^^
5959
|
6060
= help: consider removing this trait bound
6161

6262
error: this trait bound is already specified in trait declaration
63-
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:100:15
63+
--> tests/ui/trait_duplication_in_bounds_unfixable.rs:101:15
6464
|
6565
LL | Self: Iterator<Item = Foo>,
6666
| ^^^^^^^^^^^^^^^^^^^^

tests/ui/type_repetition_in_bounds.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
#![deny(clippy::type_repetition_in_bounds)]
2-
#![allow(clippy::extra_unused_type_parameters)]
2+
#![allow(clippy::extra_unused_type_parameters, clippy::multiple_bound_locations)]
33

44
use serde::Deserialize;
55
use std::ops::{Add, AddAssign, Div, DivAssign, Mul, MulAssign, Sub, SubAssign};

tests/ui/unnecessary_cast.fixed

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
#![warn(clippy::unnecessary_cast)]
33
#![allow(
44
clippy::borrow_as_ptr,
5+
clippy::multiple_bound_locations,
56
clippy::no_effect,
67
clippy::nonstandard_macro_braces,
78
clippy::unnecessary_operation,

tests/ui/unnecessary_cast.rs

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
#![warn(clippy::unnecessary_cast)]
33
#![allow(
44
clippy::borrow_as_ptr,
5+
clippy::multiple_bound_locations,
56
clippy::no_effect,
67
clippy::nonstandard_macro_braces,
78
clippy::unnecessary_operation,

0 commit comments

Comments
 (0)