extern crate proc_macro;
mod parse;
use self::parse::{Int, LiteralError};
use parse::Uint;
use proc_macro::{Delimiter, Literal, Span, TokenStream, TokenTree};
#[proc_macro]
pub fn int(input: TokenStream) -> TokenStream {
match IntLiteral::generate(input) {
Ok(value) => value.into_tokens(),
Err(err) => err.into_tokens(),
}
}
#[proc_macro]
pub fn uint(input: TokenStream) -> TokenStream {
match UintLiteral::generate(input) {
Ok(value) => value.into_tokens(),
Err(err) => err.into_tokens(),
}
}
struct IntLiteral(Int, String);
impl IntLiteral {
fn generate(input: TokenStream) -> Result<Self, CompileError> {
let input = Input::parse(input)?;
Ok(Self(Int::from_literal(&input.value)?, input.crate_name))
}
fn into_tokens(self) -> TokenStream {
let (hi, lo) = self.0.into_words();
format!("{}::I256::from_words({hi}, {lo})", self.1)
.parse()
.unwrap()
}
}
struct UintLiteral(Uint, String);
impl UintLiteral {
fn generate(input: TokenStream) -> Result<Self, CompileError> {
let input = Input::parse(input)?;
Ok(Self(Uint::from_literal(&input.value)?, input.crate_name))
}
fn into_tokens(self) -> TokenStream {
let (hi, lo) = self.0.into_words();
format!("{}::U256::from_words({hi}, {lo})", self.1)
.parse()
.unwrap()
}
}
struct Input {
value: String,
span: Span,
crate_name: String,
}
impl Input {
fn parse(input: TokenStream) -> Result<Self, CompileError> {
let mut result = Input {
value: String::new(),
span: Span::call_site(),
crate_name: "::ethnum".to_owned(),
};
ParserState::start().input(input, &mut result)?.end()?;
Ok(result)
}
}
enum ParserState {
String,
CommaOrEof,
Crate,
EqualCrateName,
CrateName,
Eof,
}
impl ParserState {
fn start() -> Self {
Self::String
}
fn input(self, input: TokenStream, result: &mut Input) -> Result<Self, CompileError> {
input
.into_iter()
.try_fold(self, |state, token| state.next(token, result))
}
fn next(self, token: TokenTree, result: &mut Input) -> Result<Self, CompileError> {
match (&self, &token) {
(_, TokenTree::Group(g)) if g.delimiter() == Delimiter::None => {
self.input(g.stream(), result)
}
(Self::String, TokenTree::Literal(l)) => match parse_string(l) {
Some(value) => {
result.value = value;
result.span = token.span();
Ok(Self::CommaOrEof)
}
None => Err(self.unexpected(Some(token))),
},
(Self::CommaOrEof, TokenTree::Punct(p)) if p.as_char() == ',' => Ok(Self::Crate),
(Self::Crate, TokenTree::Ident(c)) if c.to_string() == "crate" => {
Ok(Self::EqualCrateName)
}
(Self::EqualCrateName, TokenTree::Punct(p)) if p.as_char() == '=' => {
Ok(Self::CrateName)
}
(Self::CrateName, TokenTree::Literal(l)) => match parse_string(l) {
Some(value) => {
result.crate_name = value;
Ok(Self::Eof)
}
None => Err(self.unexpected(Some(token))),
},
_ => Err(self.unexpected(Some(token))),
}
}
fn end(self) -> Result<(), CompileError> {
match self {
Self::CommaOrEof | Self::Eof => Ok(()),
_ => Err(self.unexpected(None)),
}
}
fn unexpected(self, token: Option<TokenTree>) -> CompileError {
let expected = match self {
Self::String => "string literal",
Self::CommaOrEof => "`,` or <eof>",
Self::Crate => "`crate` identifier",
Self::EqualCrateName => "`=`",
Self::CrateName => "crate name string literal",
Self::Eof => "<eof>",
};
let (value, span) = match token {
Some(TokenTree::Group(g)) => {
let delim = match g.delimiter() {
Delimiter::Parenthesis => "(",
Delimiter::Brace => "{",
Delimiter::Bracket => "[",
Delimiter::None => "Ø",
};
(delim.to_string(), Some(g.span_open()))
}
Some(t) => (t.to_string(), Some(t.span())),
None => ("<eof>".to_owned(), None),
};
CompileError {
message: format!("expected {expected} but found `{value}`"),
span,
}
}
}
struct CompileError {
message: String,
span: Option<Span>,
}
impl CompileError {
fn into_tokens(self) -> TokenStream {
let error = format!("compile_error!({:?})", self.message)
.parse::<TokenStream>()
.unwrap();
match self.span {
Some(span) => error
.into_iter()
.map(|mut token| {
token.set_span(span);
token
})
.collect(),
None => error,
}
}
}
impl From<LiteralError> for CompileError {
fn from(err: LiteralError) -> Self {
Self {
message: err.to_string(),
span: None,
}
}
}
fn parse_string(literal: &Literal) -> Option<String> {
Some(
literal
.to_string()
.strip_prefix('"')?
.strip_suffix('"')?
.to_owned(),
)
}