neon/sys/call.rs
1use std::{mem::MaybeUninit, ptr::null_mut};
2
3use smallvec::SmallVec;
4
5use super::{
6 bindings as napi,
7 raw::{Env, FunctionCallbackInfo, Local},
8};
9
10// Number of arguments to allocate on the stack. This should be large enough
11// to cover most use cases without being wasteful.
12//
13// * If the number is too large, too much space is allocated and then filled
14// with `undefined`.
15// * If the number is too small, getting arguments frequently takes two tries
16// and requires heap allocation.
17const ARGV_SIZE: usize = 4;
18
19#[repr(transparent)]
20/// List of JavaScript arguments to a function
21// `Arguments` is intended to be a small abstraction to hide the usage of
22// `SmallVec` allowing changes to `ARGV_SIZE` in a single location
23pub struct Arguments(SmallVec<[Local; ARGV_SIZE]>);
24
25impl Arguments {
26 #[inline]
27 /// Get an argument at a specific position
28 pub fn get(&self, i: usize) -> Option<Local> {
29 self.0.get(i).cloned()
30 }
31}
32
33pub unsafe fn is_construct(env: Env, info: FunctionCallbackInfo) -> bool {
34 let mut target: MaybeUninit<Local> = MaybeUninit::zeroed();
35
36 napi::get_new_target(env, info, target.as_mut_ptr()).unwrap();
37
38 // get_new_target is guaranteed to assign to target, so it's initialized.
39 let target: Local = target.assume_init();
40
41 // By the get_new_target contract, target will either be NULL if the current
42 // function was called without `new`, or a valid napi_value handle if the current
43 // function was called with `new`.
44 !target.is_null()
45}
46
47pub unsafe fn this(env: Env, info: FunctionCallbackInfo, out: &mut Local) {
48 napi::get_cb_info(env, info, null_mut(), null_mut(), out as *mut _, null_mut()).unwrap();
49}
50
51/// Gets the number of arguments passed to the function.
52// TODO: Remove this when `FunctionContext` is refactored to get call info upfront.
53pub unsafe fn len(env: Env, info: FunctionCallbackInfo) -> usize {
54 let mut argc = 0usize;
55 napi::get_cb_info(
56 env,
57 info,
58 &mut argc as *mut _,
59 null_mut(),
60 null_mut(),
61 null_mut(),
62 )
63 .unwrap();
64 argc
65}
66
67/// Returns the function arguments for a call
68pub unsafe fn argv(env: Env, info: FunctionCallbackInfo) -> Arguments {
69 // Allocate space on the stack for up to `ARGV_SIZE` values
70 let mut argv = MaybeUninit::<[Local; ARGV_SIZE]>::uninit();
71
72 // Starts as the size allocated; after `get_cb_info` it is the number of arguments
73 let mut argc = ARGV_SIZE;
74
75 napi::get_cb_info(
76 env,
77 info,
78 &mut argc as *mut _,
79 argv.as_mut_ptr().cast(),
80 null_mut(),
81 null_mut(),
82 )
83 .unwrap();
84
85 // We did not allocate enough space; allocate on the heap and try again
86 let argv = if argc > ARGV_SIZE {
87 // We know exactly how much space to reserve
88 let mut argv = Vec::with_capacity(argc);
89
90 napi::get_cb_info(
91 env,
92 info,
93 &mut argc as *mut _,
94 argv.as_mut_ptr(),
95 null_mut(),
96 null_mut(),
97 )
98 .unwrap();
99
100 // Set the size of `argv` to the number of initialized elements
101 argv.set_len(argc);
102 SmallVec::from_vec(argv)
103
104 // There were `ARGV_SIZE` or fewer arguments, use the stack allocated space
105 } else {
106 SmallVec::from_buf_and_len(argv.assume_init(), argc)
107 };
108
109 Arguments(argv)
110}