// Copyright 2012-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. #![crate_type = "proc-macro"] extern crate syn; #[macro_use] extern crate quote; extern crate proc_macro; use proc_macro::TokenStream; use syn::Body::Enum; use syn::VariantData::Unit; #[proc_macro_derive(FromPrimitive)] pub fn from_primitive(input: TokenStream) -> TokenStream { let source = input.to_string(); let ast = syn::parse_macro_input(&source).unwrap(); let name = &ast.ident; let variants = match ast.body { Enum(ref variants) => variants, _ => panic!("`FromPrimitive` can be applied only to the enums, {} is not an enum", name) }; let mut idx = 0; let variants: Vec<_> = variants.iter() .map(|variant| { let ident = &variant.ident; match variant.data { Unit => (), _ => { panic!("`FromPrimitive` can be applied only to unitary enums, {}::{} is either struct or tuple", name, ident) }, } if let Some(val) = variant.discriminant { idx = val.value; } let tt = quote!(#idx => Some(#name::#ident)); idx += 1; tt }) .collect(); let res = quote! { impl ::num::traits::FromPrimitive for #name { fn from_i64(n: i64) -> Option { Self::from_u64(n as u64) } fn from_u64(n: u64) -> Option { match n { #(variants,)* _ => None, } } } }; res.to_string().parse().unwrap() } #[proc_macro_derive(ToPrimitive)] pub fn to_primitive(input: TokenStream) -> TokenStream { let source = input.to_string(); let ast = syn::parse_macro_input(&source).unwrap(); let name = &ast.ident; let variants = match ast.body { Enum(ref variants) => variants, _ => panic!("`ToPrimitive` can be applied only to the enums, {} is not an enum", name) }; let mut idx = 0; let variants: Vec<_> = variants.iter() .map(|variant| { let ident = &variant.ident; match variant.data { Unit => (), _ => { panic!("`ToPrimitive` can be applied only to unitary enums, {}::{} is either struct or tuple", name, ident) }, } if let Some(val) = variant.discriminant { idx = val.value; } let tt = quote!(#name::#ident => #idx); idx += 1; tt }) .collect(); let res = quote! { impl ::num::traits::ToPrimitive for #name { fn to_i64(&self) -> Option { self.to_u64().map(|x| x as i64) } fn to_u64(&self) -> Option { Some(match *self { #(variants,)* }) } } }; res.to_string().parse().unwrap() }