1
1
use clippy_utils:: diagnostics:: { span_lint, span_lint_and_help, span_lint_and_sugg} ;
2
2
use clippy_utils:: source:: { snippet, snippet_with_applicability} ;
3
- use clippy_utils:: ty:: { get_associated_type , implements_trait , is_type_diagnostic_item} ;
3
+ use clippy_utils:: ty:: is_type_diagnostic_item;
4
4
use clippy_utils:: { get_parent_expr, is_lint_allowed, match_function_call, method_calls, paths} ;
5
5
use clippy_utils:: { peel_blocks, SpanlessEq } ;
6
6
use if_chain:: if_chain;
7
7
use rustc_errors:: Applicability ;
8
+ use rustc_hir:: def_id:: DefId ;
8
9
use rustc_hir:: { BinOpKind , BorrowKind , Expr , ExprKind , LangItem , QPath } ;
9
10
use rustc_lint:: { LateContext , LateLintPass , LintContext } ;
10
11
use rustc_middle:: lint:: in_external_macro;
11
- use rustc_middle:: ty:: { self , Ty } ;
12
+ use rustc_middle:: ty;
12
13
use rustc_session:: { declare_lint_pass, declare_tool_lint} ;
13
14
use rustc_span:: source_map:: Spanned ;
14
15
use rustc_span:: sym;
@@ -476,13 +477,16 @@ declare_lint_pass!(TrimSplitWhitespace => [TRIM_SPLIT_WHITESPACE]);
476
477
477
478
impl < ' tcx > LateLintPass < ' tcx > for TrimSplitWhitespace {
478
479
fn check_expr ( & mut self , cx : & LateContext < ' tcx > , expr : & Expr < ' _ > ) {
480
+ let tyckres = cx. typeck_results ( ) ;
479
481
if_chain ! {
480
- if let ExprKind :: MethodCall ( path, [ recv ] , split_ws_span) = expr. kind;
482
+ if let ExprKind :: MethodCall ( path, [ split_recv ] , split_ws_span) = expr. kind;
481
483
if path. ident. name == sym!( split_whitespace) ;
482
- if let ExprKind :: MethodCall ( path, [ recv] , trim_span) = recv. kind;
484
+ if let Some ( split_ws_def_id) = tyckres. type_dependent_def_id( expr. hir_id) ;
485
+ if cx. tcx. is_diagnostic_item( sym:: str_split_whitespace, split_ws_def_id) ;
486
+ if let ExprKind :: MethodCall ( path, [ _trim_recv] , trim_span) = split_recv. kind;
483
487
if let trim_fn_name @ ( "trim" | "trim_start" | "trim_end" ) = path. ident. name. as_str( ) ;
484
- let recv_ty = cx . typeck_results ( ) . expr_ty ( recv ) . peel_refs ( ) ;
485
- if recv_ty . is_str ( ) || implements_deref_str ( cx, recv_ty ) ;
488
+ if let Some ( trim_def_id ) = tyckres . type_dependent_def_id ( split_recv . hir_id ) ;
489
+ if is_one_of_trim_diagnostic_items ( cx, trim_def_id ) ;
486
490
then {
487
491
span_lint_and_sugg(
488
492
cx,
@@ -498,17 +502,8 @@ impl<'tcx> LateLintPass<'tcx> for TrimSplitWhitespace {
498
502
}
499
503
}
500
504
501
- /// does ty implement Deref<Target=str>?
502
- fn implements_deref_str < ' t > ( cx : & LateContext < ' t > , ty : Ty < ' t > ) -> bool {
503
- if_chain ! {
504
- if let Some ( deref_trait_id) = cx. tcx. get_diagnostic_item( sym:: Deref ) ;
505
- if implements_trait( cx, ty, deref_trait_id, & [ ] ) ;
506
- if let Some ( target) = get_associated_type( cx, ty, deref_trait_id, "Target" ) ;
507
- then {
508
- target. is_str( )
509
- }
510
- else {
511
- false
512
- }
513
- }
505
+ fn is_one_of_trim_diagnostic_items ( cx : & LateContext < ' _ > , trim_def_id : DefId ) -> bool {
506
+ cx. tcx . is_diagnostic_item ( sym:: str_trim, trim_def_id)
507
+ || cx. tcx . is_diagnostic_item ( sym:: str_trim_start, trim_def_id)
508
+ || cx. tcx . is_diagnostic_item ( sym:: str_trim_end, trim_def_id)
514
509
}
0 commit comments