2021-09-23 02:29:09 +09:00
|
|
|
use proc_macro2::{Ident, Span, TokenStream};
|
|
|
|
use quote::ToTokens;
|
|
|
|
|
|
|
|
use crate::{
|
2021-11-23 20:00:31 +09:00
|
|
|
codegen::{get_intermediate_ident, get_register_ident, js_mod_to_token_stream},
|
2021-09-23 02:29:09 +09:00
|
|
|
BindgenResult, CallbackArg, FnKind, FnSelf, NapiFn, NapiFnArgKind, TryToTokens,
|
|
|
|
};
|
|
|
|
|
|
|
|
impl TryToTokens for NapiFn {
|
|
|
|
fn try_to_tokens(&self, tokens: &mut TokenStream) -> BindgenResult<()> {
|
|
|
|
let name_str = self.name.to_string();
|
|
|
|
let intermediate_ident = get_intermediate_ident(&name_str);
|
|
|
|
let args_len = self.args.len();
|
|
|
|
|
|
|
|
let (arg_conversions, arg_names) = self.gen_arg_conversions();
|
|
|
|
let receiver = self.gen_fn_receiver();
|
|
|
|
let receiver_ret_name = Ident::new("_ret", Span::call_site());
|
|
|
|
let ret = self.gen_fn_return(&receiver_ret_name);
|
|
|
|
let register = self.gen_fn_register();
|
|
|
|
let attrs = &self.attrs;
|
2021-10-25 01:00:31 +09:00
|
|
|
|
|
|
|
let native_call = if !self.is_async {
|
|
|
|
quote! {
|
|
|
|
let #receiver_ret_name = {
|
|
|
|
#receiver(#(#arg_names),*)
|
|
|
|
};
|
|
|
|
#ret
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
let call = if self.is_ret_result {
|
|
|
|
quote! { #receiver(#(#arg_names),*).await }
|
|
|
|
} else {
|
|
|
|
quote! { Ok(#receiver(#(#arg_names),*).await) }
|
|
|
|
};
|
2021-10-01 02:37:45 +09:00
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
napi::bindgen_prelude::execute_tokio_future(env, async move { #call }, |env, #receiver_ret_name| {
|
2021-10-01 02:37:45 +09:00
|
|
|
#ret
|
2021-10-25 01:00:31 +09:00
|
|
|
})
|
2021-10-01 02:37:45 +09:00
|
|
|
}
|
2021-10-25 01:00:31 +09:00
|
|
|
};
|
|
|
|
|
2021-11-05 19:31:36 +09:00
|
|
|
let function_call = if args_len == 0
|
|
|
|
&& self.fn_self.is_none()
|
|
|
|
&& self.kind != FnKind::Constructor
|
|
|
|
&& self.kind != FnKind::Factory
|
|
|
|
{
|
|
|
|
quote! { #native_call }
|
|
|
|
} else if self.kind == FnKind::Constructor {
|
|
|
|
quote! {
|
|
|
|
// constructor function is called from class `factory`
|
|
|
|
// so we should skip the original `constructor` logic
|
2022-03-05 15:14:32 +09:00
|
|
|
if napi::bindgen_prelude::___CALL_FROM_FACTORY.with(|inner| inner.load(std::sync::atomic::Ordering::Relaxed)) {
|
2021-11-05 19:31:36 +09:00
|
|
|
return std::ptr::null_mut();
|
2021-10-28 19:11:06 +09:00
|
|
|
}
|
2021-11-15 08:34:44 +09:00
|
|
|
napi::bindgen_prelude::CallbackInfo::<#args_len>::new(env, cb, None).and_then(|mut cb| {
|
2021-11-05 19:31:36 +09:00
|
|
|
#(#arg_conversions)*
|
|
|
|
#native_call
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
napi::bindgen_prelude::CallbackInfo::<#args_len>::new(env, cb, None).and_then(|mut cb| {
|
2021-11-05 19:31:36 +09:00
|
|
|
#(#arg_conversions)*
|
|
|
|
#native_call
|
|
|
|
})
|
|
|
|
}
|
|
|
|
};
|
2021-10-25 01:00:31 +09:00
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
(quote! {
|
|
|
|
#(#attrs)*
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[allow(non_snake_case)]
|
|
|
|
#[allow(clippy::all)]
|
|
|
|
extern "C" fn #intermediate_ident(
|
2021-11-15 08:34:44 +09:00
|
|
|
env: napi::bindgen_prelude::sys::napi_env,
|
|
|
|
cb: napi::bindgen_prelude::sys::napi_callback_info
|
|
|
|
) -> napi::bindgen_prelude::sys::napi_value {
|
2021-09-23 02:29:09 +09:00
|
|
|
unsafe {
|
2021-10-25 01:00:31 +09:00
|
|
|
#function_call.unwrap_or_else(|e| {
|
2021-11-15 08:34:44 +09:00
|
|
|
napi::bindgen_prelude::JsError::from(e).throw_into(env);
|
|
|
|
std::ptr::null_mut::<napi::bindgen_prelude::sys::napi_value__>()
|
2021-09-23 02:29:09 +09:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#register
|
|
|
|
})
|
|
|
|
.to_tokens(tokens);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl NapiFn {
|
|
|
|
fn gen_arg_conversions(&self) -> (Vec<TokenStream>, Vec<TokenStream>) {
|
|
|
|
let mut arg_conversions = vec![];
|
|
|
|
let mut args = vec![];
|
|
|
|
|
|
|
|
// fetch this
|
|
|
|
if let Some(parent) = &self.parent {
|
|
|
|
match self.fn_self {
|
|
|
|
Some(FnSelf::Ref) => {
|
|
|
|
arg_conversions.push(quote! { let this = cb.unwrap_borrow::<#parent>()?; });
|
|
|
|
}
|
|
|
|
Some(FnSelf::MutRef) => {
|
|
|
|
arg_conversions.push(quote! { let this = cb.unwrap_borrow_mut::<#parent>()?; });
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut skipped_arg_count = 0;
|
|
|
|
self.args.iter().enumerate().for_each(|(i, arg)| {
|
|
|
|
let i = i - skipped_arg_count;
|
|
|
|
let ident = Ident::new(&format!("arg{}", i), Span::call_site());
|
|
|
|
|
|
|
|
match arg {
|
|
|
|
NapiFnArgKind::PatType(path) => {
|
|
|
|
if &path.ty.to_token_stream().to_string() == "Env" {
|
2021-11-15 08:34:44 +09:00
|
|
|
args.push(quote! { napi::bindgen_prelude::Env::from(env) });
|
2021-09-23 02:29:09 +09:00
|
|
|
skipped_arg_count += 1;
|
|
|
|
} else {
|
|
|
|
arg_conversions.push(self.gen_ty_arg_conversion(&ident, i, path));
|
|
|
|
args.push(quote! { #ident });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
NapiFnArgKind::Callback(cb) => {
|
|
|
|
arg_conversions.push(self.gen_cb_arg_conversion(&ident, i, cb));
|
|
|
|
args.push(quote! { #ident });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
(arg_conversions, args)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_ty_arg_conversion(
|
|
|
|
&self,
|
|
|
|
arg_name: &Ident,
|
|
|
|
index: usize,
|
|
|
|
path: &syn::PatType,
|
|
|
|
) -> TokenStream {
|
|
|
|
let ty = &*path.ty;
|
|
|
|
match ty {
|
|
|
|
syn::Type::Reference(syn::TypeReference {
|
|
|
|
mutability: Some(_),
|
|
|
|
elem,
|
|
|
|
..
|
|
|
|
}) => {
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
let #arg_name = <#elem as napi::bindgen_prelude::FromNapiMutRef>::from_napi_mut_ref(env, cb.get_arg(#index))?;
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
syn::Type::Reference(syn::TypeReference { elem, .. }) => {
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
let #arg_name = <#elem as napi::bindgen_prelude::FromNapiRef>::from_napi_ref(env, cb.get_arg(#index))?;
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
let type_check = if self.strict {
|
|
|
|
quote! {
|
2022-03-06 00:19:51 +09:00
|
|
|
let maybe_promise = <#ty as napi::bindgen_prelude::ValidateNapiValue>::validate(env, cb.get_arg(#index))?;
|
|
|
|
if !maybe_promise.is_null() {
|
|
|
|
return Ok(maybe_promise);
|
|
|
|
}
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
quote! {}
|
|
|
|
};
|
|
|
|
|
|
|
|
quote! {
|
2021-10-25 01:00:31 +09:00
|
|
|
let #arg_name = {
|
2021-09-23 02:29:09 +09:00
|
|
|
#type_check
|
2021-11-15 08:34:44 +09:00
|
|
|
<#ty as napi::bindgen_prelude::FromNapiValue>::from_napi_value(env, cb.get_arg(#index))?
|
2021-09-23 02:29:09 +09:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_cb_arg_conversion(&self, arg_name: &Ident, index: usize, cb: &CallbackArg) -> TokenStream {
|
|
|
|
let mut inputs = vec![];
|
|
|
|
let mut arg_conversions = vec![];
|
|
|
|
|
|
|
|
for (i, ty) in cb.args.iter().enumerate() {
|
|
|
|
let cb_arg_ident = Ident::new(&format!("callback_arg_{}", i), Span::call_site());
|
|
|
|
inputs.push(quote! { #cb_arg_ident: #ty });
|
2021-11-15 08:34:44 +09:00
|
|
|
arg_conversions.push(
|
|
|
|
quote! { <#ty as napi::bindgen_prelude::ToNapiValue>::to_napi_value(env, #cb_arg_ident)? },
|
|
|
|
);
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
let ret = match &cb.ret {
|
|
|
|
Some(ty) => {
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
let ret = <#ty as napi::bindgen_prelude::FromNapiValue>::from_napi_value(env, ret_ptr)?;
|
2021-09-23 02:29:09 +09:00
|
|
|
|
|
|
|
Ok(ret)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => quote! { Ok(()) },
|
|
|
|
};
|
|
|
|
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
napi::bindgen_prelude::assert_type_of!(env, cb.get_arg(#index), napi::bindgen_prelude::ValueType::Function)?;
|
2021-09-23 02:29:09 +09:00
|
|
|
let #arg_name = |#(#inputs),*| {
|
|
|
|
let args = vec![
|
|
|
|
#(#arg_conversions),*
|
|
|
|
];
|
|
|
|
|
|
|
|
let mut ret_ptr = std::ptr::null_mut();
|
|
|
|
|
2021-11-15 08:34:44 +09:00
|
|
|
napi::bindgen_prelude::check_status!(
|
|
|
|
napi::bindgen_prelude::sys::napi_call_function(
|
2021-09-23 02:29:09 +09:00
|
|
|
env,
|
|
|
|
cb.this(),
|
|
|
|
cb.get_arg(#index),
|
|
|
|
args.len(),
|
|
|
|
args.as_ptr(),
|
|
|
|
&mut ret_ptr
|
|
|
|
),
|
|
|
|
"Failed to call napi callback",
|
|
|
|
)?;
|
|
|
|
|
|
|
|
#ret
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_fn_receiver(&self) -> TokenStream {
|
|
|
|
let name = &self.name;
|
|
|
|
|
|
|
|
match self.fn_self {
|
|
|
|
Some(FnSelf::Value) => {
|
2021-10-01 02:37:45 +09:00
|
|
|
// impossible, panic! in parser
|
2021-09-23 02:29:09 +09:00
|
|
|
unimplemented!();
|
|
|
|
}
|
|
|
|
Some(FnSelf::Ref) | Some(FnSelf::MutRef) => quote! { this.#name },
|
|
|
|
None => match &self.parent {
|
|
|
|
Some(class) => quote! { #class::#name },
|
|
|
|
None => quote! { #name },
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_fn_return(&self, ret: &Ident) -> TokenStream {
|
|
|
|
let js_name = &self.js_name;
|
|
|
|
|
2021-10-25 01:00:31 +09:00
|
|
|
if let Some(ty) = &self.ret {
|
2021-11-25 23:31:11 +09:00
|
|
|
let ty_string = ty.into_token_stream().to_string();
|
|
|
|
let is_return_self = ty_string == "& Self" || ty_string == "&mut Self";
|
2021-09-24 15:45:27 +09:00
|
|
|
if self.kind == FnKind::Constructor {
|
2021-11-06 22:48:18 +09:00
|
|
|
if self.is_ret_result {
|
|
|
|
quote! { cb.construct(#js_name, #ret?) }
|
|
|
|
} else {
|
|
|
|
quote! { cb.construct(#js_name, #ret) }
|
|
|
|
}
|
2021-11-05 19:31:36 +09:00
|
|
|
} else if self.kind == FnKind::Factory {
|
2021-11-06 22:48:18 +09:00
|
|
|
if self.is_ret_result {
|
|
|
|
quote! { cb.factory(#js_name, #ret?) }
|
|
|
|
} else {
|
|
|
|
quote! { cb.factory(#js_name, #ret) }
|
|
|
|
}
|
2021-10-25 01:00:31 +09:00
|
|
|
} else if self.is_ret_result {
|
|
|
|
if self.is_async {
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
<#ty as napi::bindgen_prelude::ToNapiValue>::to_napi_value(env, #ret)
|
2021-10-25 01:00:31 +09:00
|
|
|
}
|
2021-11-26 00:35:50 +09:00
|
|
|
} else if is_return_self {
|
|
|
|
quote! { #ret.map(|_| cb.this) }
|
2021-10-25 01:00:31 +09:00
|
|
|
} else {
|
2021-11-26 00:35:50 +09:00
|
|
|
quote! {
|
|
|
|
match #ret {
|
|
|
|
Ok(value) => napi::bindgen_prelude::ToNapiValue::to_napi_value(env, value),
|
|
|
|
Err(err) => {
|
|
|
|
napi::bindgen_prelude::JsError::from(err).throw_into(env);
|
|
|
|
Ok(std::ptr::null_mut())
|
|
|
|
},
|
2021-10-25 01:00:31 +09:00
|
|
|
}
|
2021-09-24 15:45:27 +09:00
|
|
|
}
|
|
|
|
}
|
2021-11-26 00:35:50 +09:00
|
|
|
} else if is_return_self {
|
|
|
|
quote! { Ok(cb.this) }
|
2021-09-24 15:45:27 +09:00
|
|
|
} else {
|
2021-11-26 00:35:50 +09:00
|
|
|
quote! {
|
|
|
|
<#ty as napi::bindgen_prelude::ToNapiValue>::to_napi_value(env, #ret)
|
2021-09-24 15:45:27 +09:00
|
|
|
}
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
quote! {
|
2021-11-15 08:34:44 +09:00
|
|
|
<() as napi::bindgen_prelude::ToNapiValue>::to_napi_value(env, ())
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn gen_fn_register(&self) -> TokenStream {
|
|
|
|
if self.parent.is_some() {
|
|
|
|
quote! {}
|
|
|
|
} else {
|
|
|
|
let name_str = self.name.to_string();
|
2021-11-22 17:51:21 +09:00
|
|
|
let js_name = format!("{}\0", &self.js_name);
|
2021-09-29 21:18:29 +09:00
|
|
|
let name_len = js_name.len();
|
2021-09-23 02:29:09 +09:00
|
|
|
let module_register_name = get_register_ident(&name_str);
|
|
|
|
let intermediate_ident = get_intermediate_ident(&name_str);
|
2021-11-23 20:00:31 +09:00
|
|
|
let js_mod_ident = js_mod_to_token_stream(self.js_mod.as_ref());
|
2022-01-23 19:17:00 +09:00
|
|
|
let cb_name = Ident::new(&format!("{}_js_function", name_str), Span::call_site());
|
2021-09-23 02:29:09 +09:00
|
|
|
quote! {
|
2021-12-25 02:23:38 +09:00
|
|
|
#[allow(non_snake_case)]
|
|
|
|
#[allow(clippy::all)]
|
2021-11-22 17:51:21 +09:00
|
|
|
unsafe fn #cb_name(env: napi::bindgen_prelude::sys::napi_env) -> napi::bindgen_prelude::Result<napi::bindgen_prelude::sys::napi_value> {
|
|
|
|
let mut fn_ptr = std::ptr::null_mut();
|
|
|
|
|
|
|
|
napi::bindgen_prelude::check_status!(
|
|
|
|
napi::bindgen_prelude::sys::napi_create_function(
|
|
|
|
env,
|
|
|
|
#js_name.as_ptr() as *const _,
|
|
|
|
#name_len,
|
|
|
|
Some(#intermediate_ident),
|
|
|
|
std::ptr::null_mut(),
|
|
|
|
&mut fn_ptr,
|
|
|
|
),
|
|
|
|
"Failed to register function `{}`",
|
|
|
|
#name_str,
|
|
|
|
)?;
|
2022-03-05 15:14:32 +09:00
|
|
|
napi::bindgen_prelude::register_js_function(#js_name, #cb_name, Some(#intermediate_ident));
|
2021-11-22 17:51:21 +09:00
|
|
|
Ok(fn_ptr)
|
|
|
|
}
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
#[allow(clippy::all)]
|
|
|
|
#[allow(non_snake_case)]
|
2021-12-08 12:30:36 +09:00
|
|
|
#[cfg(all(not(test), not(feature = "noop")))]
|
2021-11-15 08:34:44 +09:00
|
|
|
#[napi::bindgen_prelude::ctor]
|
2021-09-23 02:29:09 +09:00
|
|
|
fn #module_register_name() {
|
2021-11-23 20:00:31 +09:00
|
|
|
napi::bindgen_prelude::register_module_export(#js_mod_ident, #js_name, #cb_name);
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|