matrix/matrix-macros/src/lib.rs

56 lines
1.8 KiB
Rust
Raw Normal View History

2024-02-23 11:32:47 -05:00
use proc_macro::TokenStream;
2024-02-26 19:00:42 -05:00
use syn::{ItemFn, parse::Parse, Token, LitInt};
2024-02-23 11:32:47 -05:00
use quote::quote;
struct NativeFuncParams {
arity: LitInt,
2024-02-26 19:00:42 -05:00
variadic: Option<Token![..]>,
2024-02-23 11:32:47 -05:00
}
impl Parse for NativeFuncParams {
fn parse(input: syn::parse::ParseStream) -> syn::Result<Self> {
let arity = input.parse()?;
let variadic = input.parse()?;
Ok(Self { arity , variadic })
}
}
#[proc_macro_attribute]
pub fn native_func(input: TokenStream, annotated_item: TokenStream) -> TokenStream {
let itemfn: ItemFn = syn::parse(annotated_item).unwrap();
let input: NativeFuncParams = syn::parse(input).unwrap();
let arity = input.arity;
2024-02-26 19:00:42 -05:00
let variadic = input.variadic.is_some();
2024-02-23 11:32:47 -05:00
let visibility = itemfn.vis;
let block = itemfn.block;
let name = itemfn.sig.ident;
let name_str = name.to_string();
let inputs = itemfn.sig.inputs;
let output = itemfn.sig.output;
assert!(itemfn.sig.constness.is_none(), "item must not be const");
assert!(itemfn.sig.asyncness.is_none(), "item must not be async");
assert!(itemfn.sig.unsafety.is_none(), "item must not be unsafe");
assert!(itemfn.sig.abi.is_none(), "item must not contain an ABI specifier");
assert!(itemfn.sig.variadic.is_none(), "item must not be variadic");
let expanded = quote! {
2024-02-29 17:04:28 -05:00
#visibility fn #name() -> ::std::rc::Rc< ::matrix_lang::prelude::Function> {
::std::rc::Rc::new( ::matrix_lang::prelude::Function {
2024-02-23 11:32:47 -05:00
name: ::std::rc::Rc::from( #name_str ),
arity: #arity,
variadic: #variadic,
2024-02-29 17:04:28 -05:00
fun: ::matrix_lang::prelude::InnerFunction::Native(
2024-02-26 19:00:42 -05:00
::std::rc::Rc::new(
|#inputs| #output #block
)
)
2024-02-23 11:32:47 -05:00
})
}
};
TokenStream::from(expanded)
}