// 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)] #![feature(box_syntax, rustc_private)] extern crate syntax; extern crate rustc; use syntax::ast::{self, TokenTree, Item, MetaItem}; use syntax::codemap::Span; use syntax::ext::base::*; use syntax::parse::token; use syntax::parse; use syntax::ptr::P; 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("forged_ident", expand_forged_ident); reg.register_macro("identity", expand_identity); reg.register_syntax_extension( token::intern("into_foo"), // FIXME (#22405): Replace `Box::new` with `box` here when/if possible. Modifier(Box::new(expand_into_foo))); 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))); } 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(); MacEager::expr(quote_expr!(&mut *cx, $expr)) } fn expand_into_foo(cx: &mut ExtCtxt, sp: Span, attr: &MetaItem, it: P) -> P { P(Item { attrs: it.attrs.clone(), ..(*quote_item!(cx, enum Foo { Bar, Baz }).unwrap()).clone() }) } fn expand_into_foo_multi(cx: &mut ExtCtxt, sp: Span, attr: &MetaItem, it: Annotatable) -> Annotatable { match it { Annotatable::Item(it) => { Annotatable::Item(P(Item { attrs: it.attrs.clone(), ..(*quote_item!(cx, enum Foo2 { Bar2, Baz2 }).unwrap()).clone() })) } Annotatable::ImplItem(it) => { quote_item!(cx, impl X { fn foo(&self) -> i32 { 42 } }).unwrap().and_then(|i| { match i.node { ast::ItemImpl(_, _, _, _, _, mut items) => { Annotatable::ImplItem(items.pop().expect("impl method not found")) } _ => unreachable!("impl parsed to something other than impl") } }) } Annotatable::TraitItem(it) => { quote_item!(cx, trait X { fn foo(&self) -> i32 { 0 } }).unwrap().and_then(|i| { match i.node { ast::ItemTrait(_, _, _, mut items) => { Annotatable::TraitItem(items.pop().expect("trait method not found")) } _ => unreachable!("trait parsed to something other than trait") } }) } } } fn expand_forged_ident(cx: &mut ExtCtxt, sp: Span, tts: &[TokenTree]) -> Box { use syntax::ext::quote::rt::*; if !tts.is_empty() { cx.span_fatal(sp, "forged_ident takes no arguments"); } // Most of this is modelled after the expansion of the `quote_expr!` // macro ... let parse_sess = cx.parse_sess(); let cfg = cx.cfg(); // ... except this is where we inject a forged identifier, // and deliberately do not call `cx.parse_tts_with_hygiene` // (because we are testing that this will be *rejected* // by the default parser). let expr = { let tt = cx.parse_tts("\x00name_2,ctxt_0\x00".to_string()); let mut parser = new_parser_from_tts(parse_sess, cfg, tt); parser.parse_expr() }; MacEager::expr(expr) } pub fn foo() {}