// Copyright 2018-2025 the Deno authors. MIT license. use std::mem::transmute; use std::ops::Deref; use std::os::raw::c_void; use std::ptr::NonNull; use deno_core::v8; /// An FFI-opaque, nullable wrapper around v8::Local. /// rusty_v8 Local handle cannot be empty but napi_value can be. #[repr(transparent)] #[derive(Clone, Copy, Debug)] pub struct NapiValue<'s>( Option>, std::marker::PhantomData<&'s ()>, ); pub type napi_value<'s> = NapiValue<'s>; impl<'s> Deref for napi_value<'s> { type Target = Option>; fn deref(&self) -> &Self::Target { // SAFETY: It is safe to transmute `Option>` to `Option<*const T>`. // v8::Local guarantees that *const T is not null but napi_value *can* be null. unsafe { transmute::<&Self, &Self::Target>(self) } } } impl<'s, T> From> for napi_value<'s> where v8::Local<'s, T>: Into>, { fn from(v: v8::Local<'s, T>) -> Self { Self(Some(NonNull::from(&*v.into())), std::marker::PhantomData) } } impl<'s, T> From>> for napi_value<'s> where v8::Local<'s, T>: Into>, { fn from(v: Option>) -> Self { if let Some(v) = v { NapiValue::from(v) } else { Self(None, std::marker::PhantomData) } } } const _: () = { assert!( std::mem::size_of::() == std::mem::size_of::<*mut c_void>() ); // Assert "nullable pointer optimization" on napi_value unsafe { type Src<'a> = napi_value<'a>; type Dst = usize; assert!(std::mem::size_of::() == std::mem::size_of::()); union Transmute<'a> { src: Src<'a>, dst: Dst, } Transmute { src: NapiValue(None, std::marker::PhantomData), } .dst }; };