rhai/codegen/src/rhai_module.rs

136 lines
4.9 KiB
Rust
Raw Normal View History

2020-08-01 18:52:26 +02:00
use quote::quote;
use crate::function::ExportedFn;
pub(crate) type ExportedConst = (String, syn::Expr);
pub(crate) fn get_register_name(function: &ExportedFn) -> String {
pub const FN_IDX_GET: &str = "index$get$";
pub const FN_IDX_SET: &str = "index$set$";
pub fn make_getter(id: &str) -> String {
format!("get${}", id)
}
pub fn make_setter(id: &str) -> String {
format!("set${}", id)
}
if let Some(ref name) = function.params.name {
name.clone()
} else if let Some(ref name) = function.params.get {
make_getter(name).clone()
} else if let Some(ref name) = function.params.set {
make_setter(name).clone()
} else if function.params.index_get {
FN_IDX_GET.to_string()
} else if function.params.index_set {
FN_IDX_SET.to_string()
} else {
function.name().to_string()
}
}
2020-08-01 18:52:26 +02:00
pub(crate) fn generate_body(
fns: &Vec<ExportedFn>,
2020-08-02 09:39:08 +02:00
consts: &Vec<ExportedConst>,
2020-08-01 18:52:26 +02:00
) -> proc_macro2::TokenStream {
let mut set_fn_stmts: Vec<syn::Stmt> = Vec::new();
let mut set_const_stmts: Vec<syn::Stmt> = Vec::new();
let str_type_path = syn::parse2::<syn::Path>(quote! { str }).unwrap();
for (const_name, const_expr) in consts {
let const_literal = syn::LitStr::new(&const_name, proc_macro2::Span::call_site());
2020-08-02 09:39:08 +02:00
set_const_stmts.push(
syn::parse2::<syn::Stmt>(quote! {
m.set_var(#const_literal, #const_expr);
})
.unwrap(),
);
2020-08-01 18:52:26 +02:00
}
// NB: these are token streams, because reparsing messes up "> >" vs ">>"
let mut gen_fn_tokens: Vec<proc_macro2::TokenStream> = Vec::new();
for function in fns {
2020-08-02 09:39:08 +02:00
let fn_token_name = syn::Ident::new(
2020-08-07 01:36:15 +02:00
&format!("{}_token", function.name().to_string()),
2020-08-02 09:39:08 +02:00
function.name().span(),
);
let reg_name = get_register_name(function);
2020-08-16 12:24:42 +02:00
let fn_literal = syn::LitStr::new(&reg_name, proc_macro2::Span::call_site());
2020-08-02 09:39:08 +02:00
let fn_input_types: Vec<syn::Expr> = function
.arg_list()
2020-08-01 18:52:26 +02:00
.map(|fnarg| match fnarg {
syn::FnArg::Receiver(_) => panic!("internal error: receiver fn outside impl!?"),
syn::FnArg::Typed(syn::PatType { ref ty, .. }) => {
let arg_type = match ty.as_ref() {
2020-08-02 09:39:08 +02:00
&syn::Type::Reference(syn::TypeReference {
mutability: None,
ref elem,
..
}) => match elem.as_ref() {
&syn::Type::Path(ref p) if p.path == str_type_path => {
syn::parse2::<syn::Type>(quote! {
2020-08-03 02:27:19 +02:00
ImmutableString })
2020-08-02 09:39:08 +02:00
.unwrap()
2020-08-01 18:52:26 +02:00
}
2020-08-02 09:39:08 +02:00
_ => panic!("internal error: non-string shared reference!?"),
2020-08-01 18:52:26 +02:00
},
2020-08-02 09:39:08 +02:00
&syn::Type::Reference(syn::TypeReference {
mutability: Some(_),
ref elem,
..
}) => match elem.as_ref() {
&syn::Type::Path(ref p) => syn::parse2::<syn::Type>(quote! {
#p })
.unwrap(),
_ => panic!("internal error: non-string shared reference!?"),
2020-08-01 18:52:26 +02:00
},
t => t.clone(),
};
syn::parse2::<syn::Expr>(quote! {
2020-08-02 09:39:08 +02:00
core::any::TypeId::of::<#arg_type>()})
.unwrap()
}
})
.collect();
2020-08-01 18:52:26 +02:00
2020-08-02 09:39:08 +02:00
set_fn_stmts.push(
syn::parse2::<syn::Stmt>(quote! {
m.set_fn(#fn_literal, FnAccess::Public, &[#(#fn_input_types),*],
2020-08-03 02:27:19 +02:00
CallableFunction::from_plugin(#fn_token_name()));
2020-08-02 09:39:08 +02:00
})
.unwrap(),
);
2020-08-01 18:52:26 +02:00
gen_fn_tokens.push(quote! {
#[allow(non_camel_case_types)]
2020-08-03 02:27:19 +02:00
struct #fn_token_name();
2020-08-01 18:52:26 +02:00
});
gen_fn_tokens.push(function.generate_impl(&fn_token_name.to_string()));
2020-08-03 02:27:19 +02:00
gen_fn_tokens.push(function.generate_callable(&fn_token_name.to_string()));
gen_fn_tokens.push(function.generate_input_types(&fn_token_name.to_string()));
2020-08-01 18:52:26 +02:00
}
let mut generate_fncall = syn::parse2::<syn::ItemMod>(quote! {
pub mod generate_info {
#[allow(unused_imports)]
2020-08-03 02:27:19 +02:00
use super::*;
2020-08-01 18:52:26 +02:00
#[allow(unused_mut)]
pub fn rhai_module_generate() -> Module {
2020-08-01 18:52:26 +02:00
let mut m = Module::new();
#(#set_fn_stmts)*
#(#set_const_stmts)*
m
}
}
2020-08-02 09:39:08 +02:00
})
.unwrap();
2020-08-01 18:52:26 +02:00
let (_, generate_call_content) = generate_fncall.content.take().unwrap();
quote! {
#(#generate_call_content)*
#(#gen_fn_tokens)*
}
}