2014-09-04 02:11:23 -05: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;
|
|
|
|
use syntax::parse::token::{IDENT, get_ident};
|
2014-10-22 07:35:32 -05:00
|
|
|
use syntax::ast::{TokenTree, TTToken};
|
2014-09-04 02:11:23 -05:00
|
|
|
use syntax::ext::base::{ExtCtxt, MacResult, DummyResult, MacExpr};
|
|
|
|
use syntax::ext::build::AstBuilder; // trait for expr_uint
|
|
|
|
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-22 07:35:32 -05:00
|
|
|
[TTToken(_, IDENT(s, _))] => get_ident(s).to_string(),
|
2014-09-04 02:11:23 -05:00
|
|
|
_ => {
|
|
|
|
cx.span_err(sp, "argument should be a single identifier");
|
|
|
|
return DummyResult::any(sp);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut text = text.as_slice();
|
|
|
|
let mut total = 0u;
|
|
|
|
while !text.is_empty() {
|
|
|
|
match NUMERALS.iter().find(|&&(rn, _)| text.starts_with(rn)) {
|
|
|
|
Some(&(rn, val)) => {
|
|
|
|
total += val;
|
|
|
|
text = text.slice_from(rn.len());
|
|
|
|
}
|
|
|
|
None => {
|
|
|
|
cx.span_err(sp, "invalid Roman numeral");
|
|
|
|
return DummyResult::any(sp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MacExpr::new(cx.expr_uint(sp, total))
|
|
|
|
}
|
|
|
|
|
|
|
|
#[plugin_registrar]
|
|
|
|
pub fn plugin_registrar(reg: &mut Registry) {
|
|
|
|
reg.register_macro("rn", expand_rn);
|
|
|
|
}
|