1
1
use crate :: utils:: { is_entrypoint_fn, span_lint} ;
2
+ use if_chain:: if_chain;
2
3
use rustc:: hir;
3
4
use rustc:: hir:: intravisit:: FnKind ;
4
5
use rustc:: hir:: { Body , Constness , FnDecl , HirId } ;
5
- use rustc:: lint:: { LateContext , LateLintPass , LintArray , LintPass } ;
6
+ use rustc:: lint:: { in_external_macro , LateContext , LateLintPass , LintArray , LintPass } ;
6
7
use rustc:: { declare_tool_lint, lint_array} ;
7
8
use rustc_mir:: transform:: qualify_min_const_fn:: is_min_const_fn;
8
9
use syntax_pos:: Span ;
@@ -82,7 +83,7 @@ impl<'a, 'tcx> LateLintPass<'a, 'tcx> for MissingConstForFn {
82
83
) {
83
84
let def_id = cx. tcx . hir ( ) . local_def_id_from_hir_id ( hir_id) ;
84
85
85
- if is_entrypoint_fn ( cx, def_id) {
86
+ if in_external_macro ( cx . tcx . sess , span ) || is_entrypoint_fn ( cx, def_id) {
86
87
return ;
87
88
}
88
89
@@ -95,7 +96,7 @@ impl<'a, 'tcx> LateLintPass<'a, 'tcx> for MissingConstForFn {
95
96
}
96
97
} ,
97
98
FnKind :: Method ( _, sig, ..) => {
98
- if already_const ( sig. header ) {
99
+ if is_trait_method ( cx , hir_id ) || already_const ( sig. header ) {
99
100
return ;
100
101
}
101
102
} ,
@@ -114,6 +115,18 @@ impl<'a, 'tcx> LateLintPass<'a, 'tcx> for MissingConstForFn {
114
115
}
115
116
}
116
117
118
+ fn is_trait_method ( cx : & LateContext < ' _ , ' _ > , hir_id : HirId ) -> bool {
119
+ // Get the implemented trait for the current function
120
+ let parent_impl = cx. tcx . hir ( ) . get_parent_item ( hir_id) ;
121
+ if_chain ! {
122
+ if parent_impl != hir:: CRATE_HIR_ID ;
123
+ if let hir:: Node :: Item ( item) = cx. tcx. hir( ) . get_by_hir_id( parent_impl) ;
124
+ if let hir:: ItemKind :: Impl ( _, _, _, _, Some ( _trait_ref) , _, _) = & item. node;
125
+ then { return true ; }
126
+ }
127
+ false
128
+ }
129
+
117
130
// We don't have to lint on something that's already `const`
118
131
fn already_const ( header : hir:: FnHeader ) -> bool {
119
132
header. constness == Constness :: Const
0 commit comments