// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license. #![allow(non_upper_case_globals)] #![deny(unsafe_op_in_unsafe_fn)] const NAPI_VERSION: u32 = 9; use crate::*; use libc::INT_MAX; use super::util::check_new_from_utf8; use super::util::check_new_from_utf8_len; use super::util::get_array_buffer_ptr; use super::util::make_external_backing_store; use super::util::napi_clear_last_error; use super::util::napi_set_last_error; use super::util::v8_name_from_property_descriptor; use crate::check_arg; use crate::check_env; use crate::function::create_function; use crate::function::create_function_template; use crate::function::CallbackInfo; use napi_sym::napi_sym; use std::ptr::NonNull; #[derive(Debug, Clone, Copy, PartialEq)] enum ReferenceOwnership { Runtime, Userland, } enum ReferenceState { Strong(v8::Global<v8::Value>), Weak(v8::Weak<v8::Value>), } struct Reference { env: *mut Env, state: ReferenceState, ref_count: u32, ownership: ReferenceOwnership, finalize_cb: Option<napi_finalize>, finalize_data: *mut c_void, finalize_hint: *mut c_void, } impl Reference { fn new( env: *mut Env, value: v8::Local<v8::Value>, initial_ref_count: u32, ownership: ReferenceOwnership, finalize_cb: Option<napi_finalize>, finalize_data: *mut c_void, finalize_hint: *mut c_void, ) -> Box<Self> { let isolate = unsafe { (*env).isolate() }; let mut reference = Box::new(Reference { env, state: ReferenceState::Strong(v8::Global::new(isolate, value)), ref_count: initial_ref_count, ownership, finalize_cb, finalize_data, finalize_hint, }); if initial_ref_count == 0 { reference.set_weak(); } reference } fn ref_(&mut self) -> u32 { self.ref_count += 1; if self.ref_count == 1 { self.set_strong(); } self.ref_count } fn unref(&mut self) -> u32 { let old_ref_count = self.ref_count; if self.ref_count > 0 { self.ref_count -= 1; } if old_ref_count == 1 && self.ref_count == 0 { self.set_weak(); } self.ref_count } fn reset(&mut self) { self.finalize_cb = None; self.finalize_data = std::ptr::null_mut(); self.finalize_hint = std::ptr::null_mut(); } fn set_strong(&mut self) { if let ReferenceState::Weak(w) = &self.state { let isolate = unsafe { (*self.env).isolate() }; if let Some(g) = w.to_global(isolate) { self.state = ReferenceState::Strong(g); } } } fn set_weak(&mut self) { let reference = self as *mut Reference; if let ReferenceState::Strong(g) = &self.state { let cb = Box::new(move |_: &mut v8::Isolate| { Reference::weak_callback(reference) }); let isolate = unsafe { (*self.env).isolate() }; self.state = ReferenceState::Weak(v8::Weak::with_finalizer(isolate, g, cb)); } } fn weak_callback(reference: *mut Reference) { let reference = unsafe { &mut *reference }; let finalize_cb = reference.finalize_cb; let finalize_data = reference.finalize_data; let finalize_hint = reference.finalize_hint; reference.reset(); // copy this value before the finalize callback, since // it might free the reference (which would be a UAF) let ownership = reference.ownership; if let Some(finalize_cb) = finalize_cb { unsafe { finalize_cb(reference.env as _, finalize_data, finalize_hint); } } if ownership == ReferenceOwnership::Runtime { unsafe { drop(Reference::from_raw(reference)) } } } fn into_raw(r: Box<Reference>) -> *mut Reference { Box::into_raw(r) } unsafe fn from_raw(r: *mut Reference) -> Box<Reference> { unsafe { Box::from_raw(r) } } unsafe fn remove(r: *mut Reference) { let r = unsafe { &mut *r }; if r.ownership == ReferenceOwnership::Userland { r.reset(); } else { unsafe { drop(Reference::from_raw(r)) } } } } #[napi_sym] fn napi_get_last_error_info( env: *mut Env, result: *mut *const napi_extended_error_info, ) -> napi_status { let env = check_env!(env); check_arg!(env, result); if env.last_error.error_code == napi_ok { napi_clear_last_error(env); } else { env.last_error.error_message = ERROR_MESSAGES[env.last_error.error_code as usize].as_ptr(); } unsafe { *result = &env.last_error; } napi_ok } #[napi_sym] fn napi_create_function<'s>( env: &'s mut Env, name: *const c_char, length: usize, cb: Option<napi_callback>, cb_info: napi_callback_info, result: *mut napi_value<'s>, ) -> napi_status { let env_ptr = env as *mut Env; check_arg!(env, result); check_arg!(env, cb); let name = if !name.is_null() { match unsafe { check_new_from_utf8_len(env, name, length) } { Ok(s) => Some(s), Err(status) => return status, } } else { None }; unsafe { *result = create_function(&mut env.scope(), env_ptr, name, cb.unwrap(), cb_info) .into(); } napi_ok } #[napi_sym] #[allow(clippy::too_many_arguments)] fn napi_define_class<'s>( env: &'s mut Env, utf8name: *const c_char, length: usize, constructor: Option<napi_callback>, callback_data: *mut c_void, property_count: usize, properties: *const napi_property_descriptor, result: *mut napi_value<'s>, ) -> napi_status { let env_ptr = env as *mut Env; check_arg!(env, result); check_arg!(env, constructor); if property_count > 0 { check_arg!(env, properties); } let name = match unsafe { check_new_from_utf8_len(env, utf8name, length) } { Ok(string) => string, Err(status) => return status, }; let tpl = create_function_template( &mut env.scope(), env_ptr, Some(name), constructor.unwrap(), callback_data, ); let napi_properties: &[napi_property_descriptor] = if property_count > 0 { unsafe { std::slice::from_raw_parts(properties, property_count) } } else { &[] }; let mut static_property_count = 0; for p in napi_properties { if p.attributes & napi_static != 0 { // Will be handled below static_property_count += 1; continue; } let name = match unsafe { v8_name_from_property_descriptor(env_ptr, p) } { Ok(name) => name, Err(status) => return status, }; let mut accessor_property = v8::PropertyAttribute::NONE; if p.attributes & napi_enumerable == 0 { accessor_property = accessor_property | v8::PropertyAttribute::DONT_ENUM; } if p.attributes & napi_configurable == 0 { accessor_property = accessor_property | v8::PropertyAttribute::DONT_DELETE; } if p.getter.is_some() || p.setter.is_some() { let getter = p.getter.map(|g| { create_function_template(&mut env.scope(), env_ptr, None, g, p.data) }); let setter = p.setter.map(|s| { create_function_template(&mut env.scope(), env_ptr, None, s, p.data) }); if getter.is_some() && setter.is_some() && (p.attributes & napi_writable) == 0 { accessor_property = accessor_property | v8::PropertyAttribute::READ_ONLY; } let proto = tpl.prototype_template(&mut env.scope()); proto.set_accessor_property(name, getter, setter, accessor_property); } else if let Some(method) = p.method { let function = create_function_template( &mut env.scope(), env_ptr, None, method, p.data, ); let proto = tpl.prototype_template(&mut env.scope()); proto.set_with_attr(name, function.into(), accessor_property); } else { let proto = tpl.prototype_template(&mut env.scope()); if (p.attributes & napi_writable) == 0 { accessor_property = accessor_property | v8::PropertyAttribute::READ_ONLY; } proto.set_with_attr(name, p.value.unwrap().into(), accessor_property); } } let value: v8::Local<v8::Value> = tpl.get_function(&mut env.scope()).unwrap().into(); unsafe { *result = value.into(); } if static_property_count > 0 { let mut static_descriptors = Vec::with_capacity(static_property_count); for p in napi_properties { if p.attributes & napi_static != 0 { static_descriptors.push(*p); } } crate::status_call!(unsafe { napi_define_properties( env_ptr, *result, static_descriptors.len(), static_descriptors.as_ptr(), ) }); } napi_ok } #[napi_sym] fn napi_get_property_names( env: *mut Env, object: napi_value, result: *mut napi_value, ) -> napi_status { unsafe { napi_get_all_property_names( env, object, napi_key_include_prototypes, napi_key_enumerable | napi_key_skip_symbols, napi_key_numbers_to_strings, result, ) } } #[napi_sym] fn napi_get_all_property_names<'s>( env: &'s mut Env, object: napi_value, key_mode: napi_key_collection_mode, key_filter: napi_key_filter, key_conversion: napi_key_conversion, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let scope = &mut env.scope(); let Some(obj) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let mut filter = v8::PropertyFilter::ALL_PROPERTIES; if key_filter & napi_key_writable != 0 { filter = filter | v8::PropertyFilter::ONLY_WRITABLE; } if key_filter & napi_key_enumerable != 0 { filter = filter | v8::PropertyFilter::ONLY_ENUMERABLE; } if key_filter & napi_key_configurable != 0 { filter = filter | v8::PropertyFilter::ONLY_CONFIGURABLE; } if key_filter & napi_key_skip_strings != 0 { filter = filter | v8::PropertyFilter::SKIP_STRINGS; } if key_filter & napi_key_skip_symbols != 0 { filter = filter | v8::PropertyFilter::SKIP_SYMBOLS; } let key_mode = match key_mode { napi_key_include_prototypes => v8::KeyCollectionMode::IncludePrototypes, napi_key_own_only => v8::KeyCollectionMode::OwnOnly, _ => return napi_invalid_arg, }; let key_conversion = match key_conversion { napi_key_keep_numbers => v8::KeyConversionMode::KeepNumbers, napi_key_numbers_to_strings => v8::KeyConversionMode::ConvertToString, _ => return napi_invalid_arg, }; let filter = v8::GetPropertyNamesArgsBuilder::new() .mode(key_mode) .property_filter(filter) .index_filter(v8::IndexFilter::IncludeIndices) .key_conversion(key_conversion) .build(); let property_names = match obj.get_property_names(scope, filter) { Some(n) => n, None => return napi_generic_failure, }; unsafe { *result = property_names.into(); } napi_ok } #[napi_sym] fn napi_set_property( env: &mut Env, object: napi_value, key: napi_value, value: napi_value, ) -> napi_status { check_arg!(env, key); check_arg!(env, value); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; if object.set(scope, key.unwrap(), value.unwrap()).is_none() { return napi_generic_failure; }; napi_ok } #[napi_sym] fn napi_has_property( env: &mut Env, object: napi_value, key: napi_value, result: *mut bool, ) -> napi_status { check_arg!(env, key); check_arg!(env, result); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(has) = object.has(scope, key.unwrap()) else { return napi_generic_failure; }; unsafe { *result = has; } napi_ok } #[napi_sym] fn napi_get_property<'s>( env: &'s mut Env, object: napi_value, key: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, key); check_arg!(env, result); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(value) = object.get(scope, key.unwrap()) else { return napi_generic_failure; }; unsafe { *result = value.into(); } napi_ok } #[napi_sym] fn napi_delete_property( env: &mut Env, object: napi_value, key: napi_value, result: *mut bool, ) -> napi_status { check_arg!(env, key); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(deleted) = object.delete(scope, key.unwrap()) else { return napi_generic_failure; }; if !result.is_null() { unsafe { *result = deleted; } } napi_ok } #[napi_sym] fn napi_has_own_property( env: &mut Env, object: napi_value, key: napi_value, result: *mut bool, ) -> napi_status { check_arg!(env, key); check_arg!(env, result); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Ok(key) = v8::Local::<v8::Name>::try_from(key.unwrap()) else { return napi_name_expected; }; let Some(has_own) = object.has_own_property(scope, key) else { return napi_generic_failure; }; unsafe { *result = has_own; } napi_ok } #[napi_sym] fn napi_has_named_property<'s>( env: &'s mut Env, object: napi_value<'s>, utf8name: *const c_char, result: *mut bool, ) -> napi_status { let env_ptr = env as *mut Env; check_arg!(env, result); let Some(object) = object.and_then(|o| o.to_object(&mut env.scope())) else { return napi_object_expected; }; let key = match unsafe { check_new_from_utf8(env_ptr, utf8name) } { Ok(key) => key, Err(status) => return status, }; let Some(has_property) = object.has(&mut env.scope(), key.into()) else { return napi_generic_failure; }; unsafe { *result = has_property; } napi_ok } #[napi_sym] fn napi_set_named_property<'s>( env: &'s mut Env, object: napi_value<'s>, utf8name: *const c_char, value: napi_value<'s>, ) -> napi_status { check_arg!(env, value); let env_ptr = env as *mut Env; let Some(object) = object.and_then(|o| o.to_object(&mut env.scope())) else { return napi_object_expected; }; let key = match unsafe { check_new_from_utf8(env_ptr, utf8name) } { Ok(key) => key, Err(status) => return status, }; let value = value.unwrap(); if !object .set(&mut env.scope(), key.into(), value) .unwrap_or(false) { return napi_generic_failure; } napi_ok } #[napi_sym] fn napi_get_named_property<'s>( env: &'s mut Env, object: napi_value<'s>, utf8name: *const c_char, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let env_ptr = env as *mut Env; let Some(object) = object.and_then(|o| o.to_object(&mut env.scope())) else { return napi_object_expected; }; let key = match unsafe { check_new_from_utf8(env_ptr, utf8name) } { Ok(key) => key, Err(status) => return status, }; let Some(value) = object.get(&mut env.scope(), key.into()) else { return napi_generic_failure; }; unsafe { *result = value.into(); } napi_ok } #[napi_sym] fn napi_set_element<'s>( env: &'s mut Env, object: napi_value<'s>, index: u32, value: napi_value<'s>, ) -> napi_status { check_arg!(env, value); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; if !object .set_index(scope, index, value.unwrap()) .unwrap_or(false) { return napi_generic_failure; } napi_ok } #[napi_sym] fn napi_has_element( env: &mut Env, object: napi_value, index: u32, result: *mut bool, ) -> napi_status { check_arg!(env, result); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(has) = object.has_index(scope, index) else { return napi_generic_failure; }; unsafe { *result = has; } napi_ok } #[napi_sym] fn napi_get_element<'s>( env: &'s mut Env, object: napi_value, index: u32, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(value) = object.get_index(scope, index) else { return napi_generic_failure; }; unsafe { *result = value.into(); } napi_ok } #[napi_sym] fn napi_delete_element( env: &mut Env, object: napi_value, index: u32, result: *mut bool, ) -> napi_status { let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(deleted) = object.delete_index(scope, index) else { return napi_generic_failure; }; if !result.is_null() { unsafe { *result = deleted; } } napi_ok } #[napi_sym] fn napi_define_properties( env: &mut Env, object: napi_value, property_count: usize, properties: *const napi_property_descriptor, ) -> napi_status { let env_ptr = env as *mut Env; if property_count > 0 { check_arg!(env, properties); } let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let properties = if property_count == 0 { &[] } else { unsafe { std::slice::from_raw_parts(properties, property_count) } }; for property in properties { let property_name = match unsafe { v8_name_from_property_descriptor(env_ptr, property) } { Ok(name) => name, Err(status) => return status, }; let writable = property.attributes & napi_writable != 0; let enumerable = property.attributes & napi_enumerable != 0; let configurable = property.attributes & napi_configurable != 0; if property.getter.is_some() || property.setter.is_some() { let local_getter: v8::Local<v8::Value> = if let Some(getter) = property.getter { create_function(&mut env.scope(), env_ptr, None, getter, property.data) .into() } else { v8::undefined(scope).into() }; let local_setter: v8::Local<v8::Value> = if let Some(setter) = property.setter { create_function(&mut env.scope(), env_ptr, None, setter, property.data) .into() } else { v8::undefined(scope).into() }; let mut desc = v8::PropertyDescriptor::new_from_get_set(local_getter, local_setter); desc.set_enumerable(enumerable); desc.set_configurable(configurable); if !object .define_property(scope, property_name, &desc) .unwrap_or(false) { return napi_invalid_arg; } } else if let Some(method) = property.method { let method: v8::Local<v8::Value> = { let function = create_function( &mut env.scope(), env_ptr, None, method, property.data, ); function.into() }; let mut desc = v8::PropertyDescriptor::new_from_value_writable(method, writable); desc.set_enumerable(enumerable); desc.set_configurable(configurable); if !object .define_property(scope, property_name, &desc) .unwrap_or(false) { return napi_generic_failure; } } else { let value = property.value.unwrap(); if enumerable & writable & configurable { if !object .create_data_property(scope, property_name, value) .unwrap_or(false) { return napi_invalid_arg; } } else { let mut desc = v8::PropertyDescriptor::new_from_value_writable(value, writable); desc.set_enumerable(enumerable); desc.set_configurable(configurable); if !object .define_property(scope, property_name, &desc) .unwrap_or(false) { return napi_invalid_arg; } } } } napi_ok } #[napi_sym] fn napi_object_freeze(env: &mut Env, object: napi_value) -> napi_status { let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; if !object .set_integrity_level(scope, v8::IntegrityLevel::Frozen) .unwrap_or(false) { return napi_generic_failure; } napi_ok } #[napi_sym] fn napi_object_seal(env: &mut Env, object: napi_value) -> napi_status { let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; if !object .set_integrity_level(scope, v8::IntegrityLevel::Sealed) .unwrap_or(false) { return napi_generic_failure; } napi_ok } #[napi_sym] fn napi_is_array( env: *mut Env, value: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); let value = value.unwrap(); unsafe { *result = value.is_array(); } napi_clear_last_error(env) } #[napi_sym] fn napi_get_array_length( env: &mut Env, value: napi_value, result: *mut u32, ) -> napi_status { check_arg!(env, value); check_arg!(env, result); let value = value.unwrap(); match v8::Local::<v8::Array>::try_from(value) { Ok(array) => { unsafe { *result = array.length(); } napi_ok } Err(_) => napi_array_expected, } } #[napi_sym] fn napi_strict_equals( env: &mut Env, lhs: napi_value, rhs: napi_value, result: *mut bool, ) -> napi_status { check_arg!(env, lhs); check_arg!(env, rhs); check_arg!(env, result); unsafe { *result = lhs.unwrap().strict_equals(rhs.unwrap()); } napi_ok } #[napi_sym] fn napi_get_prototype<'s>( env: &'s mut Env, object: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let scope = &mut env.scope(); let Some(object) = object.and_then(|o| o.to_object(scope)) else { return napi_object_expected; }; let Some(proto) = object.get_prototype(scope) else { return napi_generic_failure; }; unsafe { *result = proto.into(); } napi_ok } #[napi_sym] fn napi_create_object( env_ptr: *mut Env, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Object::new(&mut env.scope()).into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_create_array( env_ptr: *mut Env, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Array::new(&mut env.scope(), 0).into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_create_array_with_length( env_ptr: *mut Env, length: usize, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Array::new(&mut env.scope(), length as _).into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_create_string_latin1( env_ptr: *mut Env, string: *const c_char, length: usize, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); if length > 0 { check_arg!(env, string); } crate::return_status_if_false!( env, (length == NAPI_AUTO_LENGTH) || length <= INT_MAX as _, napi_invalid_arg ); let buffer = if length > 0 { unsafe { std::slice::from_raw_parts( string as _, if length == NAPI_AUTO_LENGTH { std::ffi::CStr::from_ptr(string).to_bytes().len() } else { length }, ) } } else { &[] }; let Some(string) = v8::String::new_from_one_byte( &mut env.scope(), buffer, v8::NewStringType::Normal, ) else { return napi_set_last_error(env_ptr, napi_generic_failure); }; unsafe { *result = string.into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] pub(crate) fn napi_create_string_utf8( env_ptr: *mut Env, string: *const c_char, length: usize, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); if length > 0 { check_arg!(env, string); } crate::return_status_if_false!( env, (length == NAPI_AUTO_LENGTH) || length <= INT_MAX as _, napi_invalid_arg ); let buffer = if length > 0 { unsafe { std::slice::from_raw_parts( string as _, if length == NAPI_AUTO_LENGTH { std::ffi::CStr::from_ptr(string).to_bytes().len() } else { length }, ) } } else { &[] }; let Some(string) = v8::String::new_from_utf8( &mut env.scope(), buffer, v8::NewStringType::Normal, ) else { return napi_set_last_error(env_ptr, napi_generic_failure); }; unsafe { *result = string.into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_create_string_utf16( env_ptr: *mut Env, string: *const u16, length: usize, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); if length > 0 { check_arg!(env, string); } crate::return_status_if_false!( env, (length == NAPI_AUTO_LENGTH) || length <= INT_MAX as _, napi_invalid_arg ); let buffer = if length > 0 { unsafe { std::slice::from_raw_parts( string, if length == NAPI_AUTO_LENGTH { let mut length = 0; while *(string.add(length)) != 0 { length += 1; } length } else { length }, ) } } else { &[] }; let Some(string) = v8::String::new_from_two_byte( &mut env.scope(), buffer, v8::NewStringType::Normal, ) else { return napi_set_last_error(env_ptr, napi_generic_failure); }; unsafe { *result = string.into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn node_api_create_external_string_latin1( env_ptr: *mut Env, string: *const c_char, length: usize, nogc_finalize_callback: Option<napi_finalize>, finalize_hint: *mut c_void, result: *mut napi_value, copied: *mut bool, ) -> napi_status { let status = unsafe { napi_create_string_latin1(env_ptr, string, length, result) }; if status == napi_ok { unsafe { *copied = true; } if let Some(finalize) = nogc_finalize_callback { unsafe { finalize(env_ptr as napi_env, string as *mut c_void, finalize_hint); } } } status } #[napi_sym] fn node_api_create_external_string_utf16( env_ptr: *mut Env, string: *const u16, length: usize, nogc_finalize_callback: Option<napi_finalize>, finalize_hint: *mut c_void, result: *mut napi_value, copied: *mut bool, ) -> napi_status { let status = unsafe { napi_create_string_utf16(env_ptr, string, length, result) }; if status == napi_ok { unsafe { *copied = true; } if let Some(finalize) = nogc_finalize_callback { unsafe { finalize(env_ptr as napi_env, string as *mut c_void, finalize_hint); } } } status } #[napi_sym] fn node_api_create_property_key_utf16( env_ptr: *mut Env, string: *const u16, length: usize, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); if length > 0 { check_arg!(env, string); } crate::return_status_if_false!( env, (length == NAPI_AUTO_LENGTH) || length <= INT_MAX as _, napi_invalid_arg ); let buffer = if length > 0 { unsafe { std::slice::from_raw_parts( string, if length == NAPI_AUTO_LENGTH { let mut length = 0; while *(string.add(length)) != 0 { length += 1; } length } else { length }, ) } } else { &[] }; let Some(string) = v8::String::new_from_two_byte( &mut env.scope(), buffer, v8::NewStringType::Internalized, ) else { return napi_set_last_error(env_ptr, napi_generic_failure); }; unsafe { *result = string.into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_create_double( env_ptr: *mut Env, value: f64, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Number::new(&mut env.scope(), value).into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_int32( env_ptr: *mut Env, value: i32, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Integer::new(&mut env.scope(), value).into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_uint32( env_ptr: *mut Env, value: u32, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Integer::new_from_unsigned(&mut env.scope(), value).into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_int64( env_ptr: *mut Env, value: i64, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::Number::new(&mut env.scope(), value as _).into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_bigint_int64( env_ptr: *mut Env, value: i64, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::BigInt::new_from_i64(&mut env.scope(), value).into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_bigint_uint64( env_ptr: *mut Env, value: u64, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = v8::BigInt::new_from_u64(&mut env.scope(), value).into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_bigint_words<'s>( env: &'s mut Env, sign_bit: bool, word_count: usize, words: *const u64, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, words); check_arg!(env, result); if word_count > INT_MAX as _ { return napi_invalid_arg; } match v8::BigInt::new_from_words(&mut env.scope(), sign_bit, unsafe { std::slice::from_raw_parts(words, word_count) }) { Some(value) => unsafe { *result = value.into(); }, None => { return napi_generic_failure; } } napi_ok } #[napi_sym] fn napi_get_boolean( env: *mut Env, value: bool, result: *mut napi_value, ) -> napi_status { let env = check_env!(env); check_arg!(env, result); unsafe { *result = v8::Boolean::new(env.isolate(), value).into(); } return napi_clear_last_error(env); } #[napi_sym] fn napi_create_symbol( env_ptr: *mut Env, description: napi_value, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); let description = if let Some(d) = *description { let Some(d) = d.to_string(&mut env.scope()) else { return napi_set_last_error(env, napi_string_expected); }; Some(d) } else { None }; unsafe { *result = v8::Symbol::new(&mut env.scope(), description).into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn node_api_symbol_for( env: *mut Env, utf8description: *const c_char, length: usize, result: *mut napi_value, ) -> napi_status { { let env = check_env!(env); check_arg!(env, result); let description_string = match unsafe { check_new_from_utf8_len(env, utf8description, length) } { Ok(s) => s, Err(status) => return napi_set_last_error(env, status), }; unsafe { *result = v8::Symbol::for_key(&mut env.scope(), description_string).into(); } } napi_clear_last_error(env) } macro_rules! napi_create_error_impl { ($env_ptr:ident, $code:ident, $msg:ident, $result:ident, $error:ident) => {{ let env_ptr = $env_ptr; let code = $code; let msg = $msg; let result = $result; let env = check_env!(env_ptr); check_arg!(env, msg); check_arg!(env, result); let Some(message) = msg.and_then(|v| v8::Local::<v8::String>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_string_expected); }; let error = v8::Exception::$error(&mut env.scope(), message); if let Some(code) = *code { let error_obj: v8::Local<v8::Object> = error.try_into().unwrap(); let code_key = v8::String::new(&mut env.scope(), "code").unwrap(); if !error_obj .set(&mut env.scope(), code_key.into(), code) .unwrap_or(false) { return napi_set_last_error(env_ptr, napi_generic_failure); } } unsafe { *result = error.into(); } return napi_clear_last_error(env_ptr); }}; } #[napi_sym] fn napi_create_error( env_ptr: *mut Env, code: napi_value, msg: napi_value, result: *mut napi_value, ) -> napi_status { napi_create_error_impl!(env_ptr, code, msg, result, error) } #[napi_sym] fn napi_create_type_error( env_ptr: *mut Env, code: napi_value, msg: napi_value, result: *mut napi_value, ) -> napi_status { napi_create_error_impl!(env_ptr, code, msg, result, type_error) } #[napi_sym] fn napi_create_range_error( env_ptr: *mut Env, code: napi_value, msg: napi_value, result: *mut napi_value, ) -> napi_status { napi_create_error_impl!(env_ptr, code, msg, result, range_error) } #[napi_sym] fn node_api_create_syntax_error( env_ptr: *mut Env, code: napi_value, msg: napi_value, result: *mut napi_value, ) -> napi_status { napi_create_error_impl!(env_ptr, code, msg, result, syntax_error) } pub fn get_value_type(value: v8::Local<v8::Value>) -> Option<napi_valuetype> { if value.is_undefined() { Some(napi_undefined) } else if value.is_null() { Some(napi_null) } else if value.is_external() { Some(napi_external) } else if value.is_boolean() { Some(napi_boolean) } else if value.is_number() { Some(napi_number) } else if value.is_big_int() { Some(napi_bigint) } else if value.is_string() { Some(napi_string) } else if value.is_symbol() { Some(napi_symbol) } else if value.is_function() { Some(napi_function) } else if value.is_object() { Some(napi_object) } else { None } } #[napi_sym] fn napi_typeof( env: *mut Env, value: napi_value, result: *mut napi_valuetype, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); let Some(ty) = get_value_type(value.unwrap()) else { return napi_set_last_error(env, napi_invalid_arg); }; unsafe { *result = ty; } napi_clear_last_error(env) } #[napi_sym] fn napi_get_undefined(env: *mut Env, result: *mut napi_value) -> napi_status { let env = check_env!(env); check_arg!(env, result); unsafe { *result = v8::undefined(&mut env.scope()).into(); } return napi_clear_last_error(env); } #[napi_sym] fn napi_get_null(env: *mut Env, result: *mut napi_value) -> napi_status { let env = check_env!(env); check_arg!(env, result); unsafe { *result = v8::null(&mut env.scope()).into(); } return napi_clear_last_error(env); } #[napi_sym] fn napi_get_cb_info( env: *mut Env, cbinfo: napi_callback_info, argc: *mut i32, argv: *mut napi_value, this_arg: *mut napi_value, data: *mut *mut c_void, ) -> napi_status { let env = check_env!(env); check_arg!(env, cbinfo); let cbinfo: &CallbackInfo = unsafe { &*(cbinfo as *const CallbackInfo) }; let args = unsafe { &*(cbinfo.args as *const v8::FunctionCallbackArguments) }; if !argv.is_null() { check_arg!(env, argc); let argc = unsafe { *argc as usize }; for i in 0..argc { let arg = args.get(i as _); unsafe { *argv.add(i) = arg.into(); } } } if !argc.is_null() { unsafe { *argc = args.length(); } } if !this_arg.is_null() { unsafe { *this_arg = args.this().into(); } } if !data.is_null() { unsafe { *data = cbinfo.cb_info; } } napi_clear_last_error(env); napi_ok } #[napi_sym] fn napi_get_new_target( env: *mut Env, cbinfo: napi_callback_info, result: *mut napi_value, ) -> napi_status { let env = check_env!(env); check_arg!(env, cbinfo); check_arg!(env, result); let cbinfo: &CallbackInfo = unsafe { &*(cbinfo as *const CallbackInfo) }; let args = unsafe { &*(cbinfo.args as *const v8::FunctionCallbackArguments) }; unsafe { *result = args.new_target().into(); } return napi_clear_last_error(env); } #[napi_sym] fn napi_call_function<'s>( env: &'s mut Env, recv: napi_value<'s>, func: napi_value<'s>, argc: usize, argv: *const napi_value<'s>, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, recv); let args = if argc > 0 { check_arg!(env, argv); unsafe { std::slice::from_raw_parts(argv as *mut v8::Local<v8::Value>, argc) } } else { &[] }; let Some(func) = func.and_then(|f| v8::Local::<v8::Function>::try_from(f).ok()) else { return napi_function_expected; }; let Some(v) = func.call(&mut env.scope(), recv.unwrap(), args) else { return napi_generic_failure; }; if !result.is_null() { unsafe { *result = v.into(); } } napi_ok } #[napi_sym] fn napi_get_global(env_ptr: *mut Env, result: *mut napi_value) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); let global = v8::Local::new(&mut env.scope(), &env.global); unsafe { *result = global.into(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_throw(env: *mut Env, error: napi_value) -> napi_status { let env = check_env!(env); check_arg!(env, error); if env.last_exception.is_some() { return napi_pending_exception; } let error = error.unwrap(); env.scope().throw_exception(error); let error = v8::Global::new(&mut env.scope(), error); env.last_exception = Some(error); napi_clear_last_error(env) } macro_rules! napi_throw_error_impl { ($env:ident, $code:ident, $msg:ident, $error:ident) => {{ let env = check_env!($env); let env_ptr = env as *mut Env; let code = $code; let msg = $msg; if env.last_exception.is_some() { return napi_pending_exception; } let str_ = match unsafe { check_new_from_utf8(env, msg) } { Ok(s) => s, Err(status) => return status, }; let error = v8::Exception::$error(&mut env.scope(), str_); if !code.is_null() { let error_obj: v8::Local<v8::Object> = error.try_into().unwrap(); let code = match unsafe { check_new_from_utf8(env_ptr, code) } { Ok(s) => s, Err(status) => return napi_set_last_error(env, status), }; let code_key = v8::String::new(&mut env.scope(), "code").unwrap(); if !error_obj .set(&mut env.scope(), code_key.into(), code.into()) .unwrap_or(false) { return napi_set_last_error(env, napi_generic_failure); } } env.scope().throw_exception(error); let error = v8::Global::new(&mut env.scope(), error); env.last_exception = Some(error); napi_clear_last_error(env) }}; } #[napi_sym] fn napi_throw_error( env: *mut Env, code: *const c_char, msg: *const c_char, ) -> napi_status { napi_throw_error_impl!(env, code, msg, error) } #[napi_sym] fn napi_throw_type_error( env: *mut Env, code: *const c_char, msg: *const c_char, ) -> napi_status { napi_throw_error_impl!(env, code, msg, type_error) } #[napi_sym] fn napi_throw_range_error( env: *mut Env, code: *const c_char, msg: *const c_char, ) -> napi_status { napi_throw_error_impl!(env, code, msg, range_error) } #[napi_sym] fn node_api_throw_syntax_error( env: *mut Env, code: *const c_char, msg: *const c_char, ) -> napi_status { napi_throw_error_impl!(env, code, msg, syntax_error) } #[napi_sym] fn napi_is_error( env: *mut Env, value: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); unsafe { *result = value.unwrap().is_native_error(); } return napi_clear_last_error(env); } #[napi_sym] fn napi_get_value_double( env_ptr: *mut Env, value: napi_value, result: *mut f64, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); let Some(number) = value.and_then(|v| v8::Local::<v8::Number>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_number_expected); }; unsafe { *result = number.value(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_int32( env_ptr: *mut Env, value: napi_value, result: *mut i32, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); let Some(value) = value.unwrap().int32_value(&mut env.scope()) else { return napi_set_last_error(env, napi_number_expected); }; unsafe { *result = value; } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_uint32( env_ptr: *mut Env, value: napi_value, result: *mut u32, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); let Some(value) = value.unwrap().uint32_value(&mut env.scope()) else { return napi_set_last_error(env, napi_number_expected); }; unsafe { *result = value; } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_int64( env_ptr: *mut Env, value: napi_value, result: *mut i64, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); let Some(number) = value.and_then(|v| v8::Local::<v8::Number>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_number_expected); }; let value = number.value(); unsafe { if value.is_finite() { *result = value as _; } else { *result = 0; } } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_bigint_int64( env_ptr: *mut Env, value: napi_value, result: *mut i64, lossless: *mut bool, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); check_arg!(env, lossless); let Some(bigint) = value.and_then(|v| v8::Local::<v8::BigInt>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_bigint_expected); }; let (result_, lossless_) = bigint.i64_value(); unsafe { *result = result_; *lossless = lossless_; } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_bigint_uint64( env_ptr: *mut Env, value: napi_value, result: *mut u64, lossless: *mut bool, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); check_arg!(env, lossless); let Some(bigint) = value.and_then(|v| v8::Local::<v8::BigInt>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_bigint_expected); }; let (result_, lossless_) = bigint.u64_value(); unsafe { *result = result_; *lossless = lossless_; } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_bigint_words( env_ptr: *mut Env, value: napi_value, sign_bit: *mut i32, word_count: *mut usize, words: *mut u64, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, word_count); let Some(bigint) = value.and_then(|v| v8::Local::<v8::BigInt>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_bigint_expected); }; let word_count_int; if sign_bit.is_null() && words.is_null() { word_count_int = bigint.word_count(); } else { check_arg!(env, sign_bit); check_arg!(env, words); let out_words = unsafe { std::slice::from_raw_parts_mut(words, *word_count) }; let (sign, slice_) = bigint.to_words_array(out_words); word_count_int = slice_.len(); unsafe { *sign_bit = sign as i32; } } unsafe { *word_count = word_count_int; } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_bool( env_ptr: *mut Env, value: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, result); let Some(boolean) = value.and_then(|v| v8::Local::<v8::Boolean>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_boolean_expected); }; unsafe { *result = boolean.is_true(); } return napi_clear_last_error(env_ptr); } #[napi_sym] fn napi_get_value_string_latin1( env_ptr: *mut Env, value: napi_value, buf: *mut c_char, bufsize: usize, result: *mut usize, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); let Some(value) = value.and_then(|v| v8::Local::<v8::String>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_string_expected); }; if buf.is_null() { check_arg!(env, result); unsafe { *result = value.length(); } } else if bufsize != 0 { let buffer = unsafe { std::slice::from_raw_parts_mut(buf as _, bufsize - 1) }; let copied = value.write_one_byte( &mut env.scope(), buffer, 0, v8::WriteOptions::NO_NULL_TERMINATION, ); unsafe { buf.add(copied).write(0); } if !result.is_null() { unsafe { *result = copied; } } } else if !result.is_null() { unsafe { *result = 0; } } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_get_value_string_utf8( env_ptr: *mut Env, value: napi_value, buf: *mut u8, bufsize: usize, result: *mut usize, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); let Some(value) = value.and_then(|v| v8::Local::<v8::String>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_string_expected); }; if buf.is_null() { check_arg!(env, result); unsafe { *result = value.utf8_length(env.isolate()); } } else if bufsize != 0 { let buffer = unsafe { std::slice::from_raw_parts_mut(buf as _, bufsize - 1) }; let copied = value.write_utf8( &mut env.scope(), buffer, None, v8::WriteOptions::REPLACE_INVALID_UTF8 | v8::WriteOptions::NO_NULL_TERMINATION, ); unsafe { buf.add(copied).write(0); } if !result.is_null() { unsafe { *result = copied; } } } else if !result.is_null() { unsafe { *result = 0; } } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_get_value_string_utf16( env_ptr: *mut Env, value: napi_value, buf: *mut u16, bufsize: usize, result: *mut usize, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); let Some(value) = value.and_then(|v| v8::Local::<v8::String>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_string_expected); }; if buf.is_null() { check_arg!(env, result); unsafe { *result = value.length(); } } else if bufsize != 0 { let buffer = unsafe { std::slice::from_raw_parts_mut(buf as _, bufsize - 1) }; let copied = value.write( &mut env.scope(), buffer, 0, v8::WriteOptions::NO_NULL_TERMINATION, ); unsafe { buf.add(copied).write(0); } if !result.is_null() { unsafe { *result = copied; } } } else if !result.is_null() { unsafe { *result = 0; } } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_coerce_to_bool<'s>( env: &'s mut Env, value: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, value); check_arg!(env, result); let coerced = value.unwrap().to_boolean(&mut env.scope()); unsafe { *result = coerced.into(); } napi_ok } #[napi_sym] fn napi_coerce_to_number<'s>( env: &'s mut Env, value: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, value); check_arg!(env, result); let Some(coerced) = value.unwrap().to_number(&mut env.scope()) else { return napi_number_expected; }; unsafe { *result = coerced.into(); } napi_ok } #[napi_sym] fn napi_coerce_to_object<'s>( env: &'s mut Env, value: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, value); check_arg!(env, result); let Some(coerced) = value.unwrap().to_object(&mut env.scope()) else { return napi_object_expected; }; unsafe { *result = coerced.into(); } napi_ok } #[napi_sym] fn napi_coerce_to_string<'s>( env: &'s mut Env, value: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, value); check_arg!(env, result); let Some(coerced) = value.unwrap().to_string(&mut env.scope()) else { return napi_string_expected; }; unsafe { *result = coerced.into(); } napi_ok } #[napi_sym] fn napi_wrap( env: &mut Env, js_object: napi_value, native_object: *mut c_void, finalize_cb: Option<napi_finalize>, finalize_hint: *mut c_void, result: *mut napi_ref, ) -> napi_status { check_arg!(env, js_object); let env_ptr = env as *mut Env; let Some(obj) = js_object.and_then(|v| v8::Local::<v8::Object>::try_from(v).ok()) else { return napi_invalid_arg; }; let napi_wrap = v8::Local::new(&mut env.scope(), &env.shared().napi_wrap); if obj .has_private(&mut env.scope(), napi_wrap) .unwrap_or(false) { return napi_invalid_arg; } if !result.is_null() { check_arg!(env, finalize_cb); } let ownership = if result.is_null() { ReferenceOwnership::Runtime } else { ReferenceOwnership::Userland }; let reference = Reference::new( env_ptr, obj.into(), 0, ownership, finalize_cb, native_object, finalize_hint, ); let reference = Reference::into_raw(reference) as *mut c_void; if !result.is_null() { check_arg!(env, finalize_cb); unsafe { *result = reference; } } let external = v8::External::new(&mut env.scope(), reference); assert!(obj .set_private(&mut env.scope(), napi_wrap, external.into()) .unwrap()); napi_ok } fn unwrap( env: &mut Env, obj: napi_value, result: *mut *mut c_void, keep: bool, ) -> napi_status { check_arg!(env, obj); if keep { check_arg!(env, result); } let Some(obj) = obj.and_then(|v| v8::Local::<v8::Object>::try_from(v).ok()) else { return napi_invalid_arg; }; let napi_wrap = v8::Local::new(&mut env.scope(), &env.shared().napi_wrap); let Some(val) = obj.get_private(&mut env.scope(), napi_wrap) else { return napi_invalid_arg; }; let Ok(external) = v8::Local::<v8::External>::try_from(val) else { return napi_invalid_arg; }; let reference = external.value() as *mut Reference; let reference = unsafe { &mut *reference }; if !result.is_null() { unsafe { *result = reference.finalize_data; } } if !keep { assert!(obj .delete_private(&mut env.scope(), napi_wrap) .unwrap_or(false)); unsafe { Reference::remove(reference) }; } napi_ok } #[napi_sym] fn napi_unwrap( env: &mut Env, obj: napi_value, result: *mut *mut c_void, ) -> napi_status { unwrap(env, obj, result, true) } #[napi_sym] fn napi_remove_wrap( env: &mut Env, obj: napi_value, result: *mut *mut c_void, ) -> napi_status { unwrap(env, obj, result, false) } struct ExternalWrapper { data: *mut c_void, type_tag: Option<napi_type_tag>, } #[napi_sym] fn napi_create_external<'s>( env: &'s mut Env, data: *mut c_void, finalize_cb: Option<napi_finalize>, finalize_hint: *mut c_void, result: *mut napi_value<'s>, ) -> napi_status { let env_ptr = env as *mut Env; check_arg!(env, result); let wrapper = Box::new(ExternalWrapper { data, type_tag: None, }); let wrapper = Box::into_raw(wrapper); let external = v8::External::new(&mut env.scope(), wrapper as _); if let Some(finalize_cb) = finalize_cb { Reference::into_raw(Reference::new( env_ptr, external.into(), 0, ReferenceOwnership::Runtime, Some(finalize_cb), data, finalize_hint, )); } unsafe { *result = external.into(); } napi_ok } #[napi_sym] fn napi_type_tag_object( env: &mut Env, object_or_external: napi_value, type_tag: *const napi_type_tag, ) -> napi_status { check_arg!(env, object_or_external); check_arg!(env, type_tag); let val = object_or_external.unwrap(); if let Ok(external) = v8::Local::<v8::External>::try_from(val) { let wrapper_ptr = external.value() as *mut ExternalWrapper; let wrapper = unsafe { &mut *wrapper_ptr }; if wrapper.type_tag.is_some() { return napi_invalid_arg; } wrapper.type_tag = Some(unsafe { *type_tag }); return napi_ok; } let Some(object) = val.to_object(&mut env.scope()) else { return napi_object_expected; }; let key = v8::Local::new(&mut env.scope(), &env.shared().type_tag); if object.has_private(&mut env.scope(), key).unwrap_or(false) { return napi_invalid_arg; } let slice = unsafe { std::slice::from_raw_parts(type_tag as *const u64, 2) }; let Some(tag) = v8::BigInt::new_from_words(&mut env.scope(), false, slice) else { return napi_generic_failure; }; if !object .set_private(&mut env.scope(), key, tag.into()) .unwrap_or(false) { return napi_generic_failure; } napi_ok } #[napi_sym] fn napi_check_object_type_tag( env: &mut Env, object_or_external: napi_value, type_tag: *const napi_type_tag, result: *mut bool, ) -> napi_status { check_arg!(env, object_or_external); check_arg!(env, type_tag); check_arg!(env, result); let type_tag = unsafe { *type_tag }; let val = object_or_external.unwrap(); if let Ok(external) = v8::Local::<v8::External>::try_from(val) { let wrapper_ptr = external.value() as *mut ExternalWrapper; let wrapper = unsafe { &mut *wrapper_ptr }; unsafe { *result = match wrapper.type_tag { Some(t) => t == type_tag, None => false, }; }; return napi_ok; } let Some(object) = val.to_object(&mut env.scope()) else { return napi_object_expected; }; let key = v8::Local::new(&mut env.scope(), &env.shared().type_tag); let Some(val) = object.get_private(&mut env.scope(), key) else { return napi_generic_failure; }; unsafe { *result = false; } if let Ok(bigint) = v8::Local::<v8::BigInt>::try_from(val) { let mut words = [0u64; 2]; let (sign, words) = bigint.to_words_array(&mut words); if !sign { let pass = if words.len() == 2 { type_tag.lower == words[0] && type_tag.upper == words[1] } else if words.len() == 1 { type_tag.lower == words[0] && type_tag.upper == 0 } else if words.is_empty() { type_tag.lower == 0 && type_tag.upper == 0 } else { false }; unsafe { *result = pass; } } } napi_ok } #[napi_sym] fn napi_get_value_external( env: *mut Env, value: napi_value, result: *mut *mut c_void, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); let Some(external) = value.and_then(|v| v8::Local::<v8::External>::try_from(v).ok()) else { return napi_set_last_error(env, napi_invalid_arg); }; let wrapper_ptr = external.value() as *const ExternalWrapper; let wrapper = unsafe { &*wrapper_ptr }; unsafe { *result = wrapper.data; } napi_clear_last_error(env) } #[napi_sym] fn napi_create_reference( env: *mut Env, value: napi_value, initial_refcount: u32, result: *mut napi_ref, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); let value = value.unwrap(); let reference = Reference::new( env, value, initial_refcount, ReferenceOwnership::Userland, None, std::ptr::null_mut(), std::ptr::null_mut(), ); let ptr = Reference::into_raw(reference); unsafe { *result = ptr as _; } napi_clear_last_error(env) } #[napi_sym] fn napi_delete_reference(env: *mut Env, ref_: napi_ref) -> napi_status { let env = check_env!(env); check_arg!(env, ref_); let reference = unsafe { Reference::from_raw(ref_ as _) }; drop(reference); napi_clear_last_error(env) } #[napi_sym] fn napi_reference_ref( env: *mut Env, ref_: napi_ref, result: *mut u32, ) -> napi_status { let env = check_env!(env); check_arg!(env, ref_); let reference = unsafe { &mut *(ref_ as *mut Reference) }; let count = reference.ref_(); if !result.is_null() { unsafe { *result = count; } } napi_clear_last_error(env) } #[napi_sym] fn napi_reference_unref( env: *mut Env, ref_: napi_ref, result: *mut u32, ) -> napi_status { let env = check_env!(env); check_arg!(env, ref_); let reference = unsafe { &mut *(ref_ as *mut Reference) }; if reference.ref_count == 0 { return napi_set_last_error(env, napi_generic_failure); } let count = reference.unref(); if !result.is_null() { unsafe { *result = count; } } napi_clear_last_error(env) } #[napi_sym] fn napi_get_reference_value( env_ptr: *mut Env, ref_: napi_ref, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, ref_); check_arg!(env, result); let reference = unsafe { &mut *(ref_ as *mut Reference) }; let value = match &reference.state { ReferenceState::Strong(g) => Some(v8::Local::new(&mut env.scope(), g)), ReferenceState::Weak(w) => w.to_local(&mut env.scope()), }; unsafe { *result = value.into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_open_handle_scope( env: *mut Env, _result: *mut napi_handle_scope, ) -> napi_status { let env = check_env!(env); napi_clear_last_error(env) } #[napi_sym] fn napi_close_handle_scope( env: *mut Env, _scope: napi_handle_scope, ) -> napi_status { let env = check_env!(env); napi_clear_last_error(env) } #[napi_sym] fn napi_open_escapable_handle_scope( env: *mut Env, _result: *mut napi_escapable_handle_scope, ) -> napi_status { let env = check_env!(env); napi_clear_last_error(env) } #[napi_sym] fn napi_close_escapable_handle_scope( env: *mut Env, _scope: napi_escapable_handle_scope, ) -> napi_status { let env = check_env!(env); napi_clear_last_error(env) } #[napi_sym] fn napi_escape_handle<'s>( env: *mut Env, _scope: napi_escapable_handle_scope, escapee: napi_value<'s>, result: *mut napi_value<'s>, ) -> napi_status { let env = check_env!(env); unsafe { *result = escapee; } napi_clear_last_error(env) } #[napi_sym] fn napi_new_instance<'s>( env: &'s mut Env, constructor: napi_value, argc: usize, argv: *const napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, constructor); if argc > 0 { check_arg!(env, argv); } check_arg!(env, result); let Some(func) = constructor.and_then(|v| v8::Local::<v8::Function>::try_from(v).ok()) else { return napi_invalid_arg; }; let args = if argc > 0 { unsafe { std::slice::from_raw_parts(argv as *mut v8::Local<v8::Value>, argc) } } else { &[] }; let Some(value) = func.new_instance(&mut env.scope(), args) else { return napi_pending_exception; }; unsafe { *result = value.into(); } napi_ok } #[napi_sym] fn napi_instanceof( env: &mut Env, object: napi_value, constructor: napi_value, result: *mut bool, ) -> napi_status { check_arg!(env, object); check_arg!(env, result); let Some(ctor) = constructor.and_then(|v| v.to_object(&mut env.scope())) else { return napi_object_expected; }; if !ctor.is_function() { unsafe { napi_throw_type_error( env, c"ERR_NAPI_CONS_FUNCTION".as_ptr(), c"Constructor must be a function".as_ptr(), ); } return napi_function_expected; } let Some(res) = object.unwrap().instance_of(&mut env.scope(), ctor) else { return napi_generic_failure; }; unsafe { *result = res; } napi_ok } #[napi_sym] fn napi_is_exception_pending( env_ptr: *mut Env, result: *mut bool, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); unsafe { *result = env.last_exception.is_some(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_get_and_clear_last_exception( env_ptr: *mut Env, result: *mut napi_value, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, result); let ex: v8::Local<v8::Value> = if let Some(last_exception) = env.last_exception.take() { v8::Local::new(&mut env.scope(), last_exception) } else { v8::undefined(&mut env.scope()).into() }; unsafe { *result = ex.into(); } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_is_arraybuffer( env: *mut Env, value: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); unsafe { *result = value.unwrap().is_array_buffer(); } napi_clear_last_error(env) } #[napi_sym] fn napi_create_arraybuffer<'s>( env: &'s mut Env, len: usize, data: *mut *mut c_void, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let buffer = v8::ArrayBuffer::new(&mut env.scope(), len); if !data.is_null() { unsafe { *data = get_array_buffer_ptr(buffer); } } unsafe { *result = buffer.into(); } napi_ok } #[napi_sym] fn napi_create_external_arraybuffer<'s>( env: &'s mut Env, data: *mut c_void, byte_length: usize, finalize_cb: napi_finalize, finalize_hint: *mut c_void, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let store = make_external_backing_store( env, data, byte_length, std::ptr::null_mut(), finalize_cb, finalize_hint, ); let ab = v8::ArrayBuffer::with_backing_store(&mut env.scope(), &store.make_shared()); let value: v8::Local<v8::Value> = ab.into(); unsafe { *result = value.into(); } napi_ok } #[napi_sym] fn napi_get_arraybuffer_info( env: *mut Env, value: napi_value, data: *mut *mut c_void, length: *mut usize, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); let Some(buf) = value.and_then(|v| v8::Local::<v8::ArrayBuffer>::try_from(v).ok()) else { return napi_set_last_error(env, napi_invalid_arg); }; if !data.is_null() { unsafe { *data = get_array_buffer_ptr(buf); } } if !length.is_null() { unsafe { *length = buf.byte_length(); } } napi_ok } #[napi_sym] fn napi_is_typedarray( env: *mut Env, value: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); unsafe { *result = value.unwrap().is_typed_array(); } napi_ok } #[napi_sym] fn napi_create_typedarray<'s>( env: &'s mut Env, ty: napi_typedarray_type, length: usize, arraybuffer: napi_value, byte_offset: usize, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, arraybuffer); check_arg!(env, result); let Some(ab) = arraybuffer.and_then(|v| v8::Local::<v8::ArrayBuffer>::try_from(v).ok()) else { return napi_arraybuffer_expected; }; macro_rules! create { ($TypedArray:ident, $size_of_element:expr) => {{ let soe = $size_of_element; if soe > 1 && byte_offset % soe != 0 { let message = v8::String::new( &mut env.scope(), format!( "start offset of {} should be multiple of {}", stringify!($TypedArray), soe ) .as_str(), ) .unwrap(); let exc = v8::Exception::range_error(&mut env.scope(), message); env.scope().throw_exception(exc); return napi_pending_exception; } if length * soe + byte_offset > ab.byte_length() { let message = v8::String::new(&mut env.scope(), "Invalid typed array length") .unwrap(); let exc = v8::Exception::range_error(&mut env.scope(), message); env.scope().throw_exception(exc); return napi_pending_exception; } let Some(ta) = v8::$TypedArray::new(&mut env.scope(), ab, byte_offset, length) else { return napi_generic_failure; }; ta.into() }}; } let typedarray: v8::Local<v8::Value> = match ty { napi_uint8_array => create!(Uint8Array, 1), napi_uint8_clamped_array => create!(Uint8ClampedArray, 1), napi_int8_array => create!(Int8Array, 1), napi_uint16_array => create!(Uint16Array, 2), napi_int16_array => create!(Int16Array, 2), napi_uint32_array => create!(Uint32Array, 4), napi_int32_array => create!(Int32Array, 4), napi_float32_array => create!(Float32Array, 4), napi_float64_array => create!(Float64Array, 8), napi_bigint64_array => create!(BigInt64Array, 8), napi_biguint64_array => create!(BigUint64Array, 8), _ => { return napi_invalid_arg; } }; unsafe { *result = typedarray.into(); } napi_ok } #[napi_sym] fn napi_get_typedarray_info( env_ptr: *mut Env, typedarray: napi_value, type_: *mut napi_typedarray_type, length: *mut usize, data: *mut *mut c_void, arraybuffer: *mut napi_value, byte_offset: *mut usize, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, typedarray); let Some(array) = typedarray.and_then(|v| v8::Local::<v8::TypedArray>::try_from(v).ok()) else { return napi_set_last_error(env_ptr, napi_invalid_arg); }; if !type_.is_null() { let tatype = if array.is_int8_array() { napi_int8_array } else if array.is_uint8_array() { napi_uint8_array } else if array.is_uint8_clamped_array() { napi_uint8_clamped_array } else if array.is_int16_array() { napi_int16_array } else if array.is_uint16_array() { napi_uint16_array } else if array.is_int32_array() { napi_int32_array } else if array.is_uint32_array() { napi_uint32_array } else if array.is_float32_array() { napi_float32_array } else if array.is_float64_array() { napi_float64_array } else if array.is_big_int64_array() { napi_bigint64_array } else if array.is_big_uint64_array() { napi_biguint64_array } else { unreachable!(); }; unsafe { *type_ = tatype; } } if !length.is_null() { unsafe { *length = array.length(); } } if !data.is_null() { unsafe { *data = array.data(); } } if !arraybuffer.is_null() { let buf = array.buffer(&mut env.scope()).unwrap(); unsafe { *arraybuffer = buf.into(); } } if !byte_offset.is_null() { unsafe { *byte_offset = array.byte_offset(); } } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_create_dataview<'s>( env: &'s mut Env, byte_length: usize, arraybuffer: napi_value<'s>, byte_offset: usize, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, arraybuffer); check_arg!(env, result); let Some(buffer) = arraybuffer.and_then(|v| v8::Local::<v8::ArrayBuffer>::try_from(v).ok()) else { return napi_invalid_arg; }; if byte_length + byte_offset > buffer.byte_length() { unsafe { return napi_throw_range_error( env, c"ERR_NAPI_INVALID_DATAVIEW_ARGS".as_ptr(), c"byte_offset + byte_length should be less than or equal to the size in bytes of the array passed in".as_ptr(), ); } } let dataview = v8::DataView::new(&mut env.scope(), buffer, byte_offset, byte_length); unsafe { *result = dataview.into(); } napi_ok } #[napi_sym] fn napi_is_dataview( env: *mut Env, value: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, result); unsafe { *result = value.unwrap().is_data_view(); } napi_clear_last_error(env) } #[napi_sym] fn napi_get_dataview_info( env_ptr: *mut Env, dataview: napi_value, byte_length: *mut usize, data: *mut *mut c_void, arraybuffer: *mut napi_value, byte_offset: *mut usize, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, dataview); let Some(array) = dataview.and_then(|v| v8::Local::<v8::DataView>::try_from(v).ok()) else { return napi_invalid_arg; }; if !byte_length.is_null() { unsafe { *byte_length = array.byte_length(); } } if !arraybuffer.is_null() { let Some(buffer) = array.buffer(&mut env.scope()) else { return napi_generic_failure; }; unsafe { *arraybuffer = buffer.into(); } } if !data.is_null() { unsafe { *data = array.data(); } } if !byte_offset.is_null() { unsafe { *byte_offset = array.byte_offset(); } } napi_clear_last_error(env_ptr) } #[napi_sym] fn napi_get_version(env: *mut Env, result: *mut u32) -> napi_status { let env = check_env!(env); check_arg!(env, result); unsafe { *result = NAPI_VERSION; } napi_clear_last_error(env) } #[napi_sym] fn napi_create_promise<'s>( env: &'s mut Env, deferred: *mut napi_deferred, promise: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, deferred); check_arg!(env, promise); let resolver = v8::PromiseResolver::new(&mut env.scope()).unwrap(); let global = v8::Global::new(&mut env.scope(), resolver); let global_ptr = global.into_raw().as_ptr() as napi_deferred; let p = resolver.get_promise(&mut env.scope()); unsafe { *deferred = global_ptr; } unsafe { *promise = p.into(); } napi_ok } #[napi_sym] fn napi_resolve_deferred( env: &mut Env, deferred: napi_deferred, result: napi_value, ) -> napi_status { check_arg!(env, result); check_arg!(env, deferred); // Make sure microtasks don't run and call back into JS env .scope() .set_microtasks_policy(v8::MicrotasksPolicy::Explicit); let deferred_ptr = unsafe { NonNull::new_unchecked(deferred as *mut v8::PromiseResolver) }; let global = unsafe { v8::Global::from_raw(env.isolate(), deferred_ptr) }; let resolver = v8::Local::new(&mut env.scope(), global); let success = resolver .resolve(&mut env.scope(), result.unwrap()) .unwrap_or(false); // Restore policy env .scope() .set_microtasks_policy(v8::MicrotasksPolicy::Auto); if success { napi_ok } else { napi_generic_failure } } #[napi_sym] fn napi_reject_deferred( env: &mut Env, deferred: napi_deferred, result: napi_value, ) -> napi_status { check_arg!(env, result); check_arg!(env, deferred); let deferred_ptr = unsafe { NonNull::new_unchecked(deferred as *mut v8::PromiseResolver) }; let global = unsafe { v8::Global::from_raw(env.isolate(), deferred_ptr) }; let resolver = v8::Local::new(&mut env.scope(), global); if !resolver .reject(&mut env.scope(), result.unwrap()) .unwrap_or(false) { return napi_generic_failure; } napi_ok } #[napi_sym] fn napi_is_promise( env: *mut Env, value: napi_value, is_promise: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, is_promise); unsafe { *is_promise = value.unwrap().is_promise(); } napi_clear_last_error(env) } #[napi_sym] fn napi_create_date<'s>( env: &'s mut Env, time: f64, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, result); let Some(date) = v8::Date::new(&mut env.scope(), time) else { return napi_generic_failure; }; unsafe { *result = date.into(); } napi_ok } #[napi_sym] fn napi_is_date( env: *mut Env, value: napi_value, is_date: *mut bool, ) -> napi_status { let env = check_env!(env); check_arg!(env, value); check_arg!(env, is_date); unsafe { *is_date = value.unwrap().is_date(); } napi_clear_last_error(env) } #[napi_sym] fn napi_get_date_value( env: &mut Env, value: napi_value, result: *mut f64, ) -> napi_status { check_arg!(env, result); let Some(date) = value.and_then(|v| v8::Local::<v8::Date>::try_from(v).ok()) else { return napi_date_expected; }; unsafe { *result = date.value_of(); } napi_ok } #[napi_sym] fn napi_run_script<'s>( env: &'s mut Env, script: napi_value, result: *mut napi_value<'s>, ) -> napi_status { check_arg!(env, script); check_arg!(env, result); let Some(script) = script.and_then(|v| v8::Local::<v8::String>::try_from(v).ok()) else { return napi_string_expected; }; let Some(script) = v8::Script::compile(&mut env.scope(), script, None) else { return napi_generic_failure; }; let Some(rv) = script.run(&mut env.scope()) else { return napi_generic_failure; }; unsafe { *result = rv.into(); } napi_ok } #[napi_sym] fn napi_add_finalizer( env_ptr: *mut Env, value: napi_value, finalize_data: *mut c_void, finalize_cb: Option<napi_finalize>, finalize_hint: *mut c_void, result: *mut napi_ref, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, value); check_arg!(env, finalize_cb); let Some(value) = value.and_then(|v| v8::Local::<v8::Object>::try_from(v).ok()) else { return napi_set_last_error(env, napi_invalid_arg); }; let ownership = if result.is_null() { ReferenceOwnership::Runtime } else { ReferenceOwnership::Userland }; let reference = Reference::new( env, value.into(), 0, ownership, finalize_cb, finalize_data, finalize_hint, ); if !result.is_null() { unsafe { *result = Reference::into_raw(reference) as _; } } napi_clear_last_error(env_ptr) } #[napi_sym] fn node_api_post_finalizer( env: *mut Env, _finalize_cb: napi_finalize, _finalize_data: *mut c_void, _finalize_hint: *mut c_void, ) -> napi_status { napi_clear_last_error(env) } #[napi_sym] fn napi_adjust_external_memory( env: *mut Env, change_in_bytes: i64, adjusted_value: *mut i64, ) -> napi_status { let env = check_env!(env); check_arg!(env, adjusted_value); unsafe { *adjusted_value = env .isolate() .adjust_amount_of_external_allocated_memory(change_in_bytes); } napi_clear_last_error(env) } #[napi_sym] fn napi_set_instance_data( env: *mut Env, data: *mut c_void, finalize_cb: Option<napi_finalize>, finalize_hint: *mut c_void, ) -> napi_status { let env = check_env!(env); env.shared_mut().instance_data = Some(InstanceData { data, finalize_cb, finalize_hint, }); napi_clear_last_error(env) } #[napi_sym] fn napi_get_instance_data( env: *mut Env, data: *mut *mut c_void, ) -> napi_status { let env = check_env!(env); check_arg!(env, data); let instance_data = match &env.shared().instance_data { Some(v) => v.data, None => std::ptr::null_mut(), }; unsafe { *data = instance_data }; napi_clear_last_error(env) } #[napi_sym] fn napi_detach_arraybuffer(env: *mut Env, value: napi_value) -> napi_status { let env = check_env!(env); check_arg!(env, value); let Some(ab) = value.and_then(|v| v8::Local::<v8::ArrayBuffer>::try_from(v).ok()) else { return napi_set_last_error(env, napi_arraybuffer_expected); }; if !ab.is_detachable() { return napi_set_last_error(env, napi_detachable_arraybuffer_expected); } // Expected to crash for None. ab.detach(None).unwrap(); napi_clear_last_error(env); napi_ok } #[napi_sym] fn napi_is_detached_arraybuffer( env_ptr: *mut Env, arraybuffer: napi_value, result: *mut bool, ) -> napi_status { let env = check_env!(env_ptr); check_arg!(env, arraybuffer); check_arg!(env, result); let is_detached = match arraybuffer .and_then(|v| v8::Local::<v8::ArrayBuffer>::try_from(v).ok()) { Some(ab) => ab.was_detached(), None => false, }; unsafe { *result = is_detached; } napi_clear_last_error(env) }