// Copyright 2013-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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. // force-host #![feature(plugin_registrar, quote, rustc_private)] extern crate syntax; extern crate rustc; extern crate rustc_plugin; extern crate syntax_pos; use syntax::ast::{self, Item, MetaItem, ItemKind}; use syntax::codemap::DUMMY_SP; use syntax::ext::base::*; use syntax::ext::quote::rt::ToTokens; use syntax::parse::{self, token}; use syntax::ptr::P; use syntax::tokenstream::TokenTree; use syntax_pos::Span; use rustc_plugin::Registry; #[macro_export] macro_rules! exported_macro { () => (2) } macro_rules! unexported_macro { () => (3) } #[plugin_registrar] pub fn plugin_registrar(reg: &mut Registry) { reg.register_macro("make_a_1", expand_make_a_1); reg.register_macro("identity", expand_identity); reg.register_syntax_extension( token::intern("into_multi_foo"), // FIXME (#22405): Replace `Box::new` with `box` here when/if possible. MultiModifier(Box::new(expand_into_foo_multi))); reg.register_syntax_extension( token::intern("duplicate"), // FIXME (#22405): Replace `Box::new` with `box` here when/if possible. MultiDecorator(Box::new(expand_duplicate))); reg.register_syntax_extension( token::intern("caller"), // FIXME (#22405): Replace `Box::new` with `box` here when/if possible. MultiDecorator(Box::new(expand_caller))); } fn expand_make_a_1(cx: &mut ExtCtxt, sp: Span, tts: &[TokenTree]) -> Box { if !tts.is_empty() { cx.span_fatal(sp, "make_a_1 takes no arguments"); } MacEager::expr(quote_expr!(cx, 1)) } // See Issue #15750 fn expand_identity(cx: &mut ExtCtxt, _span: Span, tts: &[TokenTree]) -> Box { // Parse an expression and emit it unchanged. let mut parser = parse::new_parser_from_tts(cx.parse_sess(), cx.cfg(), tts.to_vec()); let expr = parser.parse_expr().unwrap(); MacEager::expr(quote_expr!(&mut *cx, $expr)) } fn expand_into_foo_multi(cx: &mut ExtCtxt, _sp: Span, _attr: &MetaItem, it: Annotatable) -> Vec { match it { Annotatable::Item(it) => vec![ Annotatable::Item(P(Item { attrs: it.attrs.clone(), ..(*quote_item!(cx, enum Foo2 { Bar2, Baz2 }).unwrap()).clone() })), Annotatable::Item(quote_item!(cx, enum Foo3 { Bar }).unwrap()), Annotatable::Item(quote_item!(cx, #[cfg(any())] fn foo2() {}).unwrap()), ], Annotatable::ImplItem(_it) => vec![ quote_item!(cx, impl X { fn foo(&self) -> i32 { 42 } }).unwrap().and_then(|i| { match i.node { ItemKind::Impl(_, _, _, _, _, mut items) => { Annotatable::ImplItem(P(items.pop().expect("impl method not found"))) } _ => unreachable!("impl parsed to something other than impl") } }) ], Annotatable::TraitItem(_it) => vec![ quote_item!(cx, trait X { fn foo(&self) -> i32 { 0 } }).unwrap().and_then(|i| { match i.node { ItemKind::Trait(_, _, _, mut items) => { Annotatable::TraitItem(P(items.pop().expect("trait method not found"))) } _ => unreachable!("trait parsed to something other than trait") } }) ], } } // Create a duplicate of the annotatable, based on the MetaItem fn expand_duplicate(cx: &mut ExtCtxt, _sp: Span, mi: &MetaItem, it: &Annotatable, push: &mut FnMut(Annotatable)) { let copy_name = match mi.node { ast::MetaItemKind::List(_, ref xs) => { if let Some(word) = xs[0].word() { token::str_to_ident(&word.name()) } else { cx.span_err(mi.span, "Expected word"); return; } } _ => { cx.span_err(mi.span, "Expected list"); return; } }; // Duplicate the item but replace its ident by the MetaItem match it.clone() { Annotatable::Item(it) => { let mut new_it = (*it).clone(); new_it.attrs.clear(); new_it.ident = copy_name; push(Annotatable::Item(P(new_it))); } Annotatable::ImplItem(it) => { let mut new_it = (*it).clone(); new_it.attrs.clear(); new_it.ident = copy_name; push(Annotatable::ImplItem(P(new_it))); } Annotatable::TraitItem(tt) => { let mut new_it = (*tt).clone(); new_it.attrs.clear(); new_it.ident = copy_name; push(Annotatable::TraitItem(P(new_it))); } } } pub fn token_separate(ecx: &ExtCtxt, things: &[T], token: token::Token) -> Vec { let mut output: Vec = vec![]; for (i, thing) in things.iter().enumerate() { output.extend(thing.to_tokens(ecx)); if i < things.len() - 1 { output.push(TokenTree::Token(DUMMY_SP, token.clone())); } } output } fn expand_caller(cx: &mut ExtCtxt, sp: Span, mi: &MetaItem, it: &Annotatable, push: &mut FnMut(Annotatable)) { let (orig_fn_name, ret_type) = match *it { Annotatable::Item(ref item) => match item.node { ItemKind::Fn(ref decl, _, _, _, _, _) => { (item.ident, &decl.output) } _ => cx.span_fatal(item.span, "Only functions with return types can be annotated.") }, _ => cx.span_fatal(sp, "Only functions can be annotated.") }; let (caller_name, arguments) = if let Some(list) = mi.meta_item_list() { if list.len() < 2 { cx.span_fatal(mi.span(), "Need a function name and at least one parameter."); } let fn_name = match list[0].name() { Some(name) => token::str_to_ident(&name), None => cx.span_fatal(list[0].span(), "First parameter must be an ident.") }; (fn_name, &list[1..]) } else { cx.span_fatal(mi.span, "Expected list."); }; let literals: Vec = arguments.iter().map(|arg| { if let Some(lit) = arg.literal() { lit.clone() } else { cx.span_fatal(arg.span(), "Expected literal."); } }).collect(); let arguments = token_separate(cx, literals.as_slice(), token::Comma); if let ast::FunctionRetTy::Ty(ref rt) = *ret_type { push(Annotatable::Item(quote_item!(cx, fn $caller_name() -> $rt { $orig_fn_name($arguments) }).unwrap())) } else { push(Annotatable::Item(quote_item!(cx, fn $caller_name() { $orig_fn_name($arguments) }).unwrap())) } } pub fn foo() {}