Run feature-gating on the final AST passed to the compiler.

This ensures we catch everything; previously, an unknown attribute
inserted by #[cfg_attr(...)] in a macro expansion would not be detected.
This commit is contained in:
Huon Wilson 2015-03-04 18:05:38 +11:00
parent 10426f69da
commit b5c6ab20b7
3 changed files with 50 additions and 9 deletions

View File

@ -493,12 +493,16 @@ pub fn phase_2_configure_and_expand(sess: &Session,
} }
); );
// Needs to go *after* expansion to be able to check the results of macro expansion. // Needs to go *after* expansion to be able to check the results
time(time_passes, "complete gated feature checking", (), |_| { // of macro expansion. This runs before #[cfg] to try to catch as
// much as possible (e.g. help the programmer avoid platform
// specific differences)
time(time_passes, "complete gated feature checking 1", (), |_| {
let features = let features =
syntax::feature_gate::check_crate(sess.codemap(), syntax::feature_gate::check_crate(sess.codemap(),
&sess.parse_sess.span_diagnostic, &sess.parse_sess.span_diagnostic,
&krate); &krate,
true);
*sess.features.borrow_mut() = features; *sess.features.borrow_mut() = features;
sess.abort_if_errors(); sess.abort_if_errors();
}); });
@ -521,6 +525,19 @@ pub fn phase_2_configure_and_expand(sess: &Session,
time(time_passes, "checking that all macro invocations are gone", &krate, |krate| time(time_passes, "checking that all macro invocations are gone", &krate, |krate|
syntax::ext::expand::check_for_macros(&sess.parse_sess, krate)); syntax::ext::expand::check_for_macros(&sess.parse_sess, krate));
// One final feature gating of the true AST that gets compiled
// later, to make sure we've got everything (e.g. configuration
// can insert new attributes via `cfg_attr`)
time(time_passes, "complete gated feature checking 2", (), |_| {
let features =
syntax::feature_gate::check_crate(sess.codemap(),
&sess.parse_sess.span_diagnostic,
&krate,
false);
*sess.features.borrow_mut() = features;
sess.abort_if_errors();
});
Some(krate) Some(krate)
} }

View File

@ -349,6 +349,7 @@ struct Context<'a> {
features: Vec<&'static str>, features: Vec<&'static str>,
span_handler: &'a SpanHandler, span_handler: &'a SpanHandler,
cm: &'a CodeMap, cm: &'a CodeMap,
do_warnings: bool,
} }
impl<'a> Context<'a> { impl<'a> Context<'a> {
@ -361,7 +362,7 @@ impl<'a> Context<'a> {
} }
fn warn_feature(&self, feature: &str, span: Span, explain: &str) { fn warn_feature(&self, feature: &str, span: Span, explain: &str) {
if !self.has_feature(feature) { if !self.has_feature(feature) && self.do_warnings {
emit_feature_warn(self.span_handler, feature, span, explain); emit_feature_warn(self.span_handler, feature, span, explain);
} }
} }
@ -700,6 +701,7 @@ impl<'a, 'v> Visitor<'v> for PostExpansionVisitor<'a> {
} }
fn check_crate_inner<F>(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::Crate, fn check_crate_inner<F>(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::Crate,
do_warnings: bool,
check: F) check: F)
-> Features -> Features
where F: FnOnce(&mut Context, &ast::Crate) where F: FnOnce(&mut Context, &ast::Crate)
@ -707,6 +709,7 @@ fn check_crate_inner<F>(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::C
let mut cx = Context { let mut cx = Context {
features: Vec::new(), features: Vec::new(),
span_handler: span_handler, span_handler: span_handler,
do_warnings: do_warnings,
cm: cm, cm: cm,
}; };
@ -786,13 +789,14 @@ fn check_crate_inner<F>(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::C
pub fn check_crate_macros(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::Crate) pub fn check_crate_macros(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::Crate)
-> Features { -> Features {
check_crate_inner(cm, span_handler, krate, check_crate_inner(cm, span_handler, krate, true,
|ctx, krate| visit::walk_crate(&mut MacroVisitor { context: ctx }, krate)) |ctx, krate| visit::walk_crate(&mut MacroVisitor { context: ctx }, krate))
} }
pub fn check_crate(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::Crate) pub fn check_crate(cm: &CodeMap, span_handler: &SpanHandler, krate: &ast::Crate,
-> Features { do_warnings: bool) -> Features
check_crate_inner(cm, span_handler, krate, {
check_crate_inner(cm, span_handler, krate, do_warnings,
|ctx, krate| visit::walk_crate(&mut PostExpansionVisitor { context: ctx }, |ctx, krate| visit::walk_crate(&mut PostExpansionVisitor { context: ctx },
krate)) krate))
} }

View File

@ -0,0 +1,20 @@
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
macro_rules! foo {
() => {
#[cfg_attr(all(), unknown)] //~ ERROR `unknown` is currently unknown
fn foo() {}
}
}
foo!();
fn main() {}