Skip to content

Commit 86319e4

Browse files
committed
Auto merge of #41907 - est31:macro_unused, r=jseyfried
Add lint for unused macros Addresses parts of #34938, to add a lint for unused macros. We now output warnings by default when we encounter a macro that we didn't use for expansion. Issues to be resolved before this PR is ready for merge: - [x] fix the NodeId issue described above - [x] remove all unused macros from rustc and the libraries or set `#[allow(unused_macros)]` next to them if they should be kept for some reason. This is needed for successful boostrap and bors to accept the PR. -> #41934 - [x] ~~implement the full extent of #34938, that means the macro match arm checking as well.~~ *let's not do this for now*
2 parents 0f68728 + 6dbd706 commit 86319e4

32 files changed

+138
-19
lines changed

src/libcore/num/wrapping.rs

+1
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ use super::Wrapping;
1212

1313
use ops::*;
1414

15+
#[allow(unused_macros)]
1516
macro_rules! sh_impl_signed {
1617
($t:ident, $f:ident) => (
1718
#[stable(feature = "rust1", since = "1.0.0")]

src/libcore/ops.rs

+1
Original file line numberDiff line numberDiff line change
@@ -799,6 +799,7 @@ macro_rules! neg_impl_numeric {
799799
($($t:ty)*) => { neg_impl_core!{ x => -x, $($t)*} }
800800
}
801801

802+
#[allow(unused_macros)]
802803
macro_rules! neg_impl_unsigned {
803804
($($t:ty)*) => {
804805
neg_impl_core!{ x => {

src/librustc/lint/builtin.rs

+7
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,12 @@ declare_lint! {
7676
"detects unreachable patterns"
7777
}
7878

79+
declare_lint! {
80+
pub UNUSED_MACROS,
81+
Warn,
82+
"detects macros that were not used"
83+
}
84+
7985
declare_lint! {
8086
pub WARNINGS,
8187
Warn,
@@ -259,6 +265,7 @@ impl LintPass for HardwiredLints {
259265
DEAD_CODE,
260266
UNREACHABLE_CODE,
261267
UNREACHABLE_PATTERNS,
268+
UNUSED_MACROS,
262269
WARNINGS,
263270
UNUSED_FEATURES,
264271
STABLE_FEATURES,

src/librustc/lint/context.rs

+8
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@ use hir;
4949
use hir::def_id::LOCAL_CRATE;
5050
use hir::intravisit as hir_visit;
5151
use syntax::visit as ast_visit;
52+
use syntax::tokenstream::ThinTokenStream;
5253

5354
/// Information about the registered lints.
5455
///
@@ -1125,6 +1126,13 @@ impl<'a> ast_visit::Visitor<'a> for EarlyContext<'a> {
11251126
fn visit_attribute(&mut self, attr: &'a ast::Attribute) {
11261127
run_lints!(self, check_attribute, early_passes, attr);
11271128
}
1129+
1130+
fn visit_mac_def(&mut self, _mac: &'a ThinTokenStream, id: ast::NodeId) {
1131+
let lints = self.sess.lints.borrow_mut().take(id);
1132+
for early_lint in lints {
1133+
self.early_lint(&early_lint);
1134+
}
1135+
}
11281136
}
11291137

11301138
enum CheckLintNameResult {

src/librustc/util/common.rs

+1
Original file line numberDiff line numberDiff line change
@@ -116,6 +116,7 @@ pub fn record_time<T, F>(accu: &Cell<Duration>, f: F) -> T where
116116
}
117117

118118
// Like std::macros::try!, but for Option<>.
119+
#[cfg(unix)]
119120
macro_rules! option_try(
120121
($e:expr) => (match $e { Some(e) => e, None => return None })
121122
);

src/librustc_driver/driver.rs

+2
Original file line numberDiff line numberDiff line change
@@ -699,6 +699,8 @@ pub fn phase_2_configure_and_expand<F>(sess: &Session,
699699

700700
let krate = ecx.monotonic_expander().expand_crate(krate);
701701

702+
ecx.check_unused_macros();
703+
702704
let mut missing_fragment_specifiers: Vec<_> =
703705
ecx.parse_sess.missing_fragment_specifiers.borrow().iter().cloned().collect();
704706
missing_fragment_specifiers.sort();

src/librustc_incremental/persist/preds/compress/test_macro.rs

-11
Original file line numberDiff line numberDiff line change
@@ -37,14 +37,3 @@ macro_rules! graph {
3737
}
3838
}
3939
}
40-
41-
macro_rules! set {
42-
($( $value:expr ),*) => {
43-
{
44-
use $crate::rustc_data_structures::fx::FxHashSet;
45-
let mut set = FxHashSet();
46-
$(set.insert($value);)*
47-
set
48-
}
49-
}
50-
}

src/librustc_lint/lib.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -171,7 +171,8 @@ pub fn register_builtins(store: &mut lint::LintStore, sess: Option<&Session>) {
171171
UNUSED_MUST_USE,
172172
UNUSED_UNSAFE,
173173
PATH_STATEMENTS,
174-
UNUSED_ATTRIBUTES);
174+
UNUSED_ATTRIBUTES,
175+
UNUSED_MACROS);
175176

176177
// Guidelines for creating a future incompatibility lint:
177178
//

src/librustc_plugin/registry.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,8 @@ impl<'a> Registry<'a> {
103103
}
104104
self.syntax_exts.push((name, match extension {
105105
NormalTT(ext, _, allow_internal_unstable) => {
106-
NormalTT(ext, Some(self.krate_span), allow_internal_unstable)
106+
let nid = ast::CRATE_NODE_ID;
107+
NormalTT(ext, Some((nid, self.krate_span)), allow_internal_unstable)
107108
}
108109
IdentTT(ext, _, allow_internal_unstable) => {
109110
IdentTT(ext, Some(self.krate_span), allow_internal_unstable)

src/librustc_resolve/lib.rs

+5
Original file line numberDiff line numberDiff line change
@@ -1187,6 +1187,10 @@ pub struct Resolver<'a> {
11871187
pub whitelisted_legacy_custom_derives: Vec<Name>,
11881188
pub found_unresolved_macro: bool,
11891189

1190+
// List of crate local macros that we need to warn about as being unused.
1191+
// Right now this only includes macro_rules! macros.
1192+
unused_macros: FxHashSet<DefId>,
1193+
11901194
// Maps the `Mark` of an expansion to its containing module or block.
11911195
invocations: FxHashMap<Mark, &'a InvocationData<'a>>,
11921196

@@ -1392,6 +1396,7 @@ impl<'a> Resolver<'a> {
13921396
potentially_unused_imports: Vec::new(),
13931397
struct_constructors: DefIdMap(),
13941398
found_unresolved_macro: false,
1399+
unused_macros: FxHashSet(),
13951400
}
13961401
}
13971402

src/librustc_resolve/macros.rs

+24-2
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ use resolve_imports::ImportResolver;
1616
use rustc::hir::def_id::{DefId, BUILTIN_MACROS_CRATE, CRATE_DEF_INDEX, DefIndex};
1717
use rustc::hir::def::{Def, Export};
1818
use rustc::hir::map::{self, DefCollector};
19-
use rustc::ty;
19+
use rustc::{ty, lint};
2020
use syntax::ast::{self, Name, Ident};
2121
use syntax::attr::{self, HasAttrs};
2222
use syntax::errors::DiagnosticBuilder;
@@ -291,12 +291,32 @@ impl<'a> base::Resolver for Resolver<'a> {
291291
},
292292
};
293293
self.macro_defs.insert(invoc.expansion_data.mark, def.def_id());
294+
self.unused_macros.remove(&def.def_id());
294295
Ok(Some(self.get_macro(def)))
295296
}
296297

297298
fn resolve_macro(&mut self, scope: Mark, path: &ast::Path, kind: MacroKind, force: bool)
298299
-> Result<Rc<SyntaxExtension>, Determinacy> {
299-
self.resolve_macro_to_def(scope, path, kind, force).map(|def| self.get_macro(def))
300+
self.resolve_macro_to_def(scope, path, kind, force).map(|def| {
301+
self.unused_macros.remove(&def.def_id());
302+
self.get_macro(def)
303+
})
304+
}
305+
306+
fn check_unused_macros(&self) {
307+
for did in self.unused_macros.iter() {
308+
let id_span = match *self.macro_map[did] {
309+
SyntaxExtension::NormalTT(_, isp, _) => isp,
310+
_ => None,
311+
};
312+
if let Some((id, span)) = id_span {
313+
let lint = lint::builtin::UNUSED_MACROS;
314+
let msg = "unused macro definition".to_string();
315+
self.session.add_lint(lint, id, span, msg);
316+
} else {
317+
bug!("attempted to create unused macro error, but span not available");
318+
}
319+
}
300320
}
301321
}
302322

@@ -687,6 +707,8 @@ impl<'a> Resolver<'a> {
687707
if attr::contains_name(&item.attrs, "macro_export") {
688708
let def = Def::Macro(def_id, MacroKind::Bang);
689709
self.macro_exports.push(Export { name: ident.name, def: def, span: item.span });
710+
} else {
711+
self.unused_macros.insert(def_id);
690712
}
691713
}
692714

src/libsyntax/ext/base.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -535,7 +535,7 @@ pub enum SyntaxExtension {
535535
///
536536
/// The `bool` dictates whether the contents of the macro can
537537
/// directly use `#[unstable]` things (true == yes).
538-
NormalTT(Box<TTMacroExpander>, Option<Span>, bool),
538+
NormalTT(Box<TTMacroExpander>, Option<(ast::NodeId, Span)>, bool),
539539

540540
/// A function-like syntax extension that has an extra ident before
541541
/// the block.
@@ -589,6 +589,7 @@ pub trait Resolver {
589589
-> Result<Option<Rc<SyntaxExtension>>, Determinacy>;
590590
fn resolve_macro(&mut self, scope: Mark, path: &ast::Path, kind: MacroKind, force: bool)
591591
-> Result<Rc<SyntaxExtension>, Determinacy>;
592+
fn check_unused_macros(&self);
592593
}
593594

594595
#[derive(Copy, Clone, Debug)]
@@ -618,6 +619,7 @@ impl Resolver for DummyResolver {
618619
_force: bool) -> Result<Rc<SyntaxExtension>, Determinacy> {
619620
Err(Determinacy::Determined)
620621
}
622+
fn check_unused_macros(&self) {}
621623
}
622624

623625
#[derive(Clone)]
@@ -800,6 +802,10 @@ impl<'a> ExtCtxt<'a> {
800802
pub fn name_of(&self, st: &str) -> ast::Name {
801803
Symbol::intern(st)
802804
}
805+
806+
pub fn check_unused_macros(&self) {
807+
self.resolver.check_unused_macros();
808+
}
803809
}
804810

805811
/// Extract a string literal from the macro expanded version of `expr`,

src/libsyntax/ext/expand.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -469,7 +469,7 @@ impl<'a, 'b> MacroExpander<'a, 'b> {
469469
call_site: span,
470470
callee: NameAndSpan {
471471
format: MacroBang(Symbol::intern(&format!("{}", path))),
472-
span: exp_span,
472+
span: exp_span.map(|(_, s)| s),
473473
allow_internal_unstable: allow_internal_unstable,
474474
},
475475
});

src/libsyntax/ext/tt/macro_rules.rs

+3-1
Original file line numberDiff line numberDiff line change
@@ -252,7 +252,9 @@ pub fn compile(sess: &ParseSess, features: &RefCell<Features>, def: &ast::Item)
252252
valid: valid,
253253
});
254254

255-
NormalTT(exp, Some(def.span), attr::contains_name(&def.attrs, "allow_internal_unstable"))
255+
NormalTT(exp,
256+
Some((def.id, def.span)),
257+
attr::contains_name(&def.attrs, "allow_internal_unstable"))
256258
}
257259

258260
fn check_lhs_nt_follows(sess: &ParseSess,

src/libsyntax/visit.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ use abi::Abi;
2727
use ast::*;
2828
use syntax_pos::Span;
2929
use codemap::Spanned;
30+
use tokenstream::ThinTokenStream;
3031

3132
#[derive(Copy, Clone, PartialEq, Eq)]
3233
pub enum FnKind<'a> {
@@ -112,6 +113,9 @@ pub trait Visitor<'ast>: Sized {
112113
// definition in your trait impl:
113114
// visit::walk_mac(self, _mac)
114115
}
116+
fn visit_mac_def(&mut self, _mac: &'ast ThinTokenStream, _id: NodeId) {
117+
// Nothing to do
118+
}
115119
fn visit_path(&mut self, path: &'ast Path, _id: NodeId) {
116120
walk_path(self, path)
117121
}
@@ -290,7 +294,7 @@ pub fn walk_item<'a, V: Visitor<'a>>(visitor: &mut V, item: &'a Item) {
290294
walk_list!(visitor, visit_trait_item, methods);
291295
}
292296
ItemKind::Mac(ref mac) => visitor.visit_mac(mac),
293-
ItemKind::MacroDef(..) => {},
297+
ItemKind::MacroDef(ref ts) => visitor.visit_mac_def(ts, item.id),
294298
}
295299
walk_list!(visitor, visit_attribute, &item.attrs);
296300
}

src/test/compile-fail-fulldeps/proc-macro/resolve-error.rs

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414
// aux-build:bang_proc_macro.rs
1515

1616
#![feature(proc_macro)]
17+
#![allow(unused_macros)]
1718

1819
#[macro_use]
1920
extern crate derive_foo;

src/test/compile-fail/feature-gate-allow-internal-unstable-nested-macro.rs

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010

1111
// gate-test-allow_internal_unstable
1212

13+
#![allow(unused_macros)]
14+
1315
macro_rules! bar {
1416
() => {
1517
// more layers don't help:

src/test/compile-fail/feature-gate-allow-internal-unstable.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
#[allow_internal_unstable] //~ ERROR allow_internal_unstable side-steps
1214
macro_rules! foo {
1315
() => {}

src/test/compile-fail/invalid-macro-matcher.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
macro_rules! invalid {
1214
_ => (); //~ ERROR invalid macro matcher
1315
}

src/test/compile-fail/issue-21356.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
macro_rules! test { ($wrong:t_ty ..) => () }
1214
//~^ ERROR: invalid fragment specifier `t_ty`
1315

src/test/compile-fail/issue-39388.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
macro_rules! assign {
1214
(($($a:tt)*) = ($($b:tt))*) => { //~ ERROR expected `*` or `+`
1315
$($a)* = $($b)*

src/test/compile-fail/issue-39404.rs

+1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
// except according to those terms.
1010

1111
#![deny(missing_fragment_specifier)] //~ NOTE lint level defined here
12+
#![allow(unused_macros)]
1213

1314
macro_rules! m { ($i) => {} }
1415
//~^ ERROR missing fragment specifier

src/test/compile-fail/issue-5067.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
macro_rules! foo {
1214
( $()* ) => {};
1315
//~^ ERROR repetition matches empty token tree

src/test/compile-fail/macro-expansion-tests.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
mod macros_cant_escape_fns {
1214
fn f() {
1315
macro_rules! m { () => { 3 + 4 } }

src/test/compile-fail/macro-follow.rs

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010
//
1111
// Check the macro follow sets (see corresponding rpass test).
1212

13+
#![allow(unused_macros)]
14+
1315
// FOLLOW(pat) = {FatArrow, Comma, Eq, Or, Ident(if), Ident(in)}
1416
macro_rules! follow_pat {
1517
($p:pat ()) => {}; //~ERROR `$p:pat` is followed by `(`

src/test/compile-fail/macro-followed-by-seq-bad.rs

+2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@
1111
// Regression test for issue #25436: check that things which can be
1212
// followed by any token also permit X* to come afterwards.
1313

14+
#![allow(unused_macros)]
15+
1416
macro_rules! foo {
1517
( $a:expr $($b:tt)* ) => { }; //~ ERROR not allowed for `expr` fragments
1618
( $a:ty $($b:tt)* ) => { }; //~ ERROR not allowed for `ty` fragments

src/test/compile-fail/macro-input-future-proofing.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
macro_rules! errors_everywhere {
1214
($ty:ty <) => (); //~ ERROR `$ty:ty` is followed by `<`, which is not allowed for `ty`
1315
($ty:ty < foo ,) => (); //~ ERROR `$ty:ty` is followed by `<`, which is not allowed for `ty`

src/test/compile-fail/macro-shadowing.rs

+2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@
1010

1111
// aux-build:two_macros.rs
1212

13+
#![allow(unused_macros)]
14+
1315
macro_rules! foo { () => {} }
1416
macro_rules! macro_one { () => {} }
1517
#[macro_use(macro_two)] extern crate two_macros;

src/test/compile-fail/unused-macro-with-bad-frag-spec.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
// Issue #21370
1214

1315
macro_rules! test {

src/test/compile-fail/unused-macro-with-follow-violation.rs

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11+
#![allow(unused_macros)]
12+
1113
macro_rules! test {
1214
($e:expr +) => () //~ ERROR not allowed for `expr` fragments
1315
}

0 commit comments

Comments
 (0)