2015-05-04 12:01:34 +02:00
use rustc ::lint ::* ;
2016-04-07 17:46:48 +02:00
use rustc ::hir ::* ;
2015-10-23 21:30:57 -07:00
use std ::f64 ::consts as f64 ;
2016-02-12 18:35:44 +01:00
use syntax ::ast ::{ Lit , LitKind , FloatTy } ;
2016-02-24 17:38:57 +01:00
use utils ::span_lint ;
2015-05-04 12:01:34 +02:00
2016-02-06 00:41:54 +01:00
/// **What it does:** This lint checks for floating point literals that approximate constants which are defined in [`std::f32::consts`](https://doc.rust-lang.org/stable/std/f32/consts/#constants) or [`std::f64::consts`](https://doc.rust-lang.org/stable/std/f64/consts/#constants), respectively, suggesting to use the predefined constant.
2015-12-11 01:22:27 +01:00
///
/// **Why is this bad?** Usually, the definition in the standard library is more precise than what people come up with. If you find that your definition is actually more precise, please [file a Rust issue](https://github.com/rust-lang/rust/issues).
///
/// **Known problems:** If you happen to have a value that is within 1/8192 of a known constant, but is not *and should not* be the same, this lint will report your value anyway. We have not yet noticed any false positives in code we tested clippy with (this includes servo), but YMMV.
///
/// **Example:** `let x = 3.14;`
2015-05-04 12:01:34 +02:00
declare_lint! {
pub APPROX_CONSTANT ,
Warn ,
2015-08-13 10:32:35 +02:00
" the approximate of a known float constant (in `std::f64::consts` or `std::f32::consts`) \
is found ; suggests to use the constant "
2015-05-04 12:01:34 +02:00
}
2015-10-23 21:30:57 -07:00
// Tuples are of the form (constant, name, min_digits)
2016-01-04 09:56:12 +05:30
const KNOWN_CONSTS : & 'static [ ( f64 , & 'static str , usize ) ] = & [ ( f64 ::E , " E " , 4 ) ,
( f64 ::FRAC_1_PI , " FRAC_1_PI " , 4 ) ,
( f64 ::FRAC_1_SQRT_2 , " FRAC_1_SQRT_2 " , 5 ) ,
( f64 ::FRAC_2_PI , " FRAC_2_PI " , 5 ) ,
( f64 ::FRAC_2_SQRT_PI , " FRAC_2_SQRT_PI " , 5 ) ,
( f64 ::FRAC_PI_2 , " FRAC_PI_2 " , 5 ) ,
( f64 ::FRAC_PI_3 , " FRAC_PI_3 " , 5 ) ,
( f64 ::FRAC_PI_4 , " FRAC_PI_4 " , 5 ) ,
( f64 ::FRAC_PI_6 , " FRAC_PI_6 " , 5 ) ,
( f64 ::FRAC_PI_8 , " FRAC_PI_8 " , 5 ) ,
( f64 ::LN_10 , " LN_10 " , 5 ) ,
( f64 ::LN_2 , " LN_2 " , 5 ) ,
( f64 ::LOG10_E , " LOG10_E " , 5 ) ,
( f64 ::LOG2_E , " LOG2_E " , 5 ) ,
( f64 ::PI , " PI " , 3 ) ,
( f64 ::SQRT_2 , " SQRT_2 " , 5 ) ] ;
2015-05-04 12:01:34 +02:00
#[ derive(Copy,Clone) ]
2016-06-10 16:17:20 +02:00
pub struct Pass ;
2015-05-04 12:01:34 +02:00
2016-06-10 16:17:20 +02:00
impl LintPass for Pass {
2015-05-04 12:01:34 +02:00
fn get_lints ( & self ) -> LintArray {
lint_array! ( APPROX_CONSTANT )
}
2015-09-19 08:23:04 +05:30
}
2015-08-11 20:22:20 +02:00
2016-06-10 16:17:20 +02:00
impl LateLintPass for Pass {
2015-09-19 08:23:04 +05:30
fn check_expr ( & mut self , cx : & LateContext , e : & Expr ) {
2015-11-25 02:44:40 +09:00
if let ExprLit ( ref lit ) = e . node {
2015-09-11 15:59:19 +02:00
check_lit ( cx , lit , e ) ;
2015-08-11 20:22:20 +02:00
}
2015-05-04 12:01:34 +02:00
}
}
2015-09-19 08:23:04 +05:30
fn check_lit ( cx : & LateContext , lit : & Lit , e : & Expr ) {
2015-08-21 20:44:48 +02:00
match lit . node {
2016-02-12 18:35:44 +01:00
LitKind ::Float ( ref s , FloatTy ::F32 ) = > check_known_consts ( cx , e , s , " f32 " ) ,
LitKind ::Float ( ref s , FloatTy ::F64 ) = > check_known_consts ( cx , e , s , " f64 " ) ,
LitKind ::FloatUnsuffixed ( ref s ) = > check_known_consts ( cx , e , s , " f{32, 64} " ) ,
2016-01-04 09:56:12 +05:30
_ = > ( ) ,
2015-08-11 20:22:20 +02:00
}
2015-05-04 12:01:34 +02:00
}
2015-10-22 15:19:03 -07:00
fn check_known_consts ( cx : & LateContext , e : & Expr , s : & str , module : & str ) {
2015-10-23 21:30:57 -07:00
if let Ok ( _ ) = s . parse ::< f64 > ( ) {
for & ( constant , name , min_digits ) in KNOWN_CONSTS {
if is_approx_const ( constant , s , min_digits ) {
2016-01-04 09:56:12 +05:30
span_lint ( cx ,
APPROX_CONSTANT ,
e . span ,
2016-06-08 19:58:29 -04:00
& format! ( " approximate value of ` {} ::consts:: {} ` found. Consider using it directly " , module , & name ) ) ;
2015-10-23 21:30:57 -07:00
return ;
2015-10-22 15:19:03 -07:00
}
2015-08-11 20:22:20 +02:00
}
}
2015-05-04 12:01:34 +02:00
}
2015-10-23 21:30:57 -07:00
/// Returns false if the number of significant figures in `value` are
/// less than `min_digits`; otherwise, returns true if `value` is equal
/// to `constant`, rounded to the number of digits present in `value`.
fn is_approx_const ( constant : f64 , value : & str , min_digits : usize ) -> bool {
if value . len ( ) < = min_digits {
false
} else {
let round_const = format! ( " {:.*} " , value . len ( ) - 2 , constant ) ;
let mut trunc_const = constant . to_string ( ) ;
if trunc_const . len ( ) > value . len ( ) {
trunc_const . truncate ( value . len ( ) ) ;
}
( value = = round_const ) | | ( value = = trunc_const )
}
}