2015-10-28 16:50:00 +00:00
use rustc ::lint ::{ LateContext , LateLintPass , LintArray , LintPass } ;
2016-05-13 14:43:47 +00:00
use rustc ::hir ::def ::{ Def , PathResolution } ;
2016-05-25 16:51:35 +00:00
use rustc ::hir ::{ Expr , Expr_ , Stmt , StmtSemi , BlockCheckMode , UnsafeSource } ;
2016-05-13 14:43:47 +00:00
use utils ::{ in_macro , span_lint , snippet_opt , span_lint_and_then } ;
use std ::ops ::Deref ;
2015-10-28 16:50:00 +00:00
2016-02-05 23:41:54 +00:00
/// **What it does:** This lint checks for statements which have no effect.
2015-12-14 21:16:56 +00:00
///
/// **Why is this bad?** Similar to dead code, these statements are actually executed. However, as they have no effect, all they do is make the code less readable.
///
/// **Known problems:** None.
///
/// **Example:** `0;`
2015-10-28 16:50:00 +00:00
declare_lint! {
pub NO_EFFECT ,
Warn ,
" statements with no effect "
}
2016-05-13 14:43:47 +00:00
/// **What it does:** This lint checks for expression statements that can be reduced to a sub-expression
///
/// **Why is this bad?** Expressions by themselves often have no side-effects. Having such expressions reduces redability.
///
/// **Known problems:** None.
///
/// **Example:** `compute_array()[0];`
declare_lint! {
pub UNNECESSARY_OPERATION ,
Warn ,
" outer expressions with no effect "
}
2015-10-28 16:50:00 +00:00
fn has_no_effect ( cx : & LateContext , expr : & Expr ) -> bool {
if in_macro ( cx , expr . span ) {
return false ;
}
match expr . node {
2016-02-11 12:50:41 +00:00
Expr_ ::ExprLit ( .. ) |
Expr_ ::ExprClosure ( .. ) |
Expr_ ::ExprPath ( .. ) = > true ,
Expr_ ::ExprIndex ( ref a , ref b ) |
Expr_ ::ExprBinary ( _ , ref a , ref b ) = > has_no_effect ( cx , a ) & & has_no_effect ( cx , b ) ,
Expr_ ::ExprVec ( ref v ) |
Expr_ ::ExprTup ( ref v ) = > v . iter ( ) . all ( | val | has_no_effect ( cx , val ) ) ,
Expr_ ::ExprRepeat ( ref inner , _ ) |
Expr_ ::ExprCast ( ref inner , _ ) |
Expr_ ::ExprType ( ref inner , _ ) |
Expr_ ::ExprUnary ( _ , ref inner ) |
Expr_ ::ExprField ( ref inner , _ ) |
Expr_ ::ExprTupField ( ref inner , _ ) |
Expr_ ::ExprAddrOf ( _ , ref inner ) |
Expr_ ::ExprBox ( ref inner ) = > has_no_effect ( cx , inner ) ,
Expr_ ::ExprStruct ( _ , ref fields , ref base ) = > {
2015-10-28 16:50:00 +00:00
fields . iter ( ) . all ( | field | has_no_effect ( cx , & field . expr ) ) & &
match * base {
Some ( ref base ) = > has_no_effect ( cx , base ) ,
None = > true ,
}
}
2016-02-11 12:50:41 +00:00
Expr_ ::ExprCall ( ref callee , ref args ) = > {
2015-10-28 16:50:00 +00:00
let def = cx . tcx . def_map . borrow ( ) . get ( & callee . id ) . map ( | d | d . full_def ( ) ) ;
match def {
2016-01-22 12:24:44 +00:00
Some ( Def ::Struct ( .. ) ) |
Some ( Def ::Variant ( .. ) ) = > args . iter ( ) . all ( | arg | has_no_effect ( cx , arg ) ) ,
2015-10-28 16:50:00 +00:00
_ = > false ,
}
}
2016-03-07 15:31:38 +00:00
Expr_ ::ExprBlock ( ref block ) = > {
2016-04-14 18:14:03 +00:00
block . stmts . is_empty ( ) & &
if let Some ( ref expr ) = block . expr {
2016-03-07 15:31:38 +00:00
has_no_effect ( cx , expr )
} else {
false
}
}
2015-10-28 16:50:00 +00:00
_ = > false ,
}
}
#[ derive(Copy, Clone) ]
pub struct NoEffectPass ;
impl LintPass for NoEffectPass {
fn get_lints ( & self ) -> LintArray {
2016-05-13 14:43:47 +00:00
lint_array! ( NO_EFFECT , UNNECESSARY_OPERATION )
2015-10-28 16:50:00 +00:00
}
}
impl LateLintPass for NoEffectPass {
fn check_stmt ( & mut self , cx : & LateContext , stmt : & Stmt ) {
if let StmtSemi ( ref expr , _ ) = stmt . node {
if has_no_effect ( cx , expr ) {
2016-01-04 04:26:12 +00:00
span_lint ( cx , NO_EFFECT , stmt . span , " statement with no effect " ) ;
2016-05-13 14:43:47 +00:00
} else if let Some ( reduced ) = reduce_expression ( cx , expr ) {
let mut snippet = String ::new ( ) ;
for e in reduced {
if in_macro ( cx , e . span ) {
return ;
}
if let Some ( snip ) = snippet_opt ( cx , e . span ) {
snippet . push_str ( & snip ) ;
snippet . push ( ';' ) ;
} else {
return ;
}
}
span_lint_and_then ( cx , UNNECESSARY_OPERATION , stmt . span , " statement can be reduced " , | db | {
db . span_suggestion ( stmt . span , " replace it with " , snippet ) ;
} ) ;
}
}
}
}
fn reduce_expression < ' a > ( cx : & LateContext , expr : & ' a Expr ) -> Option < Vec < & ' a Expr > > {
if in_macro ( cx , expr . span ) {
return None ;
}
match expr . node {
Expr_ ::ExprIndex ( ref a , ref b ) |
Expr_ ::ExprBinary ( _ , ref a , ref b ) = > Some ( vec! [ & * * a , & * * b ] ) ,
Expr_ ::ExprVec ( ref v ) |
Expr_ ::ExprTup ( ref v ) = > Some ( v . iter ( ) . map ( Deref ::deref ) . collect ( ) ) ,
Expr_ ::ExprRepeat ( ref inner , _ ) |
Expr_ ::ExprCast ( ref inner , _ ) |
Expr_ ::ExprType ( ref inner , _ ) |
Expr_ ::ExprUnary ( _ , ref inner ) |
Expr_ ::ExprField ( ref inner , _ ) |
Expr_ ::ExprTupField ( ref inner , _ ) |
Expr_ ::ExprAddrOf ( _ , ref inner ) |
Expr_ ::ExprBox ( ref inner ) = > reduce_expression ( cx , inner ) . or_else ( | | Some ( vec! [ inner ] ) ) ,
2016-06-05 23:42:39 +00:00
Expr_ ::ExprStruct ( _ , ref fields , ref base ) = > {
Some ( fields . iter ( ) . map ( | f | & f . expr ) . chain ( base ) . map ( Deref ::deref ) . collect ( ) )
}
2016-05-13 14:43:47 +00:00
Expr_ ::ExprCall ( ref callee , ref args ) = > {
match cx . tcx . def_map . borrow ( ) . get ( & callee . id ) . map ( PathResolution ::full_def ) {
Some ( Def ::Struct ( .. ) ) |
Some ( Def ::Variant ( .. ) ) = > Some ( args . iter ( ) . map ( Deref ::deref ) . collect ( ) ) ,
_ = > None ,
}
}
Expr_ ::ExprBlock ( ref block ) = > {
if block . stmts . is_empty ( ) {
2016-06-05 23:42:39 +00:00
block . expr . as_ref ( ) . and_then ( | e | {
match block . rules {
BlockCheckMode ::UnsafeBlock ( UnsafeSource ::UserProvided ) = > None ,
BlockCheckMode ::DefaultBlock = > Some ( vec! [ & * * e ] ) ,
// in case of compiler-inserted signaling blocks
_ = > reduce_expression ( cx , e ) ,
}
2016-05-13 14:43:47 +00:00
} )
} else {
None
2015-10-28 16:50:00 +00:00
}
}
2016-05-13 14:43:47 +00:00
_ = > None ,
2015-10-28 16:50:00 +00:00
}
}