1
+ use clippy_config:: Conf ;
1
2
use clippy_utils:: diagnostics:: span_lint_and_sugg;
3
+ use clippy_utils:: msrvs:: { self , Msrv } ;
2
4
use clippy_utils:: sugg:: Sugg ;
3
- use clippy_utils:: { SpanlessEq , is_integer_literal} ;
5
+ use clippy_utils:: { SpanlessEq , is_in_const_context , is_integer_literal} ;
4
6
use rustc_errors:: Applicability ;
5
7
use rustc_hir:: { BinOpKind , Expr , ExprKind } ;
6
8
use rustc_lint:: { LateContext , LateLintPass } ;
7
9
use rustc_middle:: ty;
8
- use rustc_session:: declare_lint_pass ;
10
+ use rustc_session:: impl_lint_pass ;
9
11
10
12
declare_clippy_lint ! {
11
13
/// ### What it does
@@ -31,7 +33,36 @@ declare_clippy_lint! {
31
33
"manually reimplementing `is_power_of_two`"
32
34
}
33
35
34
- declare_lint_pass ! ( ManualIsPowerOfTwo => [ MANUAL_IS_POWER_OF_TWO ] ) ;
36
+ pub struct ManualIsPowerOfTwo {
37
+ msrv : Msrv ,
38
+ }
39
+
40
+ impl_lint_pass ! ( ManualIsPowerOfTwo => [ MANUAL_IS_POWER_OF_TWO ] ) ;
41
+
42
+ impl ManualIsPowerOfTwo {
43
+ pub fn new ( conf : & ' static Conf ) -> Self {
44
+ Self { msrv : conf. msrv }
45
+ }
46
+
47
+ fn build_sugg ( & self , cx : & LateContext < ' _ > , expr : & Expr < ' _ > , receiver : & Expr < ' _ > ) {
48
+ if is_in_const_context ( cx) && !self . msrv . meets ( cx, msrvs:: CONST_IS_POWER_OF_TWO ) {
49
+ return ;
50
+ }
51
+
52
+ let mut applicability = Applicability :: MachineApplicable ;
53
+ let snippet = Sugg :: hir_with_applicability ( cx, receiver, "_" , & mut applicability) ;
54
+
55
+ span_lint_and_sugg (
56
+ cx,
57
+ MANUAL_IS_POWER_OF_TWO ,
58
+ expr. span ,
59
+ "manually reimplementing `is_power_of_two`" ,
60
+ "consider using `.is_power_of_two()`" ,
61
+ format ! ( "{}.is_power_of_two()" , snippet. maybe_par( ) ) ,
62
+ applicability,
63
+ ) ;
64
+ }
65
+ }
35
66
36
67
impl < ' tcx > LateLintPass < ' tcx > for ManualIsPowerOfTwo {
37
68
fn check_expr ( & mut self , cx : & LateContext < ' tcx > , expr : & Expr < ' tcx > ) {
@@ -44,39 +75,24 @@ impl<'tcx> LateLintPass<'tcx> for ManualIsPowerOfTwo {
44
75
if let Some ( a) = count_ones_receiver ( cx, lhs)
45
76
&& is_integer_literal ( rhs, 1 )
46
77
{
47
- build_sugg ( cx, expr, a) ;
78
+ self . build_sugg ( cx, expr, a) ;
48
79
} else if let Some ( a) = count_ones_receiver ( cx, rhs)
49
80
&& is_integer_literal ( lhs, 1 )
50
81
{
51
- build_sugg ( cx, expr, a) ;
82
+ self . build_sugg ( cx, expr, a) ;
52
83
} else if is_integer_literal ( rhs, 0 )
53
84
&& let Some ( a) = is_and_minus_one ( cx, lhs)
54
85
{
55
- build_sugg ( cx, expr, a) ;
86
+ self . build_sugg ( cx, expr, a) ;
56
87
} else if is_integer_literal ( lhs, 0 )
57
88
&& let Some ( a) = is_and_minus_one ( cx, rhs)
58
89
{
59
- build_sugg ( cx, expr, a) ;
90
+ self . build_sugg ( cx, expr, a) ;
60
91
}
61
92
}
62
93
}
63
94
}
64
95
65
- fn build_sugg ( cx : & LateContext < ' _ > , expr : & Expr < ' _ > , receiver : & Expr < ' _ > ) {
66
- let mut applicability = Applicability :: MachineApplicable ;
67
- let snippet = Sugg :: hir_with_applicability ( cx, receiver, "_" , & mut applicability) ;
68
-
69
- span_lint_and_sugg (
70
- cx,
71
- MANUAL_IS_POWER_OF_TWO ,
72
- expr. span ,
73
- "manually reimplementing `is_power_of_two`" ,
74
- "consider using `.is_power_of_two()`" ,
75
- format ! ( "{}.is_power_of_two()" , snippet. maybe_par( ) ) ,
76
- applicability,
77
- ) ;
78
- }
79
-
80
96
/// Return the unsigned integer receiver of `.count_ones()`
81
97
fn count_ones_receiver < ' tcx > ( cx : & LateContext < ' tcx > , expr : & Expr < ' tcx > ) -> Option < & ' tcx Expr < ' tcx > > {
82
98
if let ExprKind :: MethodCall ( method_name, receiver, [ ] , _) = expr. kind
0 commit comments