1 // error-pattern:cargo-clippy
3 #![feature(box_syntax)]
4 #![feature(collections)]
5 #![feature(custom_attribute)]
6 #![feature(rustc_private)]
7 #![feature(slice_patterns)]
8 #![feature(stmt_expr_attributes)]
9 #![feature(repeat_str)]
11 #![allow(indexing_slicing, shadow_reuse, unknown_lints, missing_docs_in_private_items)]
12 #![allow(needless_lifetimes)]
21 // Only for the compile time checking of paths
23 extern crate collections;
25 // for unicode nfc normalization
26 extern crate unicode_normalization;
28 // for semver check in attrs.rs
32 extern crate regex_syntax;
34 // for finding minimal boolean expressions
35 extern crate quine_mc_cluskey;
37 extern crate rustc_serialize;
39 extern crate rustc_errors;
40 extern crate rustc_plugin;
41 extern crate rustc_const_eval;
42 extern crate rustc_const_math;
45 extern crate matches as matches_macro;
47 macro_rules! declare_restriction_lint {
48 { pub $name:tt, $description:tt } => {
49 declare_lint! { pub $name, Allow, $description }
57 // begin lints modules, do not remove this comment, it’s used in `update_lints`
60 pub mod array_indexing;
64 pub mod blacklisted_name;
65 pub mod block_in_if_condition;
67 pub mod collapsible_if;
69 pub mod cyclomatic_complexity;
75 pub mod enum_glob_use;
76 pub mod enum_variants;
79 pub mod eta_reduction;
80 pub mod eval_order_dependence;
85 pub mod if_let_redundant_pattern_matching;
87 pub mod items_after_statements;
101 pub mod mut_reference;
102 pub mod mutex_atomic;
103 pub mod needless_bool;
104 pub mod needless_borrow;
105 pub mod needless_update;
106 pub mod neg_multiply;
107 pub mod new_without_default;
109 pub mod non_expressive_names;
111 pub mod open_options;
112 pub mod overflow_check_conditional;
114 pub mod partialeq_ne_impl;
126 pub mod temporary_assignment;
130 pub mod unsafe_removed_from_name;
131 pub mod unused_label;
133 pub mod zero_div_zero;
134 // end lints modules, do not remove this comment, it’s used in `update_lints`
137 pub use syntax::ast::{Name, NodeId};
140 #[cfg_attr(rustfmt, rustfmt_skip)]
141 pub fn register_plugins(reg: &mut rustc_plugin::Registry) {
142 let conf = match utils::conf::file_from_args(reg.args()) {
144 // if the user specified a file, it must exist, otherwise default to `clippy.toml` but
145 // do not require the file to exist
146 let file_name = if let Some(file_name) = file_name {
149 match utils::conf::lookup_conf_file() {
152 reg.sess.struct_err(&format!("error reading Clippy's configuration file: {}", error)).emit();
158 let (conf, errors) = utils::conf::read(file_name.as_ref().map(|p| p.as_ref()));
160 // all conf errors are non-fatal, we just use the default conf in case of error
161 for error in errors {
162 reg.sess.struct_err(&format!("error reading Clippy's configuration file: {}", error)).emit();
167 Err((err, span)) => {
168 reg.sess.struct_span_err(span, err)
169 .span_note(span, "Clippy will use default configuration")
171 utils::conf::Conf::default()
175 let mut store = reg.sess.lint_store.borrow_mut();
176 store.register_removed(
178 "`Vec::as_slice` has been stabilized in 1.7",
180 store.register_removed(
181 "unstable_as_mut_slice",
182 "`Vec::as_mut_slice` has been stabilized in 1.7",
184 store.register_removed(
186 "using `str::to_string` is common even today and specialization will likely happen soon",
188 store.register_removed(
190 "using `string::to_string` is common even today and specialization will likely happen soon",
192 // end deprecated lints, do not remove this comment, it’s used in `update_lints`
194 reg.register_late_lint_pass(box serde::Serde);
195 reg.register_early_lint_pass(box utils::internal_lints::Clippy);
196 reg.register_late_lint_pass(box utils::internal_lints::LintWithoutLintPass::default());
197 reg.register_late_lint_pass(box utils::inspector::Pass);
198 reg.register_late_lint_pass(box types::TypePass);
199 reg.register_late_lint_pass(box booleans::NonminimalBool);
200 reg.register_late_lint_pass(box eq_op::EqOp);
201 reg.register_early_lint_pass(box enum_variants::EnumVariantNames::new(conf.enum_variant_name_threshold));
202 reg.register_late_lint_pass(box enum_glob_use::EnumGlobUse);
203 reg.register_late_lint_pass(box enum_clike::UnportableVariant);
204 reg.register_late_lint_pass(box bit_mask::BitMask);
205 reg.register_late_lint_pass(box ptr::PointerPass);
206 reg.register_late_lint_pass(box needless_bool::NeedlessBool);
207 reg.register_late_lint_pass(box needless_bool::BoolComparison);
208 reg.register_late_lint_pass(box approx_const::Pass);
209 reg.register_late_lint_pass(box misc::Pass);
210 reg.register_early_lint_pass(box precedence::Precedence);
211 reg.register_late_lint_pass(box eta_reduction::EtaPass);
212 reg.register_late_lint_pass(box identity_op::IdentityOp);
213 reg.register_early_lint_pass(box items_after_statements::ItemsAfterStatements);
214 reg.register_late_lint_pass(box mut_mut::MutMut);
215 reg.register_late_lint_pass(box mut_reference::UnnecessaryMutPassed);
216 reg.register_late_lint_pass(box len_zero::LenZero);
217 reg.register_late_lint_pass(box attrs::AttrPass);
218 reg.register_early_lint_pass(box collapsible_if::CollapsibleIf);
219 reg.register_late_lint_pass(box block_in_if_condition::BlockInIfCondition);
220 reg.register_late_lint_pass(box unicode::Unicode);
221 reg.register_late_lint_pass(box strings::StringAdd);
222 reg.register_early_lint_pass(box returns::ReturnPass);
223 reg.register_late_lint_pass(box methods::Pass);
224 reg.register_late_lint_pass(box shadow::Pass);
225 reg.register_late_lint_pass(box types::LetPass);
226 reg.register_late_lint_pass(box types::UnitCmp);
227 reg.register_late_lint_pass(box loops::Pass);
228 reg.register_late_lint_pass(box lifetimes::LifetimePass);
229 reg.register_late_lint_pass(box entry::HashMapLint);
230 reg.register_late_lint_pass(box ranges::StepByZero);
231 reg.register_late_lint_pass(box types::CastPass);
232 reg.register_late_lint_pass(box types::TypeComplexityPass::new(conf.type_complexity_threshold));
233 reg.register_late_lint_pass(box matches::MatchPass);
234 reg.register_late_lint_pass(box minmax::MinMaxPass);
235 reg.register_late_lint_pass(box open_options::NonSensical);
236 reg.register_late_lint_pass(box zero_div_zero::Pass);
237 reg.register_late_lint_pass(box mutex_atomic::MutexAtomic);
238 reg.register_late_lint_pass(box needless_update::Pass);
239 reg.register_late_lint_pass(box needless_borrow::NeedlessBorrow);
240 reg.register_late_lint_pass(box no_effect::Pass);
241 reg.register_late_lint_pass(box map_clone::Pass);
242 reg.register_late_lint_pass(box temporary_assignment::Pass);
243 reg.register_late_lint_pass(box transmute::Transmute);
244 reg.register_late_lint_pass(
245 box cyclomatic_complexity::CyclomaticComplexity::new(conf.cyclomatic_complexity_threshold)
247 reg.register_late_lint_pass(box escape::Pass{too_large_for_stack: conf.too_large_for_stack});
248 reg.register_early_lint_pass(box misc_early::MiscEarly);
249 reg.register_late_lint_pass(box array_indexing::ArrayIndexing);
250 reg.register_late_lint_pass(box panic::Pass);
251 reg.register_late_lint_pass(box strings::StringLitAsBytes);
252 reg.register_late_lint_pass(box derive::Derive);
253 reg.register_late_lint_pass(box types::CharLitAsU8);
254 reg.register_late_lint_pass(box print::Pass);
255 reg.register_late_lint_pass(box vec::Pass);
256 reg.register_early_lint_pass(box non_expressive_names::NonExpressiveNames {
257 max_single_char_names: conf.max_single_char_names,
259 reg.register_late_lint_pass(box drop_ref::Pass);
260 reg.register_late_lint_pass(box types::AbsurdExtremeComparisons);
261 reg.register_late_lint_pass(box types::InvalidUpcastComparisons);
262 reg.register_late_lint_pass(box regex::Pass::default());
263 reg.register_late_lint_pass(box copies::CopyAndPaste);
264 reg.register_late_lint_pass(box format::Pass);
265 reg.register_early_lint_pass(box formatting::Formatting);
266 reg.register_late_lint_pass(box swap::Swap);
267 reg.register_early_lint_pass(box if_not_else::IfNotElse);
268 reg.register_late_lint_pass(box overflow_check_conditional::OverflowCheckConditional);
269 reg.register_late_lint_pass(box unused_label::UnusedLabel);
270 reg.register_late_lint_pass(box new_without_default::NewWithoutDefault);
271 reg.register_late_lint_pass(box blacklisted_name::BlackListedName::new(conf.blacklisted_names));
272 reg.register_late_lint_pass(box functions::Functions::new(conf.too_many_arguments_threshold));
273 reg.register_early_lint_pass(box doc::Doc::new(conf.doc_valid_idents));
274 reg.register_late_lint_pass(box neg_multiply::NegMultiply);
275 reg.register_early_lint_pass(box unsafe_removed_from_name::UnsafeNameRemoval);
276 reg.register_late_lint_pass(box mem_forget::MemForget);
277 reg.register_late_lint_pass(box arithmetic::Arithmetic::default());
278 reg.register_late_lint_pass(box assign_ops::AssignOps);
279 reg.register_late_lint_pass(box let_if_seq::LetIfSeq);
280 reg.register_late_lint_pass(box eval_order_dependence::EvalOrderDependence);
281 reg.register_late_lint_pass(box missing_doc::MissingDoc::new());
282 reg.register_late_lint_pass(box ok_if_let::Pass);
283 reg.register_late_lint_pass(box if_let_redundant_pattern_matching::Pass);
284 reg.register_late_lint_pass(box partialeq_ne_impl::Pass);
285 reg.register_early_lint_pass(box reference::Pass);
287 reg.register_lint_group("clippy_restrictions", vec![
288 arithmetic::FLOAT_ARITHMETIC,
289 arithmetic::INTEGER_ARITHMETIC,
290 array_indexing::INDEXING_SLICING,
291 assign_ops::ASSIGN_OPS,
294 reg.register_lint_group("clippy_pedantic", vec![
295 booleans::NONMINIMAL_BOOL,
296 enum_glob_use::ENUM_GLOB_USE,
297 enum_variants::PUB_ENUM_VARIANT_NAMES,
298 enum_variants::STUTTER,
299 if_not_else::IF_NOT_ELSE,
300 items_after_statements::ITEMS_AFTER_STATEMENTS,
301 matches::SINGLE_MATCH_ELSE,
302 mem_forget::MEM_FORGET,
304 methods::OPTION_MAP_UNWRAP_OR,
305 methods::OPTION_MAP_UNWRAP_OR_ELSE,
306 methods::OPTION_UNWRAP_USED,
307 methods::RESULT_UNWRAP_USED,
308 methods::WRONG_PUB_SELF_CONVENTION,
309 misc::USED_UNDERSCORE_BINDING,
310 misc_early::UNSEPARATED_LITERAL_SUFFIX,
311 missing_doc::MISSING_DOCS_IN_PRIVATE_ITEMS,
313 mutex_atomic::MUTEX_INTEGER,
314 non_expressive_names::SIMILAR_NAMES,
317 shadow::SHADOW_REUSE,
319 shadow::SHADOW_UNRELATED,
321 strings::STRING_ADD_ASSIGN,
322 types::CAST_POSSIBLE_TRUNCATION,
323 types::CAST_POSSIBLE_WRAP,
324 types::CAST_PRECISION_LOSS,
325 types::CAST_SIGN_LOSS,
326 types::INVALID_UPCAST_COMPARISONS,
327 unicode::NON_ASCII_LITERAL,
328 unicode::UNICODE_NOT_NFC,
331 reg.register_lint_group("clippy_internal", vec![
332 utils::internal_lints::CLIPPY_LINTS_INTERNAL,
333 utils::internal_lints::LINT_WITHOUT_LINT_PASS,
336 reg.register_lint_group("clippy", vec![
337 approx_const::APPROX_CONSTANT,
338 array_indexing::OUT_OF_BOUNDS_INDEXING,
339 assign_ops::ASSIGN_OP_PATTERN,
340 assign_ops::MISREFACTORED_ASSIGN_OP,
341 attrs::DEPRECATED_SEMVER,
342 attrs::INLINE_ALWAYS,
343 attrs::USELESS_ATTRIBUTE,
344 bit_mask::BAD_BIT_MASK,
345 bit_mask::INEFFECTIVE_BIT_MASK,
346 blacklisted_name::BLACKLISTED_NAME,
347 block_in_if_condition::BLOCK_IN_IF_CONDITION_EXPR,
348 block_in_if_condition::BLOCK_IN_IF_CONDITION_STMT,
350 collapsible_if::COLLAPSIBLE_IF,
351 copies::IF_SAME_THEN_ELSE,
352 copies::IFS_SAME_COND,
353 copies::MATCH_SAME_ARMS,
354 cyclomatic_complexity::CYCLOMATIC_COMPLEXITY,
355 derive::DERIVE_HASH_XOR_EQ,
356 derive::EXPL_IMPL_CLONE_ON_COPY,
360 enum_clike::ENUM_CLIKE_UNPORTABLE_VARIANT,
361 enum_variants::ENUM_VARIANT_NAMES,
362 enum_variants::MODULE_INCEPTION,
365 eta_reduction::REDUNDANT_CLOSURE,
366 eval_order_dependence::DIVERGING_SUB_EXPRESSION,
367 eval_order_dependence::EVAL_ORDER_DEPENDENCE,
368 format::USELESS_FORMAT,
369 formatting::SUSPICIOUS_ASSIGNMENT_FORMATTING,
370 formatting::SUSPICIOUS_ELSE_FORMATTING,
371 functions::NOT_UNSAFE_PTR_ARG_DEREF,
372 functions::TOO_MANY_ARGUMENTS,
373 identity_op::IDENTITY_OP,
374 if_let_redundant_pattern_matching::IF_LET_REDUNDANT_PATTERN_MATCHING,
375 len_zero::LEN_WITHOUT_IS_EMPTY,
377 let_if_seq::USELESS_LET_IF_SEQ,
378 lifetimes::NEEDLESS_LIFETIMES,
379 lifetimes::UNUSED_LIFETIMES,
381 loops::EXPLICIT_COUNTER_LOOP,
382 loops::EXPLICIT_INTO_ITER_LOOP,
383 loops::EXPLICIT_ITER_LOOP,
385 loops::FOR_LOOP_OVER_OPTION,
386 loops::FOR_LOOP_OVER_RESULT,
387 loops::ITER_NEXT_LOOP,
388 loops::NEEDLESS_RANGE_LOOP,
389 loops::REVERSE_RANGE_LOOP,
390 loops::UNUSED_COLLECT,
391 loops::WHILE_LET_LOOP,
392 loops::WHILE_LET_ON_ITERATOR,
393 map_clone::MAP_CLONE,
395 matches::MATCH_OVERLAPPING_ARM,
396 matches::MATCH_REF_PATS,
397 matches::SINGLE_MATCH,
398 methods::CHARS_NEXT_CMP,
399 methods::CLONE_DOUBLE_REF,
400 methods::CLONE_ON_COPY,
401 methods::EXTEND_FROM_SLICE,
402 methods::FILTER_NEXT,
405 methods::ITER_SKIP_NEXT,
406 methods::NEW_RET_NO_SELF,
408 methods::OR_FUN_CALL,
409 methods::SEARCH_IS_SOME,
410 methods::SHOULD_IMPLEMENT_TRAIT,
411 methods::SINGLE_CHAR_PATTERN,
412 methods::STRING_EXTEND_CHARS,
413 methods::TEMPORARY_CSTRING_AS_PTR,
414 methods::WRONG_SELF_CONVENTION,
420 misc::REDUNDANT_PATTERN,
421 misc::TOPLEVEL_REF_ARG,
422 misc_early::BUILTIN_TYPE_SHADOW,
423 misc_early::DOUBLE_NEG,
424 misc_early::DUPLICATE_UNDERSCORE_ARGUMENT,
425 misc_early::MIXED_CASE_HEX_LITERALS,
426 misc_early::REDUNDANT_CLOSURE_CALL,
427 misc_early::UNNEEDED_FIELD_PATTERN,
428 misc_early::ZERO_PREFIXED_LITERAL,
429 mut_reference::UNNECESSARY_MUT_PASSED,
430 mutex_atomic::MUTEX_ATOMIC,
431 needless_bool::BOOL_COMPARISON,
432 needless_bool::NEEDLESS_BOOL,
433 needless_borrow::NEEDLESS_BORROW,
434 needless_update::NEEDLESS_UPDATE,
435 neg_multiply::NEG_MULTIPLY,
436 new_without_default::NEW_WITHOUT_DEFAULT,
437 new_without_default::NEW_WITHOUT_DEFAULT_DERIVE,
438 no_effect::NO_EFFECT,
439 no_effect::UNNECESSARY_OPERATION,
440 non_expressive_names::MANY_SINGLE_CHAR_NAMES,
441 ok_if_let::IF_LET_SOME_RESULT,
442 open_options::NONSENSICAL_OPEN_OPTIONS,
443 overflow_check_conditional::OVERFLOW_CHECK_CONDITIONAL,
445 partialeq_ne_impl::PARTIALEQ_NE_IMPL,
446 precedence::PRECEDENCE,
447 print::PRINT_WITH_NEWLINE,
450 ranges::RANGE_STEP_BY_ZERO,
451 ranges::RANGE_ZIP_WITH_LEN,
452 reference::DEREF_ADDROF,
453 regex::INVALID_REGEX,
455 regex::TRIVIAL_REGEX,
456 returns::LET_AND_RETURN,
457 returns::NEEDLESS_RETURN,
458 serde::SERDE_API_MISUSE,
459 strings::STRING_LIT_AS_BYTES,
460 swap::ALMOST_SWAPPED,
462 temporary_assignment::TEMPORARY_ASSIGNMENT,
463 transmute::CROSSPOINTER_TRANSMUTE,
464 transmute::TRANSMUTE_PTR_TO_REF,
465 transmute::USELESS_TRANSMUTE,
466 transmute::WRONG_TRANSMUTE,
467 types::ABSURD_EXTREME_COMPARISONS,
469 types::CHAR_LIT_AS_U8,
470 types::LET_UNIT_VALUE,
472 types::TYPE_COMPLEXITY,
474 unicode::ZERO_WIDTH_SPACE,
475 unsafe_removed_from_name::UNSAFE_REMOVED_FROM_NAME,
476 unused_label::UNUSED_LABEL,
478 zero_div_zero::ZERO_DIVIDED_BY_ZERO,
482 // only exists to let the dogfood integration test works.
483 // Don't run clippy as an executable directly
484 #[allow(dead_code, print_stdout)]
486 panic!("Please use the cargo-clippy executable");