2014-09-04 00:11:23 -07:00
|
|
|
// Copyright 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.
|
|
|
|
|
|
|
|
// force-host
|
|
|
|
|
|
|
|
#![crate_type="dylib"]
|
|
|
|
#![feature(plugin_registrar)]
|
|
|
|
|
|
|
|
extern crate syntax;
|
|
|
|
extern crate rustc;
|
|
|
|
|
|
|
|
use syntax::codemap::Span;
|
2014-10-27 19:22:52 +11:00
|
|
|
use syntax::parse::token;
|
2014-10-23 04:39:58 +11:00
|
|
|
use syntax::ast::{TokenTree, TtToken};
|
2015-02-27 11:14:42 -08:00
|
|
|
use syntax::ext::base::{ExtCtxt, MacResult, DummyResult, MacEager};
|
2015-01-17 23:49:08 +00:00
|
|
|
use syntax::ext::build::AstBuilder; // trait for expr_usize
|
2014-09-04 00:11:23 -07:00
|
|
|
use rustc::plugin::Registry;
|
|
|
|
|
|
|
|
// WARNING WARNING WARNING WARNING WARNING
|
|
|
|
// =======================================
|
|
|
|
//
|
|
|
|
// This code also appears in src/doc/guide-plugin.md. Please keep
|
|
|
|
// the two copies in sync! FIXME: have rustdoc read this file
|
|
|
|
|
|
|
|
fn expand_rn(cx: &mut ExtCtxt, sp: Span, args: &[TokenTree])
|
|
|
|
-> Box<MacResult + 'static> {
|
|
|
|
|
|
|
|
static NUMERALS: &'static [(&'static str, uint)] = &[
|
|
|
|
("M", 1000), ("CM", 900), ("D", 500), ("CD", 400),
|
|
|
|
("C", 100), ("XC", 90), ("L", 50), ("XL", 40),
|
|
|
|
("X", 10), ("IX", 9), ("V", 5), ("IV", 4),
|
|
|
|
("I", 1)];
|
|
|
|
|
|
|
|
let text = match args {
|
2014-10-27 19:22:52 +11:00
|
|
|
[TtToken(_, token::Ident(s, _))] => token::get_ident(s).to_string(),
|
2014-09-04 00:11:23 -07:00
|
|
|
_ => {
|
|
|
|
cx.span_err(sp, "argument should be a single identifier");
|
|
|
|
return DummyResult::any(sp);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-02-01 21:53:25 -05:00
|
|
|
let mut text = &*text;
|
2015-03-03 10:42:26 +02:00
|
|
|
let mut total = 0;
|
2014-09-04 00:11:23 -07:00
|
|
|
while !text.is_empty() {
|
|
|
|
match NUMERALS.iter().find(|&&(rn, _)| text.starts_with(rn)) {
|
|
|
|
Some(&(rn, val)) => {
|
|
|
|
total += val;
|
2015-01-26 21:21:15 -05:00
|
|
|
text = &text[rn.len()..];
|
2014-09-04 00:11:23 -07:00
|
|
|
}
|
|
|
|
None => {
|
|
|
|
cx.span_err(sp, "invalid Roman numeral");
|
|
|
|
return DummyResult::any(sp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-27 11:14:42 -08:00
|
|
|
MacEager::expr(cx.expr_usize(sp, total))
|
2014-09-04 00:11:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[plugin_registrar]
|
|
|
|
pub fn plugin_registrar(reg: &mut Registry) {
|
|
|
|
reg.register_macro("rn", expand_rn);
|
|
|
|
}
|