2013-12-25 11:10:33 -07:00
|
|
|
// Copyright 2013-2014 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-01-17 10:18:02 -08:00
|
|
|
// force-host
|
|
|
|
|
2014-07-26 21:05:03 -07:00
|
|
|
#![feature(globs, plugin_registrar, macro_rules, quote)]
|
2013-12-25 11:10:33 -07:00
|
|
|
|
2014-02-14 10:10:06 -08:00
|
|
|
extern crate syntax;
|
2014-05-24 21:38:16 -07:00
|
|
|
extern crate rustc;
|
2013-12-25 11:10:33 -07:00
|
|
|
|
2014-05-24 21:38:16 -07:00
|
|
|
use syntax::ast::{TokenTree, Item, MetaItem};
|
2013-12-25 11:10:33 -07:00
|
|
|
use syntax::codemap::Span;
|
|
|
|
use syntax::ext::base::*;
|
|
|
|
use syntax::parse::token;
|
2014-08-14 11:55:47 +02:00
|
|
|
use syntax::parse;
|
2014-09-12 02:33:08 +03:00
|
|
|
use syntax::ptr::P;
|
2014-05-24 21:38:16 -07:00
|
|
|
use rustc::plugin::Registry;
|
2013-12-25 11:10:33 -07:00
|
|
|
|
|
|
|
#[macro_export]
|
2014-11-14 09:18:10 -08:00
|
|
|
macro_rules! exported_macro (() => (2i));
|
2013-12-25 11:10:33 -07:00
|
|
|
|
2014-11-14 09:18:10 -08:00
|
|
|
macro_rules! unexported_macro (() => (3i));
|
2013-12-25 11:10:33 -07:00
|
|
|
|
2014-05-24 21:38:16 -07:00
|
|
|
#[plugin_registrar]
|
|
|
|
pub fn plugin_registrar(reg: &mut Registry) {
|
|
|
|
reg.register_macro("make_a_1", expand_make_a_1);
|
2014-08-13 13:15:31 +02:00
|
|
|
reg.register_macro("forged_ident", expand_forged_ident);
|
2014-08-14 11:55:47 +02:00
|
|
|
reg.register_macro("identity", expand_identity);
|
2014-05-24 21:38:16 -07:00
|
|
|
reg.register_syntax_extension(
|
|
|
|
token::intern("into_foo"),
|
2014-09-11 17:07:49 +12:00
|
|
|
Modifier(box expand_into_foo));
|
2013-12-25 11:10:33 -07:00
|
|
|
}
|
|
|
|
|
2014-05-05 18:56:44 -07:00
|
|
|
fn expand_make_a_1(cx: &mut ExtCtxt, sp: Span, tts: &[TokenTree])
|
2014-08-27 21:46:52 -04:00
|
|
|
-> Box<MacResult+'static> {
|
2013-12-25 11:10:33 -07:00
|
|
|
if !tts.is_empty() {
|
|
|
|
cx.span_fatal(sp, "make_a_1 takes no arguments");
|
|
|
|
}
|
2014-04-15 22:00:14 +10:00
|
|
|
MacExpr::new(quote_expr!(cx, 1i))
|
2013-12-25 11:10:33 -07:00
|
|
|
}
|
|
|
|
|
2014-08-14 11:55:47 +02:00
|
|
|
// See Issue #15750
|
|
|
|
fn expand_identity(cx: &mut ExtCtxt, _span: Span, tts: &[TokenTree])
|
2014-08-27 21:46:52 -04:00
|
|
|
-> Box<MacResult+'static> {
|
2014-08-14 11:55:47 +02:00
|
|
|
// Parse an expression and emit it unchanged.
|
|
|
|
let mut parser = parse::new_parser_from_tts(cx.parse_sess(),
|
2014-10-14 23:05:01 -07:00
|
|
|
cx.cfg(), tts.to_vec());
|
2014-08-14 11:55:47 +02:00
|
|
|
let expr = parser.parse_expr();
|
|
|
|
MacExpr::new(quote_expr!(&mut *cx, $expr))
|
|
|
|
}
|
|
|
|
|
2014-09-12 02:33:08 +03:00
|
|
|
fn expand_into_foo(cx: &mut ExtCtxt, sp: Span, attr: &MetaItem, it: P<Item>)
|
|
|
|
-> P<Item> {
|
|
|
|
P(Item {
|
2014-02-27 23:49:25 -08:00
|
|
|
attrs: it.attrs.clone(),
|
|
|
|
..(*quote_item!(cx, enum Foo { Bar, Baz }).unwrap()).clone()
|
2014-09-12 02:33:08 +03:00
|
|
|
})
|
2014-02-27 23:49:25 -08:00
|
|
|
}
|
|
|
|
|
2014-08-27 21:46:52 -04:00
|
|
|
fn expand_forged_ident(cx: &mut ExtCtxt, sp: Span, tts: &[TokenTree]) -> Box<MacResult+'static> {
|
2014-08-13 13:15:31 +02:00
|
|
|
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()
|
|
|
|
};
|
|
|
|
MacExpr::new(expr)
|
|
|
|
}
|
|
|
|
|
2013-12-25 11:10:33 -07:00
|
|
|
pub fn foo() {}
|