2023-06-24 07:54:10 -04:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
use super::generator_state::GeneratorState;
|
|
|
|
use super::signature::Arg;
|
|
|
|
use super::signature::NumericArg;
|
|
|
|
use super::signature::ParsedSignature;
|
|
|
|
use super::signature::RetVal;
|
|
|
|
use super::signature::Special;
|
2023-06-26 11:38:55 -04:00
|
|
|
use super::MacroConfig;
|
2023-06-24 07:54:10 -04:00
|
|
|
use super::V8MappingError;
|
|
|
|
use proc_macro2::TokenStream;
|
|
|
|
use quote::quote;
|
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
pub(crate) fn generate_dispatch_slow(
|
|
|
|
config: &MacroConfig,
|
2023-06-24 07:54:10 -04:00
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
signature: &ParsedSignature,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
let mut output = TokenStream::new();
|
2023-06-25 10:36:09 -04:00
|
|
|
|
|
|
|
// Fast ops require the slow op to check op_ctx for the last error
|
|
|
|
if config.fast && matches!(signature.ret_val, RetVal::Result(_)) {
|
|
|
|
generator_state.needs_opctx = true;
|
|
|
|
let throw_exception = throw_exception(generator_state)?;
|
|
|
|
// If the fast op returned an error, we must throw it rather than doing work.
|
|
|
|
output.extend(quote!{
|
|
|
|
// FASTCALL FALLBACK: This is where we pick up the errors for the slow-call error pickup
|
|
|
|
// path. There is no code running between this and the other FASTCALL FALLBACK comment,
|
|
|
|
// except some V8 code required to perform the fallback process. This is why the below call is safe.
|
|
|
|
|
|
|
|
// SAFETY: We guarantee that OpCtx has no mutable references once ops are live and being called,
|
|
|
|
// allowing us to perform this one little bit of mutable magic.
|
|
|
|
if let Some(err) = unsafe { opctx.unsafely_take_last_error_for_ops_only() } {
|
|
|
|
#throw_exception
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-06-24 07:54:10 -04:00
|
|
|
for (index, arg) in signature.args.iter().enumerate() {
|
|
|
|
output.extend(extract_arg(generator_state, index)?);
|
|
|
|
output.extend(from_arg(generator_state, index, arg)?);
|
|
|
|
}
|
2023-06-24 17:30:04 -04:00
|
|
|
output.extend(call(generator_state)?);
|
|
|
|
output.extend(return_value(generator_state, &signature.ret_val)?);
|
2023-06-24 07:54:10 -04:00
|
|
|
|
|
|
|
let with_scope = if generator_state.needs_scope {
|
2023-06-24 17:30:04 -04:00
|
|
|
with_scope(generator_state)
|
2023-06-24 07:54:10 -04:00
|
|
|
} else {
|
|
|
|
quote!()
|
|
|
|
};
|
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
let with_opctx = if generator_state.needs_opctx {
|
|
|
|
with_opctx(generator_state)
|
|
|
|
} else {
|
|
|
|
quote!()
|
|
|
|
};
|
|
|
|
|
2023-06-24 07:54:10 -04:00
|
|
|
let with_retval = if generator_state.needs_retval {
|
2023-06-24 17:30:04 -04:00
|
|
|
with_retval(generator_state)
|
2023-06-24 07:54:10 -04:00
|
|
|
} else {
|
|
|
|
quote!()
|
|
|
|
};
|
|
|
|
|
|
|
|
let with_args = if generator_state.needs_args {
|
2023-06-24 17:30:04 -04:00
|
|
|
with_fn_args(generator_state)
|
2023-06-24 07:54:10 -04:00
|
|
|
} else {
|
|
|
|
quote!()
|
|
|
|
};
|
|
|
|
|
2023-06-24 17:30:04 -04:00
|
|
|
let GeneratorState {
|
|
|
|
deno_core,
|
|
|
|
info,
|
|
|
|
slow_function,
|
|
|
|
..
|
|
|
|
} = &generator_state;
|
|
|
|
|
2023-06-24 07:54:10 -04:00
|
|
|
Ok(quote! {
|
|
|
|
pub extern "C" fn #slow_function(#info: *const #deno_core::v8::FunctionCallbackInfo) {
|
|
|
|
#with_scope
|
|
|
|
#with_retval
|
|
|
|
#with_args
|
2023-06-25 10:36:09 -04:00
|
|
|
#with_opctx
|
2023-06-24 07:54:10 -04:00
|
|
|
|
|
|
|
#output
|
|
|
|
}})
|
|
|
|
}
|
|
|
|
|
2023-06-24 17:30:04 -04:00
|
|
|
fn with_scope(generator_state: &mut GeneratorState) -> TokenStream {
|
|
|
|
let GeneratorState {
|
|
|
|
deno_core,
|
|
|
|
scope,
|
|
|
|
info,
|
|
|
|
..
|
|
|
|
} = &generator_state;
|
|
|
|
|
|
|
|
quote!(let #scope = &mut unsafe { #deno_core::v8::CallbackScope::new(&*#info) };)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_retval(generator_state: &mut GeneratorState) -> TokenStream {
|
|
|
|
let GeneratorState {
|
|
|
|
deno_core,
|
|
|
|
retval,
|
|
|
|
info,
|
|
|
|
..
|
|
|
|
} = &generator_state;
|
|
|
|
|
|
|
|
quote!(let mut #retval = #deno_core::v8::ReturnValue::from_function_callback_info(unsafe { &*#info });)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_fn_args(generator_state: &mut GeneratorState) -> TokenStream {
|
|
|
|
let GeneratorState {
|
|
|
|
deno_core,
|
|
|
|
fn_args,
|
|
|
|
info,
|
|
|
|
..
|
|
|
|
} = &generator_state;
|
|
|
|
|
|
|
|
quote!(let #fn_args = #deno_core::v8::FunctionCallbackArguments::from_function_callback_info(unsafe { &*#info });)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn with_opctx(generator_state: &mut GeneratorState) -> TokenStream {
|
|
|
|
let GeneratorState {
|
|
|
|
deno_core,
|
|
|
|
opctx,
|
|
|
|
fn_args,
|
2023-06-25 10:36:09 -04:00
|
|
|
needs_args,
|
2023-06-24 17:30:04 -04:00
|
|
|
..
|
2023-06-25 10:36:09 -04:00
|
|
|
} = generator_state;
|
2023-06-24 17:30:04 -04:00
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
*needs_args = true;
|
2023-06-24 17:30:04 -04:00
|
|
|
quote!(let #opctx = unsafe {
|
|
|
|
&*(#deno_core::v8::Local::<#deno_core::v8::External>::cast(#fn_args.data()).value()
|
|
|
|
as *const #deno_core::_ops::OpCtx)
|
|
|
|
};)
|
|
|
|
}
|
|
|
|
|
2023-06-24 07:54:10 -04:00
|
|
|
pub fn extract_arg(
|
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
index: usize,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
let GeneratorState { fn_args, .. } = &generator_state;
|
|
|
|
let arg_ident = generator_state.args.get(index);
|
|
|
|
|
|
|
|
Ok(quote!(
|
|
|
|
let #arg_ident = #fn_args.get(#index as i32);
|
|
|
|
))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn from_arg(
|
|
|
|
mut generator_state: &mut GeneratorState,
|
|
|
|
index: usize,
|
|
|
|
arg: &Arg,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
let GeneratorState {
|
|
|
|
deno_core, args, ..
|
|
|
|
} = &mut generator_state;
|
|
|
|
let arg_ident = args.get_mut(index).expect("Argument at index was missing");
|
|
|
|
|
|
|
|
let res = match arg {
|
|
|
|
Arg::Numeric(NumericArg::bool) => quote! {
|
|
|
|
let #arg_ident = #arg_ident.is_true();
|
|
|
|
},
|
|
|
|
Arg::Numeric(NumericArg::u8)
|
|
|
|
| Arg::Numeric(NumericArg::u16)
|
|
|
|
| Arg::Numeric(NumericArg::u32) => {
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = #deno_core::_ops::to_u32(&#arg_ident) as _;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Arg::Numeric(NumericArg::i8)
|
|
|
|
| Arg::Numeric(NumericArg::i16)
|
|
|
|
| Arg::Numeric(NumericArg::i32)
|
|
|
|
| Arg::Numeric(NumericArg::__SMI__) => {
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = #deno_core::_ops::to_i32(&#arg_ident) as _;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Arg::Numeric(NumericArg::u64) | Arg::Numeric(NumericArg::usize) => {
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = #deno_core::_ops::to_u64(&#arg_ident) as _;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Arg::Numeric(NumericArg::i64) | Arg::Numeric(NumericArg::isize) => {
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = #deno_core::_ops::to_i64(&#arg_ident) as _;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Arg::OptionNumeric(numeric) => {
|
|
|
|
// Ends the borrow of generator_state
|
|
|
|
let arg_ident = arg_ident.clone();
|
|
|
|
let some = from_arg(generator_state, index, &Arg::Numeric(*numeric))?;
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = if #arg_ident.is_null_or_undefined() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
#some
|
|
|
|
Some(#arg_ident)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Arg::Option(Special::String) => {
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = #arg_ident.to_rust_string_lossy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Arg::Special(Special::RefStr) => {
|
|
|
|
quote! {
|
|
|
|
let #arg_ident = #arg_ident.to_rust_string_lossy();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => return Err(V8MappingError::NoMapping("a slow argument", arg.clone())),
|
|
|
|
};
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn call(
|
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
let GeneratorState { result, .. } = &generator_state;
|
|
|
|
|
|
|
|
let mut tokens = TokenStream::new();
|
|
|
|
for arg in &generator_state.args {
|
|
|
|
tokens.extend(quote!( #arg , ));
|
|
|
|
}
|
|
|
|
Ok(quote! {
|
|
|
|
let #result = Self::call( #tokens );
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn return_value(
|
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
ret_type: &RetVal,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
match ret_type {
|
|
|
|
RetVal::Infallible(ret_type) => {
|
|
|
|
return_value_infallible(generator_state, ret_type)
|
|
|
|
}
|
|
|
|
RetVal::Result(ret_type) => return_value_result(generator_state, ret_type),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn return_value_infallible(
|
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
ret_type: &Arg,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
let GeneratorState {
|
|
|
|
result,
|
|
|
|
retval,
|
|
|
|
needs_retval,
|
|
|
|
..
|
|
|
|
} = generator_state;
|
|
|
|
|
|
|
|
let res = match ret_type {
|
2023-06-24 17:30:04 -04:00
|
|
|
Arg::Void => {
|
|
|
|
quote! {/* void */}
|
|
|
|
}
|
2023-06-24 07:54:10 -04:00
|
|
|
Arg::Numeric(NumericArg::u8)
|
|
|
|
| Arg::Numeric(NumericArg::u16)
|
|
|
|
| Arg::Numeric(NumericArg::u32) => {
|
|
|
|
*needs_retval = true;
|
|
|
|
quote!(#retval.set_uint32(#result as u32);)
|
|
|
|
}
|
|
|
|
Arg::Numeric(NumericArg::i8)
|
|
|
|
| Arg::Numeric(NumericArg::i16)
|
|
|
|
| Arg::Numeric(NumericArg::i32) => {
|
|
|
|
*needs_retval = true;
|
|
|
|
quote!(#retval.set_int32(#result as i32);)
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
return Err(V8MappingError::NoMapping(
|
|
|
|
"a slow return value",
|
|
|
|
ret_type.clone(),
|
|
|
|
))
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
fn return_value_result(
|
2023-06-24 07:54:10 -04:00
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
ret_type: &Arg,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
|
|
|
let infallible = return_value_infallible(generator_state, ret_type)?;
|
2023-06-25 10:36:09 -04:00
|
|
|
let exception = throw_exception(generator_state)?;
|
|
|
|
|
|
|
|
let GeneratorState { result, .. } = &generator_state;
|
|
|
|
|
|
|
|
let tokens = quote!(
|
|
|
|
match #result {
|
|
|
|
Ok(#result) => {
|
|
|
|
#infallible
|
|
|
|
}
|
|
|
|
Err(err) => {
|
|
|
|
#exception
|
|
|
|
}
|
|
|
|
};
|
|
|
|
);
|
|
|
|
Ok(tokens)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Generates code to throw an exception, adding required additional dependencies as needed.
|
|
|
|
fn throw_exception(
|
|
|
|
generator_state: &mut GeneratorState,
|
|
|
|
) -> Result<TokenStream, V8MappingError> {
|
2023-06-24 17:30:04 -04:00
|
|
|
let maybe_scope = if generator_state.needs_scope {
|
|
|
|
quote!()
|
|
|
|
} else {
|
|
|
|
with_scope(generator_state)
|
|
|
|
};
|
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
let maybe_opctx = if generator_state.needs_opctx {
|
2023-06-24 17:30:04 -04:00
|
|
|
quote!()
|
|
|
|
} else {
|
2023-06-25 10:36:09 -04:00
|
|
|
with_opctx(generator_state)
|
2023-06-24 17:30:04 -04:00
|
|
|
};
|
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
let maybe_args = if generator_state.needs_args {
|
2023-06-24 17:30:04 -04:00
|
|
|
quote!()
|
|
|
|
} else {
|
2023-06-25 10:36:09 -04:00
|
|
|
with_fn_args(generator_state)
|
2023-06-24 17:30:04 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
let GeneratorState {
|
|
|
|
deno_core,
|
|
|
|
scope,
|
|
|
|
opctx,
|
|
|
|
..
|
|
|
|
} = &generator_state;
|
2023-06-24 07:54:10 -04:00
|
|
|
|
2023-06-25 10:36:09 -04:00
|
|
|
Ok(quote! {
|
|
|
|
#maybe_scope
|
|
|
|
#maybe_args
|
|
|
|
#maybe_opctx
|
|
|
|
let opstate = ::std::cell::RefCell::borrow(&*#opctx.state);
|
|
|
|
let exception = #deno_core::error::to_v8_error(
|
|
|
|
#scope,
|
|
|
|
opstate.get_error_class_fn,
|
|
|
|
&err,
|
|
|
|
);
|
|
|
|
scope.throw_exception(exception);
|
|
|
|
return;
|
|
|
|
})
|
2023-06-24 07:54:10 -04:00
|
|
|
}
|