use rustc_ast::visit::FnKind; use rustc_ast::{NodeId, WherePredicate}; use rustc_data_structures::fx::FxHashMap; use rustc_lint::{EarlyContext, EarlyLintPass}; use rustc_session::declare_lint_pass; use rustc_span::Span; use clippy_utils::diagnostics::span_lint; use clippy_utils::source::snippet_opt; declare_clippy_lint! { /// ### What it does /// Check if a generic is defined both in the bound predicate and in the `where` clause. /// /// ### Why is this bad? /// It can be confusing for developers when seeing bounds for a generic in multiple places. /// /// ### Example /// ```no_run /// fn ty(a: F) /// where /// F: Sized, /// {} /// ``` /// Use instead: /// ```no_run /// fn ty(a: F) /// where /// F: Sized + std::fmt::Debug, /// {} /// ``` #[clippy::version = "1.77.0"] pub MULTIPLE_BOUND_LOCATIONS, suspicious, "defining generic bounds in multiple locations" } declare_lint_pass!(MultipleBoundLocations => [MULTIPLE_BOUND_LOCATIONS]); impl EarlyLintPass for MultipleBoundLocations { fn check_fn(&mut self, cx: &EarlyContext<'_>, kind: FnKind<'_>, _: Span, _: NodeId) { if let FnKind::Fn(_, _, _, _, generics, _) = kind && !generics.params.is_empty() && !generics.where_clause.predicates.is_empty() { let mut generic_params_with_bounds = FxHashMap::default(); for param in &generics.params { if !param.bounds.is_empty() { generic_params_with_bounds.insert(param.ident.name.as_str(), param.ident.span); } } for clause in &generics.where_clause.predicates { match clause { WherePredicate::BoundPredicate(pred) => { if (!pred.bound_generic_params.is_empty() || !pred.bounds.is_empty()) && let Some(name) = snippet_opt(cx, pred.bounded_ty.span) && let Some(bound_span) = generic_params_with_bounds.get(name.as_str()) { emit_lint(cx, *bound_span, pred.bounded_ty.span); } }, WherePredicate::RegionPredicate(pred) => { if !pred.bounds.is_empty() && let Some(bound_span) = generic_params_with_bounds.get(&pred.lifetime.ident.name.as_str()) { emit_lint(cx, *bound_span, pred.lifetime.ident.span); } }, WherePredicate::EqPredicate(_) => {}, } } } } } fn emit_lint(cx: &EarlyContext<'_>, bound_span: Span, where_span: Span) { span_lint( cx, MULTIPLE_BOUND_LOCATIONS, vec![bound_span, where_span], "bound is defined in more than one place", ); }