2022-04-02 16:19:53 +09:00
|
|
|
use std::cell::Cell;
|
2021-11-05 19:31:36 +09:00
|
|
|
use std::ffi::c_void;
|
|
|
|
use std::ptr;
|
2022-04-02 16:19:53 +09:00
|
|
|
use std::rc::Rc;
|
2021-11-05 19:31:36 +09:00
|
|
|
use std::sync::atomic::{AtomicBool, Ordering};
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
use crate::{bindgen_prelude::*, check_status, sys, Result};
|
2021-11-05 19:31:36 +09:00
|
|
|
|
2022-03-05 15:14:32 +09:00
|
|
|
thread_local! {
|
|
|
|
#[doc(hidden)]
|
|
|
|
/// Determined is `constructor` called from Class `factory`
|
|
|
|
pub static ___CALL_FROM_FACTORY: AtomicBool = AtomicBool::new(false);
|
|
|
|
}
|
2021-09-23 02:29:09 +09:00
|
|
|
|
|
|
|
pub struct CallbackInfo<const N: usize> {
|
|
|
|
env: sys::napi_env,
|
2021-11-25 23:31:11 +09:00
|
|
|
pub this: sys::napi_value,
|
2021-11-05 19:31:36 +09:00
|
|
|
pub args: [sys::napi_value; N],
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<const N: usize> CallbackInfo<N> {
|
2022-03-05 22:29:57 +09:00
|
|
|
#[allow(clippy::not_unsafe_ptr_arg_deref)]
|
2021-09-23 02:29:09 +09:00
|
|
|
pub fn new(
|
|
|
|
env: sys::napi_env,
|
|
|
|
callback_info: sys::napi_callback_info,
|
|
|
|
required_argc: Option<usize>,
|
|
|
|
) -> Result<Self> {
|
|
|
|
let mut this = ptr::null_mut();
|
|
|
|
let mut args = [ptr::null_mut(); N];
|
|
|
|
let mut argc = N;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
check_status!(
|
|
|
|
sys::napi_get_cb_info(
|
|
|
|
env,
|
|
|
|
callback_info,
|
|
|
|
&mut argc,
|
|
|
|
args.as_mut_ptr(),
|
|
|
|
&mut this,
|
|
|
|
ptr::null_mut(),
|
|
|
|
),
|
|
|
|
"Failed to initialize napi function call."
|
|
|
|
)?;
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Some(required_argc) = required_argc {
|
|
|
|
if required_argc > argc {
|
|
|
|
return Err(Error::new(
|
|
|
|
Status::InvalidArg,
|
|
|
|
format!(
|
|
|
|
"{} arguments required by received {}.",
|
|
|
|
required_argc, &argc
|
|
|
|
),
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(Self { env, this, args })
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_arg(&self, index: usize) -> sys::napi_value {
|
2021-09-24 15:45:27 +09:00
|
|
|
self.args[index]
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn this(&self) -> sys::napi_value {
|
|
|
|
self.this
|
|
|
|
}
|
|
|
|
|
2022-03-21 17:36:06 +09:00
|
|
|
pub fn construct<T: 'static>(&self, js_name: &str, obj: T) -> Result<sys::napi_value> {
|
2021-09-23 02:29:09 +09:00
|
|
|
let obj = Box::new(obj);
|
|
|
|
let this = self.this();
|
2022-03-21 17:36:06 +09:00
|
|
|
let value_ref = Box::into_raw(obj) as *mut c_void;
|
|
|
|
let mut object_ref = ptr::null_mut();
|
2022-04-02 16:19:53 +09:00
|
|
|
let initial_finalize: Box<dyn FnOnce()> = Box::new(|| {});
|
|
|
|
let finalize_callbacks_ptr = Rc::into_raw(Rc::new(Cell::new(Box::into_raw(initial_finalize))));
|
2021-09-23 02:29:09 +09:00
|
|
|
unsafe {
|
|
|
|
check_status!(
|
|
|
|
sys::napi_wrap(
|
|
|
|
self.env,
|
|
|
|
this,
|
2022-03-21 17:36:06 +09:00
|
|
|
value_ref,
|
2021-09-23 02:29:09 +09:00
|
|
|
Some(raw_finalize_unchecked::<T>),
|
|
|
|
ptr::null_mut(),
|
2022-03-21 17:36:06 +09:00
|
|
|
&mut object_ref
|
2021-09-23 02:29:09 +09:00
|
|
|
),
|
|
|
|
"Failed to initialize class `{}`",
|
|
|
|
js_name,
|
|
|
|
)?;
|
|
|
|
};
|
|
|
|
|
2022-04-02 16:19:53 +09:00
|
|
|
Reference::<T>::add_ref(value_ref, (value_ref, object_ref, finalize_callbacks_ptr));
|
2021-09-23 02:29:09 +09:00
|
|
|
Ok(this)
|
|
|
|
}
|
|
|
|
|
2022-03-21 17:36:06 +09:00
|
|
|
pub fn factory<T: 'static>(&self, js_name: &str, obj: T) -> Result<sys::napi_value> {
|
2021-11-05 19:31:36 +09:00
|
|
|
let this = self.this();
|
|
|
|
let mut instance = ptr::null_mut();
|
|
|
|
unsafe {
|
2022-03-05 15:14:32 +09:00
|
|
|
___CALL_FROM_FACTORY.with(|inner| inner.store(true, Ordering::Relaxed));
|
2021-11-05 19:31:36 +09:00
|
|
|
let status = sys::napi_new_instance(self.env, this, 0, ptr::null_mut(), &mut instance);
|
2022-03-05 15:14:32 +09:00
|
|
|
___CALL_FROM_FACTORY.with(|inner| inner.store(false, Ordering::Relaxed));
|
2021-11-05 19:31:36 +09:00
|
|
|
// Error thrown in `constructor`
|
|
|
|
if status == sys::Status::napi_pending_exception {
|
|
|
|
let mut exception = ptr::null_mut();
|
|
|
|
sys::napi_get_and_clear_last_exception(self.env, &mut exception);
|
|
|
|
sys::napi_throw(self.env, exception);
|
|
|
|
return Ok(ptr::null_mut());
|
|
|
|
}
|
2022-04-02 16:19:53 +09:00
|
|
|
let obj = Box::new(obj);
|
|
|
|
let initial_finalize: Box<dyn FnOnce()> = Box::new(|| {});
|
|
|
|
let finalize_callbacks_ptr =
|
|
|
|
Rc::into_raw(Rc::new(Cell::new(Box::into_raw(initial_finalize))));
|
2022-03-21 17:36:06 +09:00
|
|
|
let mut object_ref = ptr::null_mut();
|
|
|
|
let value_ref = Box::into_raw(obj) as *mut c_void;
|
2021-11-05 19:31:36 +09:00
|
|
|
check_status!(
|
|
|
|
sys::napi_wrap(
|
|
|
|
self.env,
|
|
|
|
instance,
|
2022-03-21 17:36:06 +09:00
|
|
|
value_ref,
|
2021-11-05 19:31:36 +09:00
|
|
|
Some(raw_finalize_unchecked::<T>),
|
|
|
|
ptr::null_mut(),
|
2022-03-21 17:36:06 +09:00
|
|
|
&mut object_ref
|
2021-11-05 19:31:36 +09:00
|
|
|
),
|
|
|
|
"Failed to initialize class `{}`",
|
|
|
|
js_name,
|
|
|
|
)?;
|
2022-03-21 17:36:06 +09:00
|
|
|
|
2022-04-02 16:19:53 +09:00
|
|
|
Reference::<T>::add_ref(value_ref, (value_ref, object_ref, finalize_callbacks_ptr));
|
2021-11-05 19:31:36 +09:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(instance)
|
|
|
|
}
|
|
|
|
|
2021-09-23 02:29:09 +09:00
|
|
|
pub fn unwrap_borrow_mut<T>(&mut self) -> Result<&'static mut T>
|
|
|
|
where
|
|
|
|
T: FromNapiMutRef + TypeName,
|
|
|
|
{
|
2022-04-02 16:19:53 +09:00
|
|
|
unsafe { self.unwrap_raw::<T>() }.map(|raw| Box::leak(unsafe { Box::from_raw(raw) }))
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn unwrap_borrow<T>(&mut self) -> Result<&'static T>
|
|
|
|
where
|
|
|
|
T: FromNapiRef + TypeName,
|
2022-04-02 16:19:53 +09:00
|
|
|
{
|
|
|
|
unsafe { self.unwrap_raw::<T>() }
|
|
|
|
.map(|raw| Box::leak(unsafe { Box::from_raw(raw) }) as &'static T)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[inline]
|
|
|
|
pub unsafe fn unwrap_raw<T>(&mut self) -> Result<*mut T>
|
|
|
|
where
|
|
|
|
T: TypeName,
|
2021-09-23 02:29:09 +09:00
|
|
|
{
|
|
|
|
let mut wrapped_val: *mut c_void = std::ptr::null_mut();
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
check_status!(
|
|
|
|
sys::napi_unwrap(self.env, self.this, &mut wrapped_val),
|
2022-04-02 16:19:53 +09:00
|
|
|
"Failed to unwrap exclusive reference of `{}` type from napi value",
|
2021-09-23 02:29:09 +09:00
|
|
|
T::type_name(),
|
|
|
|
)?;
|
|
|
|
|
2022-04-02 16:19:53 +09:00
|
|
|
Ok(wrapped_val as *mut T)
|
2021-09-23 02:29:09 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|