2016-04-07 15:46:48 +00:00
use rustc ::hir ::* ;
use rustc ::hir ::def ::{ Def , PathResolution } ;
2015-04-30 09:48:43 +00:00
use rustc ::lint ::* ;
2016-03-31 15:05:43 +00:00
use rustc_const_eval ::lookup_const_by_id ;
2016-02-12 17:35:44 +00:00
use syntax ::ast ::LitKind ;
2016-02-24 16:38:57 +00:00
use syntax ::codemap ::Span ;
2015-07-26 14:53:11 +00:00
use utils ::span_lint ;
2015-04-30 09:48:43 +00:00
2016-02-05 23:41:54 +00:00
/// **What it does:** This lint checks for incompatible bit masks in comparisons.
2015-12-11 00:22:27 +00:00
///
/// The formula for detecting if an expression of the type `_ <bit_op> m <cmp_op> c` (where `<bit_op>`
2015-12-21 19:47:19 +00:00
/// is one of {`&`, `|`} and `<cmp_op>` is one of {`!=`, `>=`, `>`, `!=`, `>=`, `>`}) can be determined from the following table:
2015-12-11 00:22:27 +00:00
///
2016-03-28 19:23:21 +00:00
/// |Comparison |Bit Op|Example |is always|Formula |
/// |------------|------|------------|---------|----------------------|
/// |`==` or `!=`| `&` |`x & 2 == 3`|`false` |`c & m != c` |
/// |`<` or `>=`| `&` |`x & 2 < 3` |`true` |`m < c` |
/// |`>` or `<=`| `&` |`x & 1 > 1` |`false` |`m <= c` |
/// |`==` or `!=`| `|` |`x | 1 == 0`|`false` |`c | m != c` |
/// |`<` or `>=`| `|` |`x | 1 < 1` |`false` |`m >= c` |
/// |`<=` or `>` | `|` |`x | 1 > 0` |`true` |`m > c` |
2015-12-11 00:22:27 +00:00
///
/// **Why is this bad?** If the bits that the comparison cares about are always set to zero or one by the bit mask, the comparison is constant `true` or `false` (depending on mask, compared value, and operators).
///
/// So the code is actively misleading, and the only reason someone would write this intentionally is to win an underhanded Rust contest or create a test-case for this lint.
///
/// **Known problems:** None
///
/// **Example:** `x & 1 == 2` (also see table above)
2015-04-30 09:48:43 +00:00
declare_lint! {
pub BAD_BIT_MASK ,
2015-09-06 14:09:35 +00:00
Warn ,
2015-08-13 08:32:35 +00:00
" expressions of the form `_ & mask == select` that will only ever return `true` or `false` \
( because in the example ` select ` containing bits that ` mask ` doesn ' t have ) "
2015-04-30 09:48:43 +00:00
}
2015-12-11 00:22:27 +00:00
/// **What it does:** This lint checks for bit masks in comparisons which can be removed without changing the outcome. The basic structure can be seen in the following table:
///
2016-03-19 16:48:29 +00:00
/// |Comparison| Bit Op |Example |equals |
2015-12-11 00:22:27 +00:00
/// |----------|---------|-----------|-------|
/// |`>` / `<=`|`|` / `^`|`x | 2 > 3`|`x > 3`|
/// |`<` / `>=`|`|` / `^`|`x ^ 1 < 4`|`x < 4`|
///
/// **Why is this bad?** Not equally evil as [`bad_bit_mask`](#bad_bit_mask), but still a bit misleading, because the bit mask is ineffective.
///
/// **Known problems:** False negatives: This lint will only match instances where we have figured out the math (which is for a power-of-two compared value). This means things like `x | 1 >= 7` (which would be better written as `x >= 6`) will not be reported (but bit masks like this are fairly uncommon).
///
/// **Example:** `x | 1 > 3` (also see table above)
2015-05-15 12:09:29 +00:00
declare_lint! {
2015-08-11 18:22:20 +00:00
pub INEFFECTIVE_BIT_MASK ,
Warn ,
2015-08-13 08:32:35 +00:00
" expressions where a bit mask will be rendered useless by a comparison, e.g. `(x | 1) > 2` "
2015-05-15 12:09:29 +00:00
}
2015-08-11 18:22:20 +00:00
/// Checks for incompatible bit masks in comparisons, e.g. `x & 1 == 2`.
2015-05-22 22:49:13 +00:00
/// This cannot work because the bit that makes up the value two was
/// zeroed out by the bit-and with 1. So the formula for detecting if an
2015-08-11 18:22:20 +00:00
/// expression of the type `_ <bit_op> m <cmp_op> c` (where `<bit_op>`
/// is one of {`&`, '|'} and `<cmp_op>` is one of {`!=`, `>=`, `>` ,
2015-05-22 22:49:13 +00:00
/// `!=`, `>=`, `>`}) can be determined from the following table:
2015-08-11 18:22:20 +00:00
///
2016-03-28 19:23:21 +00:00
/// |Comparison |Bit Op|Example |is always|Formula |
/// |------------|------|------------|---------|----------------------|
/// |`==` or `!=`| `&` |`x & 2 == 3`|`false` |`c & m != c` |
/// |`<` or `>=`| `&` |`x & 2 < 3` |`true` |`m < c` |
/// |`>` or `<=`| `&` |`x & 1 > 1` |`false` |`m <= c` |
/// |`==` or `!=`| `|` |`x | 1 == 0`|`false` |`c | m != c` |
/// |`<` or `>=`| `|` |`x | 1 < 1` |`false` |`m >= c` |
/// |`<=` or `>` | `|` |`x | 1 > 0` |`true` |`m > c` |
2015-08-11 18:22:20 +00:00
///
2015-05-22 22:49:13 +00:00
/// This lint is **deny** by default
///
/// There is also a lint that warns on ineffective masks that is *warn*
2015-08-19 07:07:50 +00:00
/// by default.
///
2016-03-28 19:23:21 +00:00
/// |Comparison|Bit Op |Example |equals |Formula|
2015-08-19 07:07:50 +00:00
/// |`>` / `<=`|`|` / `^`|`x | 2 > 3`|`x > 3`|`¹ && m <= c`|
/// |`<` / `>=`|`|` / `^`|`x ^ 1 < 4`|`x < 4`|`¹ && m < c` |
///
/// `¹ power_of_two(c + 1)`
2015-04-30 09:48:43 +00:00
#[ derive(Copy,Clone) ]
pub struct BitMask ;
impl LintPass for BitMask {
fn get_lints ( & self ) -> LintArray {
2015-05-15 12:09:29 +00:00
lint_array! ( BAD_BIT_MASK , INEFFECTIVE_BIT_MASK )
2015-04-30 09:48:43 +00:00
}
2015-09-19 02:53:04 +00:00
}
2015-08-11 18:22:20 +00:00
2015-09-19 02:53:04 +00:00
impl LateLintPass for BitMask {
fn check_expr ( & mut self , cx : & LateContext , e : & Expr ) {
2015-04-30 09:48:43 +00:00
if let ExprBinary ( ref cmp , ref left , ref right ) = e . node {
2016-04-07 15:46:48 +00:00
if cmp . node . is_comparison ( ) {
2016-06-05 23:16:16 +00:00
if let Some ( cmp_opt ) = fetch_int_literal ( cx , right ) {
check_compare ( cx , left , cmp . node , cmp_opt , & e . span )
} else if let Some ( cmp_val ) = fetch_int_literal ( cx , left ) {
check_compare ( cx , right , invert_cmp ( cmp . node ) , cmp_val , & e . span )
}
2015-08-11 18:22:20 +00:00
}
}
2015-04-30 09:48:43 +00:00
}
}
2016-01-04 04:26:12 +00:00
fn invert_cmp ( cmp : BinOp_ ) -> BinOp_ {
2015-08-11 18:22:20 +00:00
match cmp {
BiEq = > BiEq ,
BiNe = > BiNe ,
BiLt = > BiGt ,
BiGt = > BiLt ,
BiLe = > BiGe ,
BiGe = > BiLe ,
2016-01-04 04:26:12 +00:00
_ = > BiOr , // Dummy
2015-08-11 18:22:20 +00:00
}
2015-05-04 05:17:15 +00:00
}
2015-09-19 02:53:04 +00:00
fn check_compare ( cx : & LateContext , bit_op : & Expr , cmp_op : BinOp_ , cmp_value : u64 , span : & Span ) {
2015-09-19 03:02:56 +00:00
if let ExprBinary ( ref op , ref left , ref right ) = bit_op . node {
if op . node ! = BiBitAnd & & op . node ! = BiBitOr {
return ;
}
2016-01-04 04:26:12 +00:00
fetch_int_literal ( cx , right )
. or_else ( | | fetch_int_literal ( cx , left ) )
. map_or ( ( ) , | mask | check_bit_mask ( cx , op . node , cmp_op , mask , cmp_value , span ) )
2015-08-11 18:22:20 +00:00
}
2015-04-30 09:48:43 +00:00
}
2016-01-04 04:26:12 +00:00
fn check_bit_mask ( cx : & LateContext , bit_op : BinOp_ , cmp_op : BinOp_ , mask_value : u64 , cmp_value : u64 , span : & Span ) {
2015-08-11 18:22:20 +00:00
match cmp_op {
2016-01-04 04:26:12 +00:00
BiEq | BiNe = > {
match bit_op {
BiBitAnd = > {
if mask_value & cmp_value ! = cmp_value {
if cmp_value ! = 0 {
span_lint ( cx ,
BAD_BIT_MASK ,
* span ,
& format! ( " incompatible bit mask: `_ & {} ` can never be equal to ` {} ` " ,
mask_value ,
cmp_value ) ) ;
}
2016-01-13 17:32:05 +00:00
} else if mask_value = = 0 {
span_lint ( cx , BAD_BIT_MASK , * span , " &-masking with zero " ) ;
2016-01-04 04:26:12 +00:00
}
2015-08-11 18:22:20 +00:00
}
2016-01-04 04:26:12 +00:00
BiBitOr = > {
if mask_value | cmp_value ! = cmp_value {
span_lint ( cx ,
BAD_BIT_MASK ,
* span ,
& format! ( " incompatible bit mask: `_ | {} ` can never be equal to ` {} ` " ,
mask_value ,
cmp_value ) ) ;
}
2015-08-11 18:22:20 +00:00
}
2016-01-04 04:26:12 +00:00
_ = > ( ) ,
}
}
BiLt | BiGe = > {
match bit_op {
BiBitAnd = > {
if mask_value < cmp_value {
span_lint ( cx ,
BAD_BIT_MASK ,
* span ,
& format! ( " incompatible bit mask: `_ & {} ` will always be lower than ` {} ` " ,
mask_value ,
cmp_value ) ) ;
2016-01-13 17:32:05 +00:00
} else if mask_value = = 0 {
span_lint ( cx , BAD_BIT_MASK , * span , " &-masking with zero " ) ;
2016-01-04 04:26:12 +00:00
}
2015-08-11 18:22:20 +00:00
}
2016-01-04 04:26:12 +00:00
BiBitOr = > {
if mask_value > = cmp_value {
span_lint ( cx ,
BAD_BIT_MASK ,
* span ,
& format! ( " incompatible bit mask: `_ | {} ` will never be lower than ` {} ` " ,
mask_value ,
cmp_value ) ) ;
} else {
check_ineffective_lt ( cx , * span , mask_value , cmp_value , " | " ) ;
}
2015-08-11 18:22:20 +00:00
}
2016-01-04 04:26:12 +00:00
BiBitXor = > check_ineffective_lt ( cx , * span , mask_value , cmp_value , " ^ " ) ,
_ = > ( ) ,
}
}
BiLe | BiGt = > {
match bit_op {
BiBitAnd = > {
if mask_value < = cmp_value {
span_lint ( cx ,
BAD_BIT_MASK ,
* span ,
& format! ( " incompatible bit mask: `_ & {} ` will never be higher than ` {} ` " ,
mask_value ,
cmp_value ) ) ;
2016-01-13 17:32:05 +00:00
} else if mask_value = = 0 {
span_lint ( cx , BAD_BIT_MASK , * span , " &-masking with zero " ) ;
2016-01-04 04:26:12 +00:00
}
}
BiBitOr = > {
if mask_value > cmp_value {
span_lint ( cx ,
BAD_BIT_MASK ,
* span ,
& format! ( " incompatible bit mask: `_ | {} ` will always be higher than ` {} ` " ,
mask_value ,
cmp_value ) ) ;
} else {
check_ineffective_gt ( cx , * span , mask_value , cmp_value , " | " ) ;
}
}
BiBitXor = > check_ineffective_gt ( cx , * span , mask_value , cmp_value , " ^ " ) ,
_ = > ( ) ,
}
}
_ = > ( ) ,
2015-08-11 18:22:20 +00:00
}
2015-04-30 09:48:43 +00:00
}
2015-09-19 02:53:04 +00:00
fn check_ineffective_lt ( cx : & LateContext , span : Span , m : u64 , c : u64 , op : & str ) {
2015-08-19 07:07:50 +00:00
if c . is_power_of_two ( ) & & m < c {
2016-01-04 04:26:12 +00:00
span_lint ( cx ,
INEFFECTIVE_BIT_MASK ,
span ,
& format! ( " ineffective bit mask: `x {} {} ` compared to ` {} `, is the same as x compared directly " ,
op ,
m ,
c ) ) ;
2015-08-19 07:07:50 +00:00
}
}
2015-09-19 02:53:04 +00:00
fn check_ineffective_gt ( cx : & LateContext , span : Span , m : u64 , c : u64 , op : & str ) {
2015-08-19 07:07:50 +00:00
if ( c + 1 ) . is_power_of_two ( ) & & m < = c {
2016-01-04 04:26:12 +00:00
span_lint ( cx ,
INEFFECTIVE_BIT_MASK ,
span ,
& format! ( " ineffective bit mask: `x {} {} ` compared to ` {} `, is the same as x compared directly " ,
op ,
m ,
c ) ) ;
2015-08-19 07:07:50 +00:00
}
}
2016-01-04 04:26:12 +00:00
fn fetch_int_literal ( cx : & LateContext , lit : & Expr ) -> Option < u64 > {
2015-08-21 18:44:48 +00:00
match lit . node {
ExprLit ( ref lit_ptr ) = > {
2016-02-12 17:35:44 +00:00
if let LitKind ::Int ( value , _ ) = lit_ptr . node {
2015-11-27 15:47:24 +00:00
Some ( value ) //TODO: Handle sign
2016-01-04 04:26:12 +00:00
} else {
None
}
2015-11-17 04:39:42 +00:00
}
2015-08-21 18:44:48 +00:00
ExprPath ( _ , _ ) = > {
2016-01-04 04:26:12 +00:00
{
// Important to let the borrow expire before the const lookup to avoid double
// borrowing.
let def_map = cx . tcx . def_map . borrow ( ) ;
match def_map . get ( & lit . id ) {
2016-04-14 18:14:03 +00:00
Some ( & PathResolution { base_def : Def ::Const ( def_id ) , .. } ) = > Some ( def_id ) ,
2016-01-04 04:26:12 +00:00
_ = > None ,
}
2015-05-07 04:41:54 +00:00
}
2016-03-20 20:24:18 +00:00
. and_then ( | def_id | lookup_const_by_id ( cx . tcx , def_id , None ) )
2016-03-15 19:09:53 +00:00
. and_then ( | ( l , _ty ) | fetch_int_literal ( cx , l ) )
2015-08-11 18:22:20 +00:00
}
2016-01-04 04:26:12 +00:00
_ = > None ,
2015-08-11 18:22:20 +00:00
}
2015-04-30 09:48:43 +00:00
}