const std = @import("std"); const JSC = @import("root").bun.JSC; const strings = @import("root").bun.strings; const bun = @import("root").bun; const Lock = @import("../lock.zig").Lock; const JSValue = JSC.JSValue; const ZigString = JSC.ZigString; const TODO_EXCEPTION: JSC.C.ExceptionRef = null; const Channel = @import("../sync.zig").Channel; const log = bun.Output.scoped(.napi, false); // These wrappers exist so we can set a breakpoint in lldb fn invalidArg() napi_status { if (comptime bun.Environment.allow_assert) { log("invalid arg", .{}); } return .invalid_arg; } fn genericFailure() napi_status { if (comptime bun.Environment.allow_assert) { log("generic failure", .{}); } return .generic_failure; } pub const napi_env = *JSC.JSGlobalObject; pub const Ref = opaque { pub fn create(globalThis: *JSC.JSGlobalObject, value: JSValue) *Ref { JSC.markBinding(@src()); var ref: *Ref = undefined; std.debug.assert( napi_create_reference( globalThis, value, 1, &ref, ) == .ok, ); if (comptime bun.Environment.isDebug) { std.debug.assert(ref.get() == value); } return ref; } pub fn get(ref: *Ref) JSValue { JSC.markBinding(@src()); return napi_get_reference_value_internal(ref); } pub fn destroy(ref: *Ref) void { JSC.markBinding(@src()); napi_delete_reference_internal(ref); } pub fn set(this: *Ref, value: JSC.JSValue) void { JSC.markBinding(@src()); napi_set_ref(this, value); } extern fn napi_delete_reference_internal(ref: *Ref) void; extern fn napi_set_ref(ref: *Ref, value: JSC.JSValue) void; }; pub const napi_handle_scope = napi_env; pub const napi_escapable_handle_scope = napi_env; pub const napi_callback_info = *JSC.CallFrame; pub const napi_deferred = *JSC.napi.Ref; pub const napi_value = JSC.JSValue; pub const struct_napi_escapable_handle_scope__ = opaque {}; pub const struct_napi_deferred__ = opaque {}; const char16_t = u16; pub const napi_default: c_int = 0; pub const napi_writable: c_int = 1; pub const napi_enumerable: c_int = 2; pub const napi_configurable: c_int = 4; pub const napi_static: c_int = 1024; pub const napi_default_method: c_int = 5; pub const napi_default_jsproperty: c_int = 7; pub const napi_property_attributes = c_uint; pub const napi_valuetype = enum(c_uint) { undefined = 0, null = 1, boolean = 2, number = 3, string = 4, symbol = 5, object = 6, function = 7, external = 8, bigint = 9, }; pub const napi_typedarray_type = enum(c_uint) { int8_array = 0, uint8_array = 1, uint8_clamped_array = 2, int16_array = 3, uint16_array = 4, int32_array = 5, uint32_array = 6, float32_array = 7, float64_array = 8, bigint64_array = 9, biguint64_array = 10, pub fn fromJSType(this: JSC.JSValue.JSType) ?napi_typedarray_type { return switch (this) { .Int8Array => napi_typedarray_type.int8_array, .Uint8Array => napi_typedarray_type.uint8_array, .Uint8ClampedArray => napi_typedarray_type.uint8_clamped_array, .Int16Array => napi_typedarray_type.int16_array, .Uint16Array => napi_typedarray_type.uint16_array, .Int32Array => napi_typedarray_type.int32_array, .Uint32Array => napi_typedarray_type.uint32_array, .Float32Array => napi_typedarray_type.float32_array, .Float64Array => napi_typedarray_type.float64_array, .BigInt64Array => napi_typedarray_type.bigint64_array, .BigUint64Array => napi_typedarray_type.biguint64_array, else => null, }; } pub fn toJSType(this: napi_typedarray_type) JSC.JSValue.JSType { return switch (this) { .int8_array => .Int8Array, .uint8_array => .Uint8Array, .uint8_clamped_array => .Uint8ClampedArray, .int16_array => .Int16Array, .uint16_array => .Uint16Array, .int32_array => .Int32Array, .uint32_array => .Uint32Array, .float32_array => .Float32Array, .float64_array => .Float64Array, .bigint64_array => .BigInt64Array, .biguint64_array => .BigUint64Array, }; } pub fn toC(this: napi_typedarray_type) JSC.C.JSTypedArrayType { return this.toJSType().toC(); } }; pub const napi_status = enum(c_uint) { ok = 0, invalid_arg = 1, object_expected = 2, string_expected = 3, name_expected = 4, function_expected = 5, number_expected = 6, boolean_expected = 7, array_expected = 8, generic_failure = 9, pending_exception = 10, cancelled = 11, escape_called_twice = 12, handle_scope_mismatch = 13, callback_scope_mismatch = 14, queue_full = 15, closing = 16, bigint_expected = 17, date_expected = 18, arraybuffer_expected = 19, detachable_arraybuffer_expected = 20, would_deadlock = 21, }; pub const napi_callback = ?*const fn (napi_env, napi_callback_info) callconv(.C) napi_value; pub const napi_finalize = ?*const fn (napi_env, ?*anyopaque, ?*anyopaque) callconv(.C) void; pub const napi_property_descriptor = extern struct { utf8name: [*c]const u8, name: napi_value, method: napi_callback, getter: napi_callback, setter: napi_callback, value: napi_value, attributes: napi_property_attributes, data: ?*anyopaque, }; pub const napi_extended_error_info = extern struct { error_message: [*c]const u8, engine_reserved: ?*anyopaque, engine_error_code: u32, error_code: napi_status, }; pub const napi_key_include_prototypes: c_int = 0; pub const napi_key_own_only: c_int = 1; pub const napi_key_collection_mode = c_uint; pub const napi_key_all_properties: c_int = 0; pub const napi_key_writable: c_int = 1; pub const napi_key_enumerable: c_int = 2; pub const napi_key_configurable: c_int = 4; pub const napi_key_skip_strings: c_int = 8; pub const napi_key_skip_symbols: c_int = 16; pub const napi_key_filter = c_uint; pub const napi_key_keep_numbers: c_int = 0; pub const napi_key_numbers_to_strings: c_int = 1; pub const napi_key_conversion = c_uint; pub const napi_type_tag = extern struct { lower: u64, upper: u64, }; pub extern fn napi_get_last_error_info(env: napi_env, result: [*c][*c]const napi_extended_error_info) napi_status; pub export fn napi_get_undefined(_: napi_env, result: *napi_value) napi_status { log("napi_get_undefined", .{}); result.* = JSValue.jsUndefined(); return .ok; } pub export fn napi_get_null(_: napi_env, result: *napi_value) napi_status { log("napi_get_null", .{}); result.* = JSValue.jsNull(); return .ok; } pub extern fn napi_get_global(env: napi_env, result: *napi_value) napi_status; pub export fn napi_get_boolean(_: napi_env, value: bool, result: *napi_value) napi_status { log("napi_get_boolean", .{}); result.* = JSValue.jsBoolean(value); return .ok; } pub export fn napi_create_array(env: napi_env, result: *napi_value) napi_status { log("napi_create_array", .{}); result.* = JSValue.c(JSC.C.JSObjectMakeArray(env.ref(), 0, null, null)); return .ok; } const prefilled_undefined_args_array: [128]JSC.JSValue = brk: { var args: [128]JSC.JSValue = undefined; for (args, 0..) |_, i| { args[i] = JSValue.jsUndefined(); } break :brk args; }; pub export fn napi_create_array_with_length(env: napi_env, length: usize, result: *napi_value) napi_status { log("napi_create_array_with_length", .{}); const len = @intCast(u32, length); const array = JSC.JSValue.createEmptyArray(env, len); array.ensureStillAlive(); var i: u32 = 0; while (i < len) : (i += 1) { array.putIndex(env, i, JSValue.jsUndefined()); } array.ensureStillAlive(); result.* = array; return .ok; } pub export fn napi_create_double(_: napi_env, value: f64, result: *napi_value) napi_status { log("napi_create_double", .{}); result.* = JSValue.jsNumber(value); return .ok; } pub export fn napi_create_int32(_: napi_env, value: i32, result: *napi_value) napi_status { log("napi_create_int32", .{}); result.* = JSValue.jsNumber(value); return .ok; } pub export fn napi_create_uint32(_: napi_env, value: u32, result: *napi_value) napi_status { log("napi_create_uint32", .{}); result.* = JSValue.jsNumber(value); return .ok; } pub export fn napi_create_int64(_: napi_env, value: i64, result: *napi_value) napi_status { log("napi_create_int64", .{}); result.* = JSValue.jsNumber(value); return .ok; } inline fn setNapiValue(result: *napi_value, value: JSValue) void { value.ensureStillAlive(); result.* = value; } pub export fn napi_create_string_latin1(env: napi_env, str: [*]const u8, length: usize, result: *napi_value) napi_status { log("napi_create_string_latin1", .{}); const slice = if (NAPI_AUTO_LENGTH == length) bun.sliceTo(@ptrCast([*:0]const u8, str), 0) else str[0..length]; setNapiValue(result, JSC.ZigString.init(slice).toValueGC(env)); return .ok; } pub export fn napi_create_string_utf8(env: napi_env, str: [*]const u8, length: usize, result: *napi_value) napi_status { const slice = if (NAPI_AUTO_LENGTH == length) bun.sliceTo(@ptrCast([*:0]const u8, str), 0) else str[0..length]; log("napi_create_string_utf8: {s}", .{slice}); var string = bun.String.create(slice); defer string.deref(); setNapiValue(result, string.toJS(env)); return .ok; } pub export fn napi_create_string_utf16(env: napi_env, str: [*]const char16_t, length: usize, result: *napi_value) napi_status { log("napi_create_string_utf16", .{}); const slice = if (NAPI_AUTO_LENGTH == length) bun.sliceTo(@ptrCast([*:0]const char16_t, str), 0) else str[0..length]; setNapiValue(result, JSC.ZigString.from16(slice.ptr, length).toValueGC(env)); return .ok; } pub extern fn napi_create_symbol(env: napi_env, description: napi_value, result: *napi_value) napi_status; pub extern fn napi_create_error(env: napi_env, code: napi_value, msg: napi_value, result: *napi_value) napi_status; pub extern fn napi_create_type_error(env: napi_env, code: napi_value, msg: napi_value, result: *napi_value) napi_status; pub extern fn napi_create_range_error(env: napi_env, code: napi_value, msg: napi_value, result: *napi_value) napi_status; pub extern fn napi_typeof(env: napi_env, value: napi_value, result: *napi_valuetype) napi_status; pub export fn napi_get_value_double(_: napi_env, value: napi_value, result: *f64) napi_status { log("napi_get_value_double", .{}); result.* = value.asNumber(); return .ok; } pub export fn napi_get_value_int32(_: napi_env, value: napi_value, result: *i32) napi_status { log("napi_get_value_int32", .{}); result.* = value.to(i32); return .ok; } pub export fn napi_get_value_uint32(_: napi_env, value: napi_value, result: *u32) napi_status { log("napi_get_value_uint32", .{}); result.* = value.to(u32); return .ok; } pub export fn napi_get_value_int64(_: napi_env, value: napi_value, result: *i64) napi_status { log("napi_get_value_int64", .{}); result.* = value.to(i64); return .ok; } pub export fn napi_get_value_bool(_: napi_env, value: napi_value, result: *bool) napi_status { log("napi_get_value_bool", .{}); result.* = value.to(bool); return .ok; } inline fn maybeAppendNull(ptr: anytype, doit: bool) void { if (doit) { ptr.* = 0; } } pub export fn napi_get_value_string_latin1(env: napi_env, value: napi_value, buf_ptr_: ?[*:0]c_char, bufsize: usize, result_ptr: ?*usize) napi_status { log("napi_get_value_string_latin1", .{}); defer value.ensureStillAlive(); var buf_ptr = @ptrCast(?[*:0]u8, buf_ptr_); const str = value.toBunString(env); var buf = buf_ptr orelse { if (result_ptr) |result| { result.* = str.latin1ByteLength(); } return .ok; }; if (str.isEmpty()) { if (result_ptr) |result| { result.* = 0; } buf[0] = 0; return .ok; } var buf_ = buf[0..bufsize]; if (bufsize == NAPI_AUTO_LENGTH) { buf_ = bun.sliceTo(buf_ptr.?, 0); if (buf_.len == 0) { if (result_ptr) |result| { result.* = 0; } return .ok; } } const written = str.encodeInto(buf_, .latin1) catch unreachable; const max_buf_len = buf_.len; if (result_ptr) |result| { result.* = written; } else if (written < max_buf_len) { buf[written] = 0; } return .ok; } /// Copies a JavaScript string into a UTF-8 string buffer. The result is the /// number of bytes (excluding the null terminator) copied into buf. /// A sufficient buffer size should be greater than the length of string, /// reserving space for null terminator. /// If bufsize is insufficient, the string will be truncated and null terminated. /// If buf is NULL, this method returns the length of the string (in bytes) /// via the result parameter. /// The result argument is optional unless buf is NULL. pub extern fn napi_get_value_string_utf8(env: napi_env, value: napi_value, buf_ptr: [*c]u8, bufsize: usize, result_ptr: ?*usize) napi_status; pub export fn napi_get_value_string_utf16(env: napi_env, value: napi_value, buf_ptr: ?[*]char16_t, bufsize: usize, result_ptr: ?*usize) napi_status { log("napi_get_value_string_utf16", .{}); defer value.ensureStillAlive(); const str = value.toBunString(env); var buf = buf_ptr orelse { if (result_ptr) |result| { result.* = str.utf16ByteLength(); } return .ok; }; if (str.isEmpty()) { if (result_ptr) |result| { result.* = 0; } buf[0] = 0; return .ok; } var buf_ = buf[0..bufsize]; if (bufsize == NAPI_AUTO_LENGTH) { buf_ = bun.sliceTo(@ptrCast([*:0]u16, buf_ptr.?), 0); if (buf_.len == 0) { if (result_ptr) |result| { result.* = 0; } return .ok; } } const max_buf_len = buf_.len; const written = (str.encodeInto(std.mem.sliceAsBytes(buf_), .utf16le) catch unreachable) >> 1; if (result_ptr) |result| { result.* = written; // We should only write to the buffer is no result pointer is provided. // If we perform both operations, } else if (written < max_buf_len) { buf[written] = 0; } return .ok; } pub export fn napi_coerce_to_bool(env: napi_env, value: napi_value, result: *napi_value) napi_status { log("napi_coerce_to_bool", .{}); result.* = JSValue.jsBoolean(value.coerce(bool, env)); return .ok; } pub export fn napi_coerce_to_number(env: napi_env, value: napi_value, result: *napi_value) napi_status { log("napi_coerce_to_number", .{}); result.* = JSC.JSValue.jsNumber(JSC.C.JSValueToNumber(env.ref(), value.asObjectRef(), TODO_EXCEPTION)); return .ok; } pub export fn napi_coerce_to_object(env: napi_env, value: napi_value, result: *napi_value) napi_status { log("napi_coerce_to_object", .{}); result.* = JSValue.c(JSC.C.JSValueToObject(env.ref(), value.asObjectRef(), TODO_EXCEPTION)); return .ok; } pub export fn napi_get_prototype(env: napi_env, object: napi_value, result: *napi_value) napi_status { log("napi_get_prototype", .{}); if (!object.isObject()) { return .object_expected; } result.* = JSValue.c(JSC.C.JSObjectGetPrototype(env.ref(), object.asObjectRef())); return .ok; } // TODO: bind JSC::ownKeys // pub export fn napi_get_property_names(env: napi_env, object: napi_value, result: *napi_value) napi_status { // log("napi_get_property_names ", .{}); // if (!object.isObject()) { // return .object_expected; // } // result.* = // } pub export fn napi_set_element(env: napi_env, object: napi_value, index: c_uint, value: napi_value) napi_status { log("napi_set_element", .{}); if (!object.jsType().isIndexable()) { return .array_expected; } if (value.isEmpty()) return invalidArg(); JSC.C.JSObjectSetPropertyAtIndex(env.ref(), object.asObjectRef(), index, value.asObjectRef(), TODO_EXCEPTION); return .ok; } pub export fn napi_has_element(env: napi_env, object: napi_value, index: c_uint, result: *bool) napi_status { log("napi_has_element", .{}); if (!object.jsType().isIndexable()) { return .array_expected; } result.* = object.getLength(env) > index; return .ok; } pub extern fn napi_get_element(env: napi_env, object: napi_value, index: u32, result: *napi_value) napi_status; pub extern fn napi_define_properties(env: napi_env, object: napi_value, property_count: usize, properties: [*c]const napi_property_descriptor) napi_status; pub export fn napi_is_array(_: napi_env, value: napi_value, result: *bool) napi_status { log("napi_is_array", .{}); result.* = value.jsType().isArray(); return .ok; } pub export fn napi_get_array_length(env: napi_env, value: napi_value, result: [*c]u32) napi_status { log("napi_get_array_length", .{}); if (!value.jsType().isArray()) { return .array_expected; } result.* = @truncate(u32, value.getLength(env)); return .ok; } pub export fn napi_strict_equals(env: napi_env, lhs: napi_value, rhs: napi_value, result: *bool) napi_status { log("napi_strict_equals", .{}); // there is some nuance with NaN here i'm not sure about result.* = lhs.isSameValue(rhs, env); return .ok; } pub extern fn napi_call_function(env: napi_env, recv: napi_value, func: napi_value, argc: usize, argv: [*c]const napi_value, result: *napi_value) napi_status; pub export fn napi_new_instance(env: napi_env, constructor: napi_value, argc: usize, argv: [*c]const napi_value, result: *napi_value) napi_status { log("napi_new_instance", .{}); JSC.markBinding(@src()); if (argc > 0 and argv == null) { return invalidArg(); } var exception = [_]JSC.C.JSValueRef{null}; result.* = JSValue.c( JSC.C.JSObjectCallAsConstructor( env.ref(), constructor.asObjectRef(), argc, if (argv != null) @ptrCast([*]const JSC.C.JSValueRef, argv) else null, &exception, ), ); if (exception[0] != null) { return genericFailure(); } return .ok; } pub export fn napi_instanceof(env: napi_env, object: napi_value, constructor: napi_value, result: *bool) napi_status { log("napi_instanceof", .{}); // TODO: does this throw object_expected in node? result.* = object.isCell() and object.isInstanceOf(env, constructor); return .ok; } pub extern fn napi_get_cb_info(env: napi_env, cbinfo: napi_callback_info, argc: [*c]usize, argv: *napi_value, this_arg: *napi_value, data: [*]*anyopaque) napi_status; pub extern fn napi_get_new_target(env: napi_env, cbinfo: napi_callback_info, result: *napi_value) napi_status; pub extern fn napi_define_class( env: napi_env, utf8name: [*c]const u8, length: usize, constructor: napi_callback, data: ?*anyopaque, property_count: usize, properties: [*c]const napi_property_descriptor, result: *napi_value, ) napi_status; pub extern fn napi_wrap(env: napi_env, js_object: napi_value, native_object: ?*anyopaque, finalize_cb: napi_finalize, finalize_hint: ?*anyopaque, result: [*]*Ref) napi_status; pub extern fn napi_unwrap(env: napi_env, js_object: napi_value, result: [*]*anyopaque) napi_status; pub extern fn napi_remove_wrap(env: napi_env, js_object: napi_value, result: [*]*anyopaque) napi_status; pub extern fn napi_create_object(env: napi_env, result: *napi_value) napi_status; pub extern fn napi_create_external(env: napi_env, data: ?*anyopaque, finalize_cb: napi_finalize, finalize_hint: ?*anyopaque, result: *napi_value) napi_status; pub extern fn napi_get_value_external(env: napi_env, value: napi_value, result: [*]*anyopaque) napi_status; pub extern fn napi_create_reference(env: napi_env, value: napi_value, initial_refcount: u32, result: **Ref) napi_status; pub extern fn napi_delete_reference(env: napi_env, ref: *Ref) napi_status; pub extern fn napi_reference_ref(env: napi_env, ref: *Ref, result: [*c]u32) napi_status; pub extern fn napi_reference_unref(env: napi_env, ref: *Ref, result: [*c]u32) napi_status; pub extern fn napi_get_reference_value(env: napi_env, ref: *Ref, result: *napi_value) napi_status; pub extern fn napi_get_reference_value_internal(ref: *Ref) JSC.JSValue; // JSC scans the stack // we don't need this pub export fn napi_open_handle_scope(env: napi_env, result: *napi_handle_scope) napi_status { log("napi_open_handle_scope", .{}); result.* = env; return .ok; } // JSC scans the stack // we don't need this pub export fn napi_close_handle_scope(_: napi_env, _: napi_handle_scope) napi_status { log("napi_close_handle_scope", .{}); return .ok; } // we don't support async contexts pub export fn napi_async_init(env: napi_env, _: napi_value, _: napi_value, async_ctx: **anyopaque) napi_status { log("napi_async_init", .{}); async_ctx.* = env; return .ok; } // we don't support async contexts pub export fn napi_async_destroy(_: napi_env, _: *anyopaque) napi_status { log("napi_async_destroy", .{}); return .ok; } // this is just a regular function call pub export fn napi_make_callback(env: napi_env, _: *anyopaque, recv: napi_value, func: napi_value, arg_count: usize, args: ?[*]const napi_value, result: *napi_value) napi_status { log("napi_make_callback", .{}); if (func.isEmptyOrUndefinedOrNull() or !func.isCallable(env.vm())) { return .function_expected; } const res = func.callWithThis( env, if (recv != .zero) recv else JSC.JSValue.jsUndefined(), if (arg_count > 0 and args != null) @ptrCast([*]const JSC.JSValue, args.?)[0..arg_count] else &.{}, ); result.* = res; // TODO: this is likely incorrect if (res.isAnyError()) { return .pending_exception; } return .ok; } // Sometimes shared libraries reference symbols which are not used // We don't want to fail to load the library because of that // so we instead return an error and warn the user fn notImplementedYet(comptime name: []const u8) void { bun.once( struct { pub fn warn() void { if (JSC.VirtualMachine.get().log.level.atLeast(.warn)) { bun.Output.prettyErrorln("warning: Node-API function \"{s}\" is not implemented yet.\n Track the status of Node-API in Bun: https://github.com/oven-sh/bun/issues/158", .{name}); bun.Output.flush(); } } }.warn, void, ); } // JSC stack scanning will handle this pub export fn napi_open_escapable_handle_scope(env: napi_env, handle: *napi_escapable_handle_scope) napi_status { log("napi_open_escapable_handle_scope", .{}); handle.* = env; return .ok; } pub export fn napi_close_escapable_handle_scope(_: napi_env, _: napi_escapable_handle_scope) napi_status { log("napi_close_escapable_handle_scope", .{}); return .ok; } pub export fn napi_escape_handle(_: napi_env, _: napi_escapable_handle_scope, value: napi_value, result: *napi_value) napi_status { log("napi_escape_handle", .{}); value.ensureStillAlive(); result.* = value; return .ok; } pub export fn napi_type_tag_object(_: napi_env, _: napi_value, _: [*c]const napi_type_tag) napi_status { log("napi_type_tag_object", .{}); notImplementedYet("napi_type_tag_object"); return genericFailure(); } pub export fn napi_check_object_type_tag(_: napi_env, _: napi_value, _: [*c]const napi_type_tag, _: *bool) napi_status { log("napi_check_object_type_tag", .{}); notImplementedYet("napi_check_object_type_tag"); return genericFailure(); } // do nothing for both of these pub export fn napi_open_callback_scope(_: napi_env, _: napi_value, _: *anyopaque, _: *anyopaque) napi_status { log("napi_open_callback_scope", .{}); return .ok; } pub export fn napi_close_callback_scope(_: napi_env, _: *anyopaque) napi_status { log("napi_close_callback_scope", .{}); return .ok; } pub extern fn napi_throw(env: napi_env, @"error": napi_value) napi_status; pub extern fn napi_throw_error(env: napi_env, code: [*c]const u8, msg: [*c]const u8) napi_status; pub extern fn napi_throw_type_error(env: napi_env, code: [*c]const u8, msg: [*c]const u8) napi_status; pub extern fn napi_throw_range_error(env: napi_env, code: [*c]const u8, msg: [*c]const u8) napi_status; pub export fn napi_is_error(_: napi_env, value: napi_value, result: *bool) napi_status { log("napi_is_error", .{}); result.* = value.isAnyError(); return .ok; } pub extern fn napi_is_exception_pending(env: napi_env, result: *bool) napi_status; pub extern fn napi_get_and_clear_last_exception(env: napi_env, result: *napi_value) napi_status; pub export fn napi_is_arraybuffer(_: napi_env, value: napi_value, result: *bool) napi_status { log("napi_is_arraybuffer", .{}); result.* = !value.isNumber() and value.jsTypeLoose() == .ArrayBuffer; return .ok; } pub export fn napi_create_arraybuffer(env: napi_env, byte_length: usize, data: [*]const u8, result: *napi_value) napi_status { log("napi_create_arraybuffer", .{}); var typed_array = JSC.C.JSObjectMakeTypedArray(env.ref(), .kJSTypedArrayTypeArrayBuffer, byte_length, TODO_EXCEPTION); var array_buffer = JSValue.c(typed_array).asArrayBuffer(env) orelse return genericFailure(); const len = @min(array_buffer.len, @truncate(u32, byte_length)); @memcpy(array_buffer.ptr[0..len], data[0..len]); result.* = JSValue.c(typed_array); return .ok; } pub export fn napi_create_external_arraybuffer(env: napi_env, external_data: ?*anyopaque, byte_length: usize, finalize_cb: napi_finalize, finalize_hint: ?*anyopaque, result: *napi_value) napi_status { log("napi_create_external_arraybuffer", .{}); var external = JSC.ExternalBuffer.create( finalize_hint, @ptrCast([*]u8, external_data.?)[0..byte_length], env, finalize_cb, env.bunVM().allocator, ) catch { return genericFailure(); }; result.* = external.toArrayBuffer(env); return .ok; } pub export fn napi_get_arraybuffer_info(env: napi_env, arraybuffer: napi_value, data: ?*[*]u8, byte_length: ?*usize) napi_status { log("napi_get_arraybuffer_info", .{}); const array_buffer = arraybuffer.asArrayBuffer(env) orelse return .arraybuffer_expected; var slice = array_buffer.slice(); if (data) |dat| dat.* = slice.ptr; if (byte_length) |len| len.* = slice.len; return .ok; } pub export fn napi_is_typedarray(_: napi_env, value: napi_value, result: ?*bool) napi_status { log("napi_is_typedarray", .{}); if (result != null) result.?.* = value.jsTypeLoose().isTypedArray(); return if (result != null) .ok else invalidArg(); } pub export fn napi_create_typedarray(env: napi_env, @"type": napi_typedarray_type, length: usize, arraybuffer: napi_value, byte_offset: usize, result: *napi_value) napi_status { log("napi_create_typedarray", .{}); result.* = JSValue.c( JSC.C.JSObjectMakeTypedArrayWithArrayBufferAndOffset( env.ref(), @"type".toC(), arraybuffer.asObjectRef(), byte_offset, length, TODO_EXCEPTION, ), ); return .ok; } pub export fn napi_get_typedarray_info( env: napi_env, typedarray: napi_value, @"type": ?*napi_typedarray_type, length: ?*usize, data: ?*[*]u8, arraybuffer: ?*napi_value, byte_offset: ?*usize, ) napi_status { log("napi_get_typedarray_info", .{}); if (typedarray.isEmptyOrUndefinedOrNull()) return invalidArg(); defer typedarray.ensureStillAlive(); const array_buffer = typedarray.asArrayBuffer(env) orelse return invalidArg(); if (@"type" != null) @"type".?.* = napi_typedarray_type.fromJSType(array_buffer.typed_array_type) orelse return invalidArg(); // TODO: handle detached if (data != null) data.?.* = array_buffer.ptr; if (length != null) length.?.* = array_buffer.len; if (arraybuffer != null) arraybuffer.?.* = JSValue.c(JSC.C.JSObjectGetTypedArrayBuffer(env.ref(), typedarray.asObjectRef(), null)); if (byte_offset != null) byte_offset.?.* = array_buffer.offset; return .ok; } pub extern fn napi_create_dataview(env: napi_env, length: usize, arraybuffer: napi_value, byte_offset: usize, result: *napi_value) napi_status; pub export fn napi_is_dataview(_: napi_env, value: napi_value, result: *bool) napi_status { log("napi_is_dataview", .{}); result.* = !value.isEmptyOrUndefinedOrNull() and value.jsTypeLoose() == .DataView; return .ok; } pub export fn napi_get_dataview_info(env: napi_env, dataview: napi_value, bytelength: *usize, data: *?[*]u8, arraybuffer: *napi_value, byte_offset: *usize) napi_status { log("napi_get_dataview_info", .{}); var array_buffer = dataview.asArrayBuffer(env) orelse return .object_expected; bytelength.* = array_buffer.byte_len; data.* = array_buffer.ptr; // TODO: will this work? will it fail due to being a DataView instead of a TypedArray? arraybuffer.* = JSValue.c(JSC.C.JSObjectGetTypedArrayBuffer(env.ref(), dataview.asObjectRef(), null)); byte_offset.* = array_buffer.offset; return .ok; } pub export fn napi_get_version(_: napi_env, result: *u32) napi_status { log("napi_get_version", .{}); result.* = NAPI_VERSION; return .ok; } pub export fn napi_create_promise(env: napi_env, deferred: *napi_deferred, promise: *napi_value) napi_status { log("napi_create_promise", .{}); deferred.* = JSC.JSPromise.Strong.init(env).strong.ref.?; promise.* = deferred.*.get(); return .ok; } pub export fn napi_resolve_deferred(env: napi_env, deferred: napi_deferred, resolution: napi_value) napi_status { log("napi_resolve_deferred", .{}); var prom = JSC.JSPromise.Strong{ .strong = .{ .ref = deferred }, }; prom.resolve(env, resolution); return .ok; } pub export fn napi_reject_deferred(env: napi_env, deferred: napi_deferred, rejection: napi_value) napi_status { log("napi_reject_deferred", .{}); var prom = JSC.JSPromise.Strong{ .strong = .{ .ref = deferred }, }; prom.reject(env, rejection); return .ok; } pub export fn napi_is_promise(_: napi_env, value: napi_value, is_promise: *bool) napi_status { log("napi_is_promise", .{}); if (value.isEmptyOrUndefinedOrNull()) { is_promise.* = false; return .ok; } is_promise.* = value.asAnyPromise() != null; return .ok; } pub export fn napi_run_script(env: napi_env, script: napi_value, result: *napi_value) napi_status { log("napi_run_script", .{}); // TODO: don't copy var ref = JSC.C.JSValueToStringCopy(env.ref(), script.asObjectRef(), TODO_EXCEPTION); defer JSC.C.JSStringRelease(ref); var exception = [_]JSC.C.JSValueRef{null}; const val = JSC.C.JSEvaluateScript(env.ref(), ref, env.ref(), null, 0, &exception); if (exception[0] != null) { return genericFailure(); } result.* = JSValue.c(val); return .ok; } pub extern fn napi_adjust_external_memory(env: napi_env, change_in_bytes: i64, adjusted_value: [*c]i64) napi_status; pub export fn napi_create_date(env: napi_env, time: f64, result: *napi_value) napi_status { log("napi_create_date", .{}); var args = [_]JSC.C.JSValueRef{JSC.JSValue.jsNumber(time).asObjectRef()}; result.* = JSValue.c(JSC.C.JSObjectMakeDate(env.ref(), 1, &args, TODO_EXCEPTION)); return .ok; } pub export fn napi_is_date(_: napi_env, value: napi_value, is_date: *bool) napi_status { log("napi_is_date", .{}); is_date.* = value.jsTypeLoose() == .JSDate; return .ok; } pub export fn napi_get_date_value(env: napi_env, value: napi_value, result: *f64) napi_status { log("napi_get_date_value", .{}); const getTimeFunction = value.get(env, "getTime") orelse { return .date_expected; }; result.* = JSValue.c( JSC.C.JSObjectCallAsFunction(env.ref(), getTimeFunction.asObjectRef(), value.asObjectRef(), 0, null, TODO_EXCEPTION), ).asNumber(); return .ok; } pub extern fn napi_add_finalizer(env: napi_env, js_object: napi_value, native_object: ?*anyopaque, finalize_cb: napi_finalize, finalize_hint: ?*anyopaque, result: *Ref) napi_status; pub export fn napi_create_bigint_int64(env: napi_env, value: i64, result: *napi_value) napi_status { log("napi_create_bigint_int64", .{}); result.* = JSC.JSValue.fromInt64NoTruncate(env, value); return .ok; } pub export fn napi_create_bigint_uint64(env: napi_env, value: u64, result: *napi_value) napi_status { log("napi_create_bigint_uint64", .{}); result.* = JSC.JSValue.fromUInt64NoTruncate(env, value); return .ok; } pub extern fn napi_create_bigint_words(env: napi_env, sign_bit: c_int, word_count: usize, words: [*c]const u64, result: *napi_value) napi_status; // TODO: lossless pub export fn napi_get_value_bigint_int64(_: napi_env, value: napi_value, result: *i64, _: *bool) napi_status { log("napi_get_value_bigint_int64", .{}); result.* = value.toInt64(); return .ok; } // TODO: lossless pub export fn napi_get_value_bigint_uint64(_: napi_env, value: napi_value, result: *u64, _: *bool) napi_status { log("napi_get_value_bigint_uint64", .{}); result.* = value.toUInt64NoTruncate(); return .ok; } pub extern fn napi_get_value_bigint_words(env: napi_env, value: napi_value, sign_bit: [*c]c_int, word_count: [*c]usize, words: [*c]u64) napi_status; pub extern fn napi_get_all_property_names(env: napi_env, object: napi_value, key_mode: napi_key_collection_mode, key_filter: napi_key_filter, key_conversion: napi_key_conversion, result: *napi_value) napi_status; pub extern fn napi_set_instance_data(env: napi_env, data: ?*anyopaque, finalize_cb: napi_finalize, finalize_hint: ?*anyopaque) napi_status; pub extern fn napi_get_instance_data(env: napi_env, data: [*]*anyopaque) napi_status; pub extern fn napi_detach_arraybuffer(env: napi_env, arraybuffer: napi_value) napi_status; pub extern fn napi_is_detached_arraybuffer(env: napi_env, value: napi_value, result: *bool) napi_status; pub const struct_napi_async_work__ = opaque {}; const WorkPool = @import("../work_pool.zig").WorkPool; const WorkPoolTask = @import("../work_pool.zig").Task; /// must be globally allocated pub const napi_async_work = struct { task: WorkPoolTask = .{ .callback = &runFromThreadPool }, concurrent_task: JSC.ConcurrentTask = .{}, completion_task: ?*anyopaque = null, event_loop: *JSC.EventLoop, global: napi_env, execute: napi_async_execute_callback = null, complete: napi_async_complete_callback = null, ctx: ?*anyopaque = null, status: std.atomic.Atomic(u32) = std.atomic.Atomic(u32).init(0), can_deinit: bool = false, wait_for_deinit: bool = false, scheduled: bool = false, ref: JSC.PollRef = .{}, pub const Status = enum(u32) { pending = 0, started = 1, completed = 2, cancelled = 3, }; pub fn create(global: napi_env, execute: napi_async_execute_callback, complete: napi_async_complete_callback, ctx: ?*anyopaque) !*napi_async_work { var work = try bun.default_allocator.create(napi_async_work); work.* = .{ .global = global, .execute = execute, .event_loop = global.bunVM().eventLoop(), .complete = complete, .ctx = ctx, }; return work; } pub fn runFromThreadPool(task: *WorkPoolTask) void { var this = @fieldParentPtr(napi_async_work, "task", task); this.run(); } pub fn run(this: *napi_async_work) void { if (this.status.compareAndSwap(@intFromEnum(Status.pending), @intFromEnum(Status.started), .SeqCst, .SeqCst)) |state| { if (state == @intFromEnum(Status.cancelled)) { if (this.wait_for_deinit) { // this might cause a segfault due to Task using a linked list! bun.default_allocator.destroy(this); } } return; } this.execute.?(this.global, this.ctx); this.status.store(@intFromEnum(Status.completed), .SeqCst); this.event_loop.enqueueTaskConcurrent(this.concurrent_task.from(this)); } pub fn schedule(this: *napi_async_work) void { if (this.scheduled) return; this.scheduled = true; this.ref.ref(this.global.bunVM()); WorkPool.schedule(&this.task); } pub fn cancel(this: *napi_async_work) bool { this.ref.unref(this.global.bunVM()); return this.status.compareAndSwap(@intFromEnum(Status.cancelled), @intFromEnum(Status.pending), .SeqCst, .SeqCst) != null; } pub fn deinit(this: *napi_async_work) void { this.ref.unref(this.global.bunVM()); if (this.can_deinit) { bun.default_allocator.destroy(this); return; } this.wait_for_deinit = true; } pub fn runFromJS(this: *napi_async_work) void { this.complete.?( this.global, if (this.status.load(.SeqCst) == @intFromEnum(Status.cancelled)) napi_status.cancelled else napi_status.ok, this.ctx.?, ); } }; pub const napi_threadsafe_function = *ThreadSafeFunction; pub const napi_threadsafe_function_release_mode = enum(c_uint) { release = 0, abort = 1, }; pub const napi_tsfn_nonblocking = 0; pub const napi_tsfn_blocking = 1; pub const napi_threadsafe_function_call_mode = c_uint; pub const napi_async_execute_callback = ?*const fn (napi_env, ?*anyopaque) callconv(.C) void; pub const napi_async_complete_callback = ?*const fn (napi_env, napi_status, ?*anyopaque) callconv(.C) void; pub const napi_threadsafe_function_call_js = *const fn (napi_env, napi_value, ?*anyopaque, ?*anyopaque) callconv(.C) void; pub const napi_node_version = extern struct { major: u32, minor: u32, patch: u32, release: [*c]const u8, pub const global: napi_node_version = .{ .major = 17, .minor = 7, .patch = 17, .release = "Bun!!!", }; }; pub const struct_napi_async_cleanup_hook_handle__ = opaque {}; pub const napi_async_cleanup_hook_handle = ?*struct_napi_async_cleanup_hook_handle__; pub const napi_async_cleanup_hook = *const fn (napi_async_cleanup_hook_handle, ?*anyopaque) callconv(.C) void; pub const napi_addon_register_func = *const fn (napi_env, napi_value) callconv(.C) napi_value; pub const struct_napi_module = extern struct { nm_version: c_int, nm_flags: c_uint, nm_filename: [*c]const u8, nm_register_func: napi_addon_register_func, nm_modname: [*c]const u8, nm_priv: ?*anyopaque, reserved: [4]?*anyopaque, }; pub const napi_module = struct_napi_module; fn napiSpan(ptr: anytype, len: usize) []const u8 { if (ptr == null) return &[_]u8{}; if (len == NAPI_AUTO_LENGTH) { return bun.sliceTo(ptr.?, 0); } return ptr.?[0..len]; } pub export fn napi_fatal_error(location_ptr: ?[*:0]const u8, location_len: usize, message_ptr: ?[*:0]const u8, message_len_: usize) noreturn { log("napi_fatal_error", .{}); var message = napiSpan(message_ptr, message_len_); if (message.len == 0) { message = "fatal error"; } const location = napiSpan(location_ptr, location_len); if (location.len > 0) { bun.Global.panic("napi: {s}\n {s}", .{ message, location }); } bun.Global.panic("napi: {s}", .{message}); } pub export fn napi_create_buffer(env: napi_env, length: usize, data: ?**anyopaque, result: *napi_value) napi_status { log("napi_create_buffer: {d}", .{length}); var buffer = JSC.JSValue.createBufferFromLength(env, length); if (length > 0) { if (data) |ptr| { ptr.* = buffer.asArrayBuffer(env).?.ptr; } } result.* = buffer; return .ok; } pub export fn napi_create_external_buffer(env: napi_env, length: usize, data: ?*anyopaque, finalize_cb: napi_finalize, finalize_hint: ?*anyopaque, result: *napi_value) napi_status { log("napi_create_external_buffer: {d}", .{length}); var buf = JSC.ExternalBuffer.create(finalize_hint, @ptrCast([*]u8, data.?)[0..length], env, finalize_cb, bun.default_allocator) catch { return genericFailure(); }; result.* = buf.toJS(env); return .ok; } pub export fn napi_create_buffer_copy(env: napi_env, length: usize, data: [*]u8, result_data: ?*?*anyopaque, result: *napi_value) napi_status { log("napi_create_buffer_copy: {d}", .{length}); var buffer = JSC.JSValue.createBufferFromLength(env, length); if (buffer.asArrayBuffer(env)) |array_buf| { if (length > 0) { @memcpy(array_buf.slice()[0..length], data[0..length]); } if (result_data) |ptr| { ptr.* = if (length > 0) array_buf.ptr else null; } } result.* = buffer; return .ok; } pub export fn napi_is_buffer(env: napi_env, value: napi_value, result: *bool) napi_status { log("napi_is_buffer", .{}); result.* = value.isBuffer(env); return .ok; } pub export fn napi_get_buffer_info(env: napi_env, value: napi_value, data: *[*]u8, length: *usize) napi_status { log("napi_get_buffer_info", .{}); const array_buf = value.asArrayBuffer(env) orelse { // TODO: is invalid_arg what to return here? return .arraybuffer_expected; }; data.* = array_buf.ptr; length.* = array_buf.byte_len; return .ok; } pub export fn napi_create_async_work( env: napi_env, _: napi_value, _: [*:0]const u8, execute: napi_async_execute_callback, complete: napi_async_complete_callback, data: ?*anyopaque, result: **napi_async_work, ) napi_status { log("napi_create_async_work", .{}); result.* = napi_async_work.create(env, execute, complete, data) catch { return genericFailure(); }; return .ok; } pub export fn napi_delete_async_work(env: napi_env, work: *napi_async_work) napi_status { log("napi_delete_async_work", .{}); std.debug.assert(env == work.global); work.deinit(); return .ok; } pub export fn napi_queue_async_work(env: napi_env, work: *napi_async_work) napi_status { log("napi_queue_async_work", .{}); std.debug.assert(env == work.global); work.schedule(); return .ok; } pub export fn napi_cancel_async_work(env: napi_env, work: *napi_async_work) napi_status { log("napi_cancel_async_work", .{}); std.debug.assert(env == work.global); if (work.cancel()) { return .ok; } return napi_status.generic_failure; } pub export fn napi_get_node_version(_: napi_env, version: **const napi_node_version) napi_status { log("napi_get_node_version", .{}); version.* = &napi_node_version.global; return .ok; } pub export fn napi_get_uv_event_loop(env: napi_env, loop: **JSC.EventLoop) napi_status { log("napi_get_uv_event_loop", .{}); // lol loop.* = env.bunVM().eventLoop(); return .ok; } pub extern fn napi_fatal_exception(env: napi_env, err: napi_value) napi_status; // We use a linked list here because we assume removing these is relatively rare // and array reallocations are relatively expensive. pub export fn napi_add_env_cleanup_hook(env: napi_env, fun: ?*const fn (?*anyopaque) callconv(.C) void, arg: ?*anyopaque) napi_status { log("napi_add_env_cleanup_hook", .{}); if (fun == null) return .ok; env.bunVM().rareData().pushCleanupHook(env, arg, fun.?); return .ok; } pub export fn napi_remove_env_cleanup_hook(env: napi_env, fun: ?*const fn (?*anyopaque) callconv(.C) void, arg: ?*anyopaque) napi_status { log("napi_remove_env_cleanup_hook", .{}); if (env.bunVM().rare_data == null or fun == null) return .ok; var rare_data = env.bunVM().rare_data.?; var hook = rare_data.cleanup_hook orelse return .ok; const cmp = JSC.RareData.CleanupHook.from(env, arg, fun.?); if (hook.eql(cmp)) { env.bunVM().allocator.destroy(hook); rare_data.cleanup_hook = null; rare_data.tail_cleanup_hook = null; } while (hook.next) |current| { if (hook.eql(cmp)) { if (current.next) |next| { hook.next = next; } else { hook.next = null; } env.bunVM().allocator.destroy(current); return .ok; } hook = current; } return .ok; } pub const Finalizer = struct { fun: napi_finalize, ctx: ?*anyopaque = null, }; // TODO: generate comptime version of this instead of runtime checking pub const ThreadSafeFunction = struct { pub const Callback = union(enum) { js: JSValue, c: struct { js: JSValue, napi_threadsafe_function_call_js: napi_threadsafe_function_call_js, }, }; /// thread-safe functions can be "referenced" and "unreferenced". A /// "referenced" thread-safe function will cause the event loop on the thread /// on which it is created to remain alive until the thread-safe function is /// destroyed. In contrast, an "unreferenced" thread-safe function will not /// prevent the event loop from exiting. The APIs napi_ref_threadsafe_function /// and napi_unref_threadsafe_function exist for this purpose. /// /// Neither does napi_unref_threadsafe_function mark the thread-safe /// functions as able to be destroyed nor does napi_ref_threadsafe_function /// prevent it from being destroyed. ref_for_process_exit: bool = false, owning_threads: std.AutoArrayHashMapUnmanaged(u64, void) = .{}, owning_thread_lock: Lock = Lock.init(), event_loop: *JSC.EventLoop, concurrent_task: JSC.ConcurrentTask = .{}, concurrent_finalizer_task: JSC.ConcurrentTask = .{}, env: napi_env, finalizer_task: JSC.AnyTask = undefined, finalizer: Finalizer = Finalizer{ .fun = null, .ctx = null }, channel: Queue, ctx: ?*anyopaque = null, callback: Callback = undefined, const ThreadSafeFunctionTask = JSC.AnyTask.New(@This(), call); pub const Queue = union(enum) { sized: Channel(?*anyopaque, .Slice), unsized: Channel(?*anyopaque, .Dynamic), pub fn isClosed(this: *const @This()) bool { return @atomicLoad( bool, switch (this.*) { .sized => &this.sized.is_closed, .unsized => &this.unsized.is_closed, }, .SeqCst, ); } pub fn close(this: *@This()) void { switch (this.*) { .sized => this.sized.close(), .unsized => this.unsized.close(), } } pub fn init(size: usize, allocator: std.mem.Allocator) @This() { switch (size) { 0 => { return .{ .unsized = Channel(?*anyopaque, .Dynamic).init(allocator), }; }, else => { var slice = allocator.alloc(?*anyopaque, size) catch unreachable; return .{ .sized = Channel(?*anyopaque, .Slice).init(slice), }; }, } } pub fn writeItem(this: *@This(), value: ?*anyopaque) !void { switch (this.*) { .sized => try this.sized.writeItem(value), .unsized => try this.unsized.writeItem(value), } } pub fn readItem(this: *@This()) !?*anyopaque { return switch (this.*) { .sized => try this.sized.readItem(), .unsized => try this.unsized.readItem(), }; } pub fn tryWriteItem(this: *@This(), value: ?*anyopaque) !bool { return switch (this.*) { .sized => try this.sized.tryWriteItem(value), .unsized => try this.unsized.tryWriteItem(value), }; } pub fn tryReadItem(this: *@This()) !??*anyopaque { return switch (this.*) { .sized => try this.sized.tryReadItem(), .unsized => try this.unsized.tryReadItem(), }; } }; pub fn call(this: *ThreadSafeFunction) void { var task = this.channel.tryReadItem() catch null orelse return; switch (this.callback) { .js => |js_function| { if (js_function.isEmptyOrUndefinedOrNull()) { return; } const err = js_function.call(this.env, &.{}); if (err.isAnyError()) { this.env.bunVM().onUnhandledError(this.env, err); } }, .c => |cb| { cb.napi_threadsafe_function_call_js(this.env, cb.js, this.ctx, task); }, } } pub fn enqueue(this: *ThreadSafeFunction, ctx: ?*anyopaque, block: bool) !void { if (block) { try this.channel.writeItem(ctx); } else { if (!try this.channel.tryWriteItem(ctx)) { return error.WouldBlock; } } this.event_loop.enqueueTaskConcurrent(this.concurrent_task.from(this)); } pub fn finalize(opaq: *anyopaque) void { var this = bun.cast(*ThreadSafeFunction, opaq); if (this.finalizer.fun) |fun| { fun(this.event_loop.global, opaq, this.finalizer.ctx); } if (this.callback == .js) { if (!this.callback.js.isEmptyOrUndefinedOrNull()) { this.callback.js.unprotect(); } } else if (this.callback == .c) { if (!this.callback.c.js.isEmptyOrUndefinedOrNull()) { this.callback.c.js.unprotect(); } } bun.default_allocator.destroy(this); } pub fn ref(this: *ThreadSafeFunction) void { this.ref_for_process_exit = true; } pub fn unref(this: *ThreadSafeFunction) void { this.ref_for_process_exit = false; } pub fn acquire(this: *ThreadSafeFunction) !void { this.owning_thread_lock.lock(); defer this.owning_thread_lock.unlock(); if (this.channel.isClosed()) return error.Closed; _ = this.owning_threads.getOrPut(bun.default_allocator, std.Thread.getCurrentId()) catch unreachable; } pub fn release(this: *ThreadSafeFunction, mode: napi_threadsafe_function_release_mode) void { this.owning_thread_lock.lock(); defer this.owning_thread_lock.unlock(); if (!this.owning_threads.swapRemove(std.Thread.getCurrentId())) return; if (mode == .abort) { this.channel.close(); } if (this.owning_threads.count() == 0) { this.finalizer_task = JSC.AnyTask{ .ctx = this, .callback = finalize }; this.event_loop.enqueueTaskConcurrent(this.concurrent_finalizer_task.from(&this.finalizer_task)); return; } } }; pub export fn napi_create_threadsafe_function( env: napi_env, func: napi_value, _: napi_value, _: napi_value, max_queue_size: usize, initial_thread_count: usize, thread_finalize_data: ?*anyopaque, thread_finalize_cb: napi_finalize, context: ?*anyopaque, call_js_cb: ?napi_threadsafe_function_call_js, result: *napi_threadsafe_function, ) napi_status { log("napi_create_threadsafe_function", .{}); if (call_js_cb == null and (func.isEmptyOrUndefinedOrNull() or !func.isCallable(env.vm()))) { return napi_status.function_expected; } if (!func.isEmptyOrUndefinedOrNull()) { func.protect(); } var function = bun.default_allocator.create(ThreadSafeFunction) catch return genericFailure(); function.* = .{ .event_loop = env.bunVM().eventLoop(), .env = env, .callback = if (call_js_cb) |c| .{ .c = .{ .napi_threadsafe_function_call_js = c, .js = if (func == .zero) JSC.JSValue.jsUndefined() else func, }, } else .{ .js = if (func == .zero) JSC.JSValue.jsUndefined() else func, }, .ctx = context, .channel = ThreadSafeFunction.Queue.init(max_queue_size, bun.default_allocator), .owning_threads = .{}, }; function.owning_threads.ensureTotalCapacity(bun.default_allocator, initial_thread_count) catch return genericFailure(); function.finalizer = .{ .ctx = thread_finalize_data, .fun = thread_finalize_cb }; result.* = function; return .ok; } pub export fn napi_get_threadsafe_function_context(func: napi_threadsafe_function, result: *?*anyopaque) napi_status { log("napi_get_threadsafe_function_context", .{}); result.* = func.ctx; return .ok; } pub export fn napi_call_threadsafe_function(func: napi_threadsafe_function, data: ?*anyopaque, is_blocking: napi_threadsafe_function_call_mode) napi_status { log("napi_call_threadsafe_function", .{}); func.enqueue(data, is_blocking == napi_tsfn_blocking) catch |err| { switch (err) { error.WouldBlock => { return napi_status.queue_full; }, else => return .closing, } }; return .ok; } pub export fn napi_acquire_threadsafe_function(func: napi_threadsafe_function) napi_status { log("napi_acquire_threadsafe_function", .{}); func.acquire() catch return .closing; return .ok; } pub export fn napi_release_threadsafe_function(func: napi_threadsafe_function, mode: napi_threadsafe_function_release_mode) napi_status { log("napi_release_threadsafe_function", .{}); func.release(mode); return .ok; } pub export fn napi_unref_threadsafe_function(env: napi_env, func: napi_threadsafe_function) napi_status { log("napi_unref_threadsafe_function", .{}); std.debug.assert(func.event_loop.global == env); func.unref(); return .ok; } pub export fn napi_ref_threadsafe_function(env: napi_env, func: napi_threadsafe_function) napi_status { log("napi_ref_threadsafe_function", .{}); std.debug.assert(func.event_loop.global == env); func.ref(); return .ok; } pub export fn napi_add_async_cleanup_hook(_: napi_env, _: napi_async_cleanup_hook, _: ?*anyopaque, _: [*c]napi_async_cleanup_hook_handle) napi_status { log("napi_add_async_cleanup_hook", .{}); // TODO: return .ok; } pub export fn napi_remove_async_cleanup_hook(_: napi_async_cleanup_hook_handle) napi_status { log("napi_remove_async_cleanup_hook", .{}); // TODO: return .ok; } pub const NAPI_VERSION_EXPERIMENTAL = @import("std").zig.c_translation.promoteIntLiteral(c_int, 2147483647, .decimal); pub const NAPI_VERSION = @as(c_int, 8); pub const NAPI_AUTO_LENGTH = std.math.maxInt(usize); pub const SRC_NODE_API_TYPES_H_ = ""; pub const NAPI_MODULE_VERSION = @as(c_int, 1); pub fn fixDeadCodeElimination() void { JSC.markBinding(@src()); std.mem.doNotOptimizeAway(&napi_acquire_threadsafe_function); std.mem.doNotOptimizeAway(&napi_add_async_cleanup_hook); std.mem.doNotOptimizeAway(&napi_add_env_cleanup_hook); std.mem.doNotOptimizeAway(&napi_add_finalizer); std.mem.doNotOptimizeAway(&napi_adjust_external_memory); std.mem.doNotOptimizeAway(&napi_async_destroy); std.mem.doNotOptimizeAway(&napi_async_init); std.mem.doNotOptimizeAway(&napi_call_function); std.mem.doNotOptimizeAway(&napi_call_threadsafe_function); std.mem.doNotOptimizeAway(&napi_cancel_async_work); std.mem.doNotOptimizeAway(&napi_check_object_type_tag); std.mem.doNotOptimizeAway(&napi_close_callback_scope); std.mem.doNotOptimizeAway(&napi_close_escapable_handle_scope); std.mem.doNotOptimizeAway(&napi_close_handle_scope); std.mem.doNotOptimizeAway(&napi_coerce_to_bool); std.mem.doNotOptimizeAway(&napi_coerce_to_number); std.mem.doNotOptimizeAway(&napi_coerce_to_object); std.mem.doNotOptimizeAway(&napi_create_array); std.mem.doNotOptimizeAway(&napi_create_array_with_length); std.mem.doNotOptimizeAway(&napi_create_arraybuffer); std.mem.doNotOptimizeAway(&napi_create_async_work); std.mem.doNotOptimizeAway(&napi_create_bigint_int64); std.mem.doNotOptimizeAway(&napi_create_bigint_uint64); std.mem.doNotOptimizeAway(&napi_create_bigint_words); std.mem.doNotOptimizeAway(&napi_create_buffer); std.mem.doNotOptimizeAway(&napi_create_buffer_copy); std.mem.doNotOptimizeAway(&napi_create_dataview); std.mem.doNotOptimizeAway(&napi_create_date); std.mem.doNotOptimizeAway(&napi_create_double); std.mem.doNotOptimizeAway(&napi_create_error); std.mem.doNotOptimizeAway(&napi_create_external); std.mem.doNotOptimizeAway(&napi_create_external_arraybuffer); std.mem.doNotOptimizeAway(&napi_create_external_buffer); std.mem.doNotOptimizeAway(&napi_create_int32); std.mem.doNotOptimizeAway(&napi_create_int64); std.mem.doNotOptimizeAway(&napi_create_object); std.mem.doNotOptimizeAway(&napi_create_promise); std.mem.doNotOptimizeAway(&napi_create_range_error); std.mem.doNotOptimizeAway(&napi_create_reference); std.mem.doNotOptimizeAway(&napi_create_string_latin1); std.mem.doNotOptimizeAway(&napi_create_string_utf16); std.mem.doNotOptimizeAway(&napi_create_string_utf8); std.mem.doNotOptimizeAway(&napi_create_symbol); std.mem.doNotOptimizeAway(&napi_create_threadsafe_function); std.mem.doNotOptimizeAway(&napi_create_type_error); std.mem.doNotOptimizeAway(&napi_create_typedarray); std.mem.doNotOptimizeAway(&napi_create_uint32); std.mem.doNotOptimizeAway(&napi_define_class); std.mem.doNotOptimizeAway(&napi_define_properties); std.mem.doNotOptimizeAway(&napi_delete_async_work); std.mem.doNotOptimizeAway(&napi_delete_reference); std.mem.doNotOptimizeAway(&napi_detach_arraybuffer); std.mem.doNotOptimizeAway(&napi_escape_handle); std.mem.doNotOptimizeAway(&napi_fatal_error); std.mem.doNotOptimizeAway(&napi_fatal_exception); std.mem.doNotOptimizeAway(&napi_get_all_property_names); std.mem.doNotOptimizeAway(&napi_get_and_clear_last_exception); std.mem.doNotOptimizeAway(&napi_get_array_length); std.mem.doNotOptimizeAway(&napi_get_arraybuffer_info); std.mem.doNotOptimizeAway(&napi_get_boolean); std.mem.doNotOptimizeAway(&napi_get_buffer_info); std.mem.doNotOptimizeAway(&napi_get_cb_info); std.mem.doNotOptimizeAway(&napi_get_dataview_info); std.mem.doNotOptimizeAway(&napi_get_date_value); std.mem.doNotOptimizeAway(&napi_get_element); std.mem.doNotOptimizeAway(&napi_get_global); std.mem.doNotOptimizeAway(&napi_get_instance_data); std.mem.doNotOptimizeAway(&napi_get_last_error_info); std.mem.doNotOptimizeAway(&napi_get_new_target); std.mem.doNotOptimizeAway(&napi_get_node_version); std.mem.doNotOptimizeAway(&napi_get_null); std.mem.doNotOptimizeAway(&napi_get_prototype); std.mem.doNotOptimizeAway(&napi_get_reference_value); std.mem.doNotOptimizeAway(&napi_get_reference_value_internal); std.mem.doNotOptimizeAway(&napi_get_threadsafe_function_context); std.mem.doNotOptimizeAway(&napi_get_typedarray_info); std.mem.doNotOptimizeAway(&napi_get_undefined); std.mem.doNotOptimizeAway(&napi_get_uv_event_loop); std.mem.doNotOptimizeAway(&napi_get_value_bigint_int64); std.mem.doNotOptimizeAway(&napi_get_value_bigint_uint64); std.mem.doNotOptimizeAway(&napi_get_value_bigint_words); std.mem.doNotOptimizeAway(&napi_get_value_bool); std.mem.doNotOptimizeAway(&napi_get_value_double); std.mem.doNotOptimizeAway(&napi_get_value_external); std.mem.doNotOptimizeAway(&napi_get_value_int32); std.mem.doNotOptimizeAway(&napi_get_value_int64); std.mem.doNotOptimizeAway(&napi_get_value_string_latin1); std.mem.doNotOptimizeAway(&napi_get_value_string_utf16); std.mem.doNotOptimizeAway(&napi_get_value_string_utf8); std.mem.doNotOptimizeAway(&napi_get_value_uint32); std.mem.doNotOptimizeAway(&napi_get_version); std.mem.doNotOptimizeAway(&napi_has_element); std.mem.doNotOptimizeAway(&napi_instanceof); std.mem.doNotOptimizeAway(&napi_is_array); std.mem.doNotOptimizeAway(&napi_is_arraybuffer); std.mem.doNotOptimizeAway(&napi_is_buffer); std.mem.doNotOptimizeAway(&napi_is_dataview); std.mem.doNotOptimizeAway(&napi_is_date); std.mem.doNotOptimizeAway(&napi_is_detached_arraybuffer); std.mem.doNotOptimizeAway(&napi_is_error); std.mem.doNotOptimizeAway(&napi_is_exception_pending); std.mem.doNotOptimizeAway(&napi_is_promise); std.mem.doNotOptimizeAway(&napi_is_typedarray); std.mem.doNotOptimizeAway(&napi_make_callback); std.mem.doNotOptimizeAway(&napi_new_instance); std.mem.doNotOptimizeAway(&napi_open_callback_scope); std.mem.doNotOptimizeAway(&napi_open_escapable_handle_scope); std.mem.doNotOptimizeAway(&napi_open_handle_scope); std.mem.doNotOptimizeAway(&napi_queue_async_work); std.mem.doNotOptimizeAway(&napi_ref_threadsafe_function); std.mem.doNotOptimizeAway(&napi_reference_ref); std.mem.doNotOptimizeAway(&napi_reference_unref); std.mem.doNotOptimizeAway(&napi_reject_deferred); std.mem.doNotOptimizeAway(&napi_release_threadsafe_function); std.mem.doNotOptimizeAway(&napi_remove_async_cleanup_hook); std.mem.doNotOptimizeAway(&napi_remove_env_cleanup_hook); std.mem.doNotOptimizeAway(&napi_remove_wrap); std.mem.doNotOptimizeAway(&napi_resolve_deferred); std.mem.doNotOptimizeAway(&napi_run_script); std.mem.doNotOptimizeAway(&napi_set_element); std.mem.doNotOptimizeAway(&napi_set_instance_data); std.mem.doNotOptimizeAway(&napi_strict_equals); std.mem.doNotOptimizeAway(&napi_throw); std.mem.doNotOptimizeAway(&napi_throw_error); std.mem.doNotOptimizeAway(&napi_throw_range_error); std.mem.doNotOptimizeAway(&napi_throw_type_error); std.mem.doNotOptimizeAway(&napi_type_tag_object); std.mem.doNotOptimizeAway(&napi_typeof); std.mem.doNotOptimizeAway(&napi_unref_threadsafe_function); std.mem.doNotOptimizeAway(&napi_unwrap); std.mem.doNotOptimizeAway(&napi_wrap); std.mem.doNotOptimizeAway(&@import("../bun.js/node/buffer.zig").BufferVectorized.fill); }