2014-05-24 16:16:10 -07:00
|
|
|
// Copyright 2012-2013 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.
|
|
|
|
|
2014-05-26 14:48:54 -07:00
|
|
|
//! Used by plugin crates to tell `rustc` about the plugins they provide.
|
|
|
|
|
2015-11-22 22:14:09 +02:00
|
|
|
use rustc::lint::{EarlyLintPassObject, LateLintPassObject, LintId, Lint};
|
|
|
|
use rustc::session::Session;
|
2014-06-18 17:26:14 -07:00
|
|
|
|
2016-02-26 18:05:50 +02:00
|
|
|
use rustc::mir::transform::MirMapPass;
|
2016-02-05 09:35:54 +01:00
|
|
|
|
2016-09-06 17:57:58 +12:00
|
|
|
use syntax::ext::base::{SyntaxExtension, NamedSyntaxExtension, NormalTT, IdentTT};
|
2016-09-13 06:25:02 +00:00
|
|
|
use syntax::ext::base::MacroExpanderFn;
|
2016-11-16 08:21:52 +00:00
|
|
|
use syntax::symbol::Symbol;
|
2014-05-24 16:16:10 -07:00
|
|
|
use syntax::ast;
|
2015-05-06 22:08:36 +05:30
|
|
|
use syntax::feature_gate::AttributeType;
|
2016-06-21 18:08:13 -04:00
|
|
|
use syntax_pos::Span;
|
2014-05-24 16:16:10 -07:00
|
|
|
|
2014-07-21 15:27:59 +12:00
|
|
|
use std::collections::HashMap;
|
2015-04-08 12:52:58 -07:00
|
|
|
use std::borrow::ToOwned;
|
2014-07-21 15:27:59 +12:00
|
|
|
|
2014-05-26 14:48:54 -07:00
|
|
|
/// Structure used to register plugins.
|
|
|
|
///
|
|
|
|
/// A plugin registrar function takes an `&mut Registry` and should call
|
|
|
|
/// methods to register its plugins.
|
|
|
|
///
|
|
|
|
/// This struct has public fields and other methods for use by `rustc`
|
|
|
|
/// itself. They are not documented here, and plugin authors should
|
|
|
|
/// not use them.
|
2014-09-15 16:09:09 -07:00
|
|
|
pub struct Registry<'a> {
|
|
|
|
/// Compiler session. Useful if you want to emit diagnostic messages
|
|
|
|
/// from the plugin registrar.
|
|
|
|
pub sess: &'a Session,
|
|
|
|
|
2015-01-02 18:26:00 -08:00
|
|
|
#[doc(hidden)]
|
2016-08-19 18:58:14 -07:00
|
|
|
pub args_hidden: Option<Vec<ast::NestedMetaItem>>,
|
2015-01-02 18:26:00 -08:00
|
|
|
|
2014-05-24 16:16:10 -07:00
|
|
|
#[doc(hidden)]
|
|
|
|
pub krate_span: Span,
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub syntax_exts: Vec<NamedSyntaxExtension>,
|
2014-06-18 17:26:14 -07:00
|
|
|
|
|
|
|
#[doc(hidden)]
|
2015-09-15 11:35:25 +12:00
|
|
|
pub early_lint_passes: Vec<EarlyLintPassObject>,
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub late_lint_passes: Vec<LateLintPassObject>,
|
2014-07-21 15:27:59 +12:00
|
|
|
|
2016-02-05 09:35:54 +01:00
|
|
|
#[doc(hidden)]
|
2016-02-26 18:05:50 +02:00
|
|
|
pub mir_passes: Vec<Box<for<'pcx> MirMapPass<'pcx>>>,
|
2016-02-05 09:35:54 +01:00
|
|
|
|
2014-07-21 15:27:59 +12:00
|
|
|
#[doc(hidden)]
|
|
|
|
pub lint_groups: HashMap<&'static str, Vec<LintId>>,
|
2015-04-08 12:52:58 -07:00
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub llvm_passes: Vec<String>,
|
2015-05-06 22:08:36 +05:30
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub attributes: Vec<(String, AttributeType)>,
|
2014-05-24 16:16:10 -07:00
|
|
|
}
|
|
|
|
|
2014-09-15 16:09:09 -07:00
|
|
|
impl<'a> Registry<'a> {
|
2014-05-24 16:16:10 -07:00
|
|
|
#[doc(hidden)]
|
2016-09-04 03:22:56 +00:00
|
|
|
pub fn new(sess: &'a Session, krate_span: Span) -> Registry<'a> {
|
2014-05-24 16:16:10 -07:00
|
|
|
Registry {
|
2014-09-15 16:09:09 -07:00
|
|
|
sess: sess,
|
2015-01-02 18:26:00 -08:00
|
|
|
args_hidden: None,
|
2016-09-04 03:22:56 +00:00
|
|
|
krate_span: krate_span,
|
2016-10-29 22:54:04 +01:00
|
|
|
syntax_exts: vec![],
|
|
|
|
early_lint_passes: vec![],
|
|
|
|
late_lint_passes: vec![],
|
2014-07-21 15:27:59 +12:00
|
|
|
lint_groups: HashMap::new(),
|
2016-10-29 22:54:04 +01:00
|
|
|
llvm_passes: vec![],
|
|
|
|
attributes: vec![],
|
2016-02-05 09:35:54 +01:00
|
|
|
mir_passes: Vec::new(),
|
2014-05-24 16:16:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-06 13:56:38 -08:00
|
|
|
/// Get the plugin's arguments, if any.
|
2015-01-02 18:26:00 -08:00
|
|
|
///
|
2015-02-06 13:56:38 -08:00
|
|
|
/// These are specified inside the `plugin` crate attribute as
|
|
|
|
///
|
|
|
|
/// ```no_run
|
|
|
|
/// #![plugin(my_plugin_name(... args ...))]
|
|
|
|
/// ```
|
2016-05-03 17:20:48 +02:00
|
|
|
///
|
|
|
|
/// Returns empty slice in case the plugin was loaded
|
|
|
|
/// with `--extra-plugins`
|
2016-08-19 18:58:14 -07:00
|
|
|
pub fn args<'b>(&'b self) -> &'b [ast::NestedMetaItem] {
|
2016-05-03 17:20:48 +02:00
|
|
|
self.args_hidden.as_ref().map(|v| &v[..]).unwrap_or(&[])
|
2015-01-02 18:26:00 -08:00
|
|
|
}
|
|
|
|
|
2014-05-26 14:48:54 -07:00
|
|
|
/// Register a syntax extension of any kind.
|
|
|
|
///
|
|
|
|
/// This is the most general hook into `libsyntax`'s expansion behavior.
|
2014-05-24 16:16:10 -07:00
|
|
|
pub fn register_syntax_extension(&mut self, name: ast::Name, extension: SyntaxExtension) {
|
2016-11-17 14:04:20 +00:00
|
|
|
if name == "macro_rules" {
|
2016-09-25 23:38:22 +00:00
|
|
|
panic!("user-defined macros may not be named `macro_rules`");
|
|
|
|
}
|
2014-05-24 16:16:10 -07:00
|
|
|
self.syntax_exts.push((name, match extension {
|
Add #[allow_internal_unstable] to track stability for macros better.
Unstable items used in a macro expansion will now always trigger
stability warnings, *unless* the unstable items are directly inside a
macro marked with `#[allow_internal_unstable]`. IOW, the compiler warns
unless the span of the unstable item is a subspan of the definition of a
macro marked with that attribute.
E.g.
#[allow_internal_unstable]
macro_rules! foo {
($e: expr) => {{
$e;
unstable(); // no warning
only_called_by_foo!();
}}
}
macro_rules! only_called_by_foo {
() => { unstable() } // warning
}
foo!(unstable()) // warning
The unstable inside `foo` is fine, due to the attribute. But the
`unstable` inside `only_called_by_foo` is not, since that macro doesn't
have the attribute, and the `unstable` passed into `foo` is also not
fine since it isn't contained in the macro itself (that is, even though
it is only used directly in the macro).
In the process this makes the stability tracking much more precise,
e.g. previously `println!("{}", unstable())` got no warning, but now it
does. As such, this is a bug fix that may cause [breaking-change]s.
The attribute is definitely feature gated, since it explicitly allows
side-stepping the feature gating system.
2015-03-01 14:09:28 +11:00
|
|
|
NormalTT(ext, _, allow_internal_unstable) => {
|
|
|
|
NormalTT(ext, Some(self.krate_span), allow_internal_unstable)
|
|
|
|
}
|
|
|
|
IdentTT(ext, _, allow_internal_unstable) => {
|
|
|
|
IdentTT(ext, Some(self.krate_span), allow_internal_unstable)
|
|
|
|
}
|
2016-09-06 17:57:58 +12:00
|
|
|
_ => extension,
|
2014-05-24 16:16:10 -07:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
2014-05-26 14:48:54 -07:00
|
|
|
/// Register a macro of the usual kind.
|
|
|
|
///
|
|
|
|
/// This is a convenience wrapper for `register_syntax_extension`.
|
2014-09-10 20:59:26 -07:00
|
|
|
/// It builds for you a `NormalTT` that calls `expander`,
|
2014-05-26 14:48:54 -07:00
|
|
|
/// and also takes care of interning the macro's name.
|
2014-05-24 16:16:10 -07:00
|
|
|
pub fn register_macro(&mut self, name: &str, expander: MacroExpanderFn) {
|
2016-11-16 08:21:52 +00:00
|
|
|
self.register_syntax_extension(Symbol::intern(name),
|
Add #[allow_internal_unstable] to track stability for macros better.
Unstable items used in a macro expansion will now always trigger
stability warnings, *unless* the unstable items are directly inside a
macro marked with `#[allow_internal_unstable]`. IOW, the compiler warns
unless the span of the unstable item is a subspan of the definition of a
macro marked with that attribute.
E.g.
#[allow_internal_unstable]
macro_rules! foo {
($e: expr) => {{
$e;
unstable(); // no warning
only_called_by_foo!();
}}
}
macro_rules! only_called_by_foo {
() => { unstable() } // warning
}
foo!(unstable()) // warning
The unstable inside `foo` is fine, due to the attribute. But the
`unstable` inside `only_called_by_foo` is not, since that macro doesn't
have the attribute, and the `unstable` passed into `foo` is also not
fine since it isn't contained in the macro itself (that is, even though
it is only used directly in the macro).
In the process this makes the stability tracking much more precise,
e.g. previously `println!("{}", unstable())` got no warning, but now it
does. As such, this is a bug fix that may cause [breaking-change]s.
The attribute is definitely feature gated, since it explicitly allows
side-stepping the feature gating system.
2015-03-01 14:09:28 +11:00
|
|
|
NormalTT(Box::new(expander), None, false));
|
2014-05-24 16:16:10 -07:00
|
|
|
}
|
2014-06-18 17:26:14 -07:00
|
|
|
|
|
|
|
/// Register a compiler lint pass.
|
2015-09-15 11:35:25 +12:00
|
|
|
pub fn register_early_lint_pass(&mut self, lint_pass: EarlyLintPassObject) {
|
|
|
|
self.early_lint_passes.push(lint_pass);
|
2014-06-18 17:26:14 -07:00
|
|
|
}
|
2014-07-21 15:27:59 +12:00
|
|
|
|
2015-09-15 11:35:25 +12:00
|
|
|
/// Register a compiler lint pass.
|
|
|
|
pub fn register_late_lint_pass(&mut self, lint_pass: LateLintPassObject) {
|
|
|
|
self.late_lint_passes.push(lint_pass);
|
|
|
|
}
|
2014-07-21 15:27:59 +12:00
|
|
|
/// Register a lint group.
|
|
|
|
pub fn register_lint_group(&mut self, name: &'static str, to: Vec<&'static Lint>) {
|
2014-09-14 20:27:36 -07:00
|
|
|
self.lint_groups.insert(name, to.into_iter().map(|x| LintId::of(x)).collect());
|
2014-07-21 15:27:59 +12:00
|
|
|
}
|
2015-04-08 12:52:58 -07:00
|
|
|
|
2016-02-05 09:35:54 +01:00
|
|
|
/// Register a MIR pass
|
2016-02-26 18:05:50 +02:00
|
|
|
pub fn register_mir_pass(&mut self, pass: Box<for<'pcx> MirMapPass<'pcx>>) {
|
2016-02-05 09:35:54 +01:00
|
|
|
self.mir_passes.push(pass);
|
|
|
|
}
|
|
|
|
|
2015-04-08 12:52:58 -07:00
|
|
|
/// Register an LLVM pass.
|
|
|
|
///
|
|
|
|
/// Registration with LLVM itself is handled through static C++ objects with
|
|
|
|
/// constructors. This method simply adds a name to the list of passes to
|
|
|
|
/// execute.
|
|
|
|
pub fn register_llvm_pass(&mut self, name: &str) {
|
|
|
|
self.llvm_passes.push(name.to_owned());
|
|
|
|
}
|
2015-05-06 22:08:36 +05:30
|
|
|
|
2015-05-13 12:23:43 +05:30
|
|
|
/// Register an attribute with an attribute type.
|
2015-05-06 22:08:36 +05:30
|
|
|
///
|
2015-05-07 15:31:20 +05:30
|
|
|
/// Registered attributes will bypass the `custom_attribute` feature gate.
|
2015-05-06 22:08:36 +05:30
|
|
|
/// `Whitelisted` attributes will additionally not trigger the `unused_attribute`
|
2015-05-07 15:31:20 +05:30
|
|
|
/// lint. `CrateLevel` attributes will not be allowed on anything other than a crate.
|
2015-05-06 22:08:36 +05:30
|
|
|
pub fn register_attribute(&mut self, name: String, ty: AttributeType) {
|
|
|
|
self.attributes.push((name, ty));
|
|
|
|
}
|
2014-05-24 16:16:10 -07:00
|
|
|
}
|