84b060ce29
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.
52 lines
1.6 KiB
Rust
52 lines
1.6 KiB
Rust
// 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.
|
|
|
|
// force-host
|
|
|
|
#![feature(plugin_registrar)]
|
|
#![feature(box_syntax)]
|
|
|
|
extern crate syntax;
|
|
extern crate rustc;
|
|
|
|
use std::borrow::ToOwned;
|
|
use syntax::ast;
|
|
use syntax::codemap::Span;
|
|
use syntax::ext::build::AstBuilder;
|
|
use syntax::ext::base::{TTMacroExpander, ExtCtxt, MacResult, MacEager, NormalTT};
|
|
use syntax::parse::token;
|
|
use syntax::print::pprust;
|
|
use syntax::ptr::P;
|
|
use rustc::plugin::Registry;
|
|
|
|
struct Expander {
|
|
args: Vec<P<ast::MetaItem>>,
|
|
}
|
|
|
|
impl TTMacroExpander for Expander {
|
|
fn expand<'cx>(&self,
|
|
ecx: &'cx mut ExtCtxt,
|
|
sp: Span,
|
|
_: &[ast::TokenTree]) -> Box<MacResult+'cx> {
|
|
let args = self.args.iter().map(|i| pprust::meta_item_to_string(&*i))
|
|
.collect::<Vec<_>>().connect(", ");
|
|
let interned = token::intern_and_get_ident(&args[..]);
|
|
MacEager::expr(ecx.expr_str(sp, interned))
|
|
}
|
|
}
|
|
|
|
#[plugin_registrar]
|
|
pub fn plugin_registrar(reg: &mut Registry) {
|
|
let args = reg.args().clone();
|
|
reg.register_syntax_extension(token::intern("plugin_args"),
|
|
// FIXME (#22405): Replace `Box::new` with `box` here when/if possible.
|
|
NormalTT(Box::new(Expander { args: args, }), None, false));
|
|
}
|