2019-12-20 08:56:32 -05:00
|
|
|
use crate::support::int;
|
2019-12-20 20:54:14 -05:00
|
|
|
use crate::support::MaybeBool;
|
2019-12-20 08:56:32 -05:00
|
|
|
use crate::Context;
|
|
|
|
use crate::Local;
|
2019-12-30 09:28:39 -05:00
|
|
|
use crate::Module;
|
2019-12-20 20:54:14 -05:00
|
|
|
use crate::String;
|
2019-12-24 18:31:36 -05:00
|
|
|
use crate::ToLocal;
|
2019-12-20 08:56:32 -05:00
|
|
|
use crate::Value;
|
2019-12-23 07:12:52 -05:00
|
|
|
use std::mem::MaybeUninit;
|
2019-12-20 08:56:32 -05:00
|
|
|
|
2019-12-23 18:09:03 -05:00
|
|
|
#[allow(non_camel_case_types)]
|
|
|
|
type v8__Module__ResolveCallback =
|
2019-12-20 20:54:14 -05:00
|
|
|
extern "C" fn(Local<Context>, Local<String>, Local<Module>) -> *mut Module;
|
|
|
|
|
2019-12-23 20:23:55 -05:00
|
|
|
/// Called during Module::instantiate_module. Provided with arguments:
|
|
|
|
/// (context, specifier, referrer)
|
|
|
|
/// Return null on error.
|
|
|
|
/// Hint: to tranform Local<Module> to *mut Module do this:
|
|
|
|
/// &mut *module
|
|
|
|
pub type ResolveCallback =
|
|
|
|
fn(Local<Context>, Local<String>, Local<Module>) -> *mut Module;
|
2019-12-23 18:09:03 -05:00
|
|
|
|
2019-12-20 20:54:14 -05:00
|
|
|
extern "C" {
|
2019-12-23 07:12:52 -05:00
|
|
|
fn v8__Module__GetStatus(this: *const Module) -> ModuleStatus;
|
|
|
|
fn v8__Module__GetException(this: *const Module) -> *mut Value;
|
|
|
|
fn v8__Module__GetModuleRequestsLength(this: *const Module) -> int;
|
2019-12-30 15:11:50 -05:00
|
|
|
fn v8__Module__GetModuleRequest(this: *const Module, i: int) -> *mut String;
|
2019-12-23 07:12:52 -05:00
|
|
|
fn v8__Module__GetModuleRequestLocation(
|
|
|
|
this: *const Module,
|
|
|
|
i: usize,
|
|
|
|
out: &mut MaybeUninit<Location>,
|
|
|
|
) -> Location;
|
|
|
|
fn v8__Module__GetIdentityHash(this: *const Module) -> int;
|
2019-12-20 20:54:14 -05:00
|
|
|
fn v8__Module__InstantiateModule(
|
|
|
|
this: *mut Module,
|
|
|
|
context: Local<Context>,
|
2019-12-23 18:09:03 -05:00
|
|
|
callback: v8__Module__ResolveCallback,
|
2019-12-20 20:54:14 -05:00
|
|
|
) -> MaybeBool;
|
2019-12-23 07:12:52 -05:00
|
|
|
fn v8__Module__Evaluate(
|
|
|
|
this: *mut Module,
|
|
|
|
context: *mut Context,
|
|
|
|
) -> *mut Value;
|
|
|
|
fn v8__Location__GetLineNumber(this: &Location) -> int;
|
|
|
|
fn v8__Location__GetColumnNumber(this: &Location) -> int;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
/// A location in JavaScript source.
|
|
|
|
pub struct Location([usize; 1]);
|
|
|
|
|
|
|
|
impl Location {
|
|
|
|
pub fn get_line_number(&self) -> int {
|
|
|
|
unsafe { v8__Location__GetLineNumber(self) }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_column_number(&self) -> int {
|
|
|
|
unsafe { v8__Location__GetColumnNumber(self) }
|
|
|
|
}
|
2019-12-20 20:54:14 -05:00
|
|
|
}
|
2019-12-20 08:56:32 -05:00
|
|
|
|
|
|
|
/// The different states a module can be in.
|
|
|
|
///
|
|
|
|
/// This corresponds to the states used in ECMAScript except that "evaluated"
|
|
|
|
/// is split into kEvaluated and kErrored, indicating success and failure,
|
|
|
|
/// respectively.
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
|
|
#[repr(C)]
|
2019-12-23 07:12:52 -05:00
|
|
|
pub enum ModuleStatus {
|
2019-12-20 08:56:32 -05:00
|
|
|
Uninstantiated,
|
|
|
|
Instantiating,
|
|
|
|
Instantiated,
|
|
|
|
Evaluating,
|
|
|
|
Evaluated,
|
|
|
|
Errored,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Module {
|
|
|
|
/// Returns the module's current status.
|
2019-12-23 07:12:52 -05:00
|
|
|
pub fn get_status(&self) -> ModuleStatus {
|
|
|
|
unsafe { v8__Module__GetStatus(self) }
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// For a module in kErrored status, this returns the corresponding exception.
|
|
|
|
pub fn get_exception(&self) -> Local<Value> {
|
2019-12-25 06:39:42 -05:00
|
|
|
unsafe { Local::from_raw(v8__Module__GetException(self)).unwrap() }
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the number of modules requested by this module.
|
2019-12-30 15:11:50 -05:00
|
|
|
pub fn get_module_requests_length(&self) -> usize {
|
|
|
|
unsafe { v8__Module__GetModuleRequestsLength(self) as usize }
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the ith module specifier in this module.
|
|
|
|
/// i must be < self.get_module_requests_length() and >= 0.
|
2019-12-23 07:12:52 -05:00
|
|
|
pub fn get_module_request(&self, i: usize) -> Local<String> {
|
2019-12-30 15:11:50 -05:00
|
|
|
unsafe {
|
|
|
|
Local::from_raw(v8__Module__GetModuleRequest(self, i as int)).unwrap()
|
|
|
|
}
|
2019-12-23 07:12:52 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the source location (line number and column number) of the ith
|
|
|
|
/// module specifier's first occurrence in this module.
|
|
|
|
pub fn get_module_request_location(&self, i: usize) -> Location {
|
|
|
|
let mut out = MaybeUninit::<Location>::uninit();
|
|
|
|
unsafe {
|
|
|
|
v8__Module__GetModuleRequestLocation(self, i, &mut out);
|
|
|
|
out.assume_init()
|
|
|
|
}
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the identity hash for this object.
|
|
|
|
pub fn get_identity_hash(&self) -> int {
|
2019-12-23 07:12:52 -05:00
|
|
|
unsafe { v8__Module__GetIdentityHash(self) }
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Instantiates the module and its dependencies.
|
|
|
|
///
|
|
|
|
/// Returns an empty Maybe<bool> if an exception occurred during
|
|
|
|
/// instantiation. (In the case where the callback throws an exception, that
|
|
|
|
/// exception is propagated.)
|
|
|
|
#[must_use]
|
|
|
|
pub fn instantiate_module(
|
2019-12-20 20:54:14 -05:00
|
|
|
&mut self,
|
|
|
|
context: Local<Context>,
|
|
|
|
callback: ResolveCallback,
|
2019-12-20 08:56:32 -05:00
|
|
|
) -> Option<bool> {
|
2019-12-23 18:09:03 -05:00
|
|
|
use std::sync::Mutex;
|
|
|
|
lazy_static! {
|
|
|
|
static ref RESOLVE_CALLBACK: Mutex<Option<ResolveCallback>> =
|
|
|
|
Mutex::new(None);
|
|
|
|
static ref INSTANTIATE_LOCK: Mutex<()> = Mutex::new(());
|
|
|
|
}
|
|
|
|
let instantiate_guard = INSTANTIATE_LOCK.lock().unwrap();
|
|
|
|
|
|
|
|
{
|
|
|
|
let mut guard = RESOLVE_CALLBACK.lock().unwrap();
|
|
|
|
*guard = Some(callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" fn c_cb(
|
|
|
|
context: Local<Context>,
|
|
|
|
specifier: Local<String>,
|
|
|
|
referrer: Local<Module>,
|
|
|
|
) -> *mut Module {
|
|
|
|
let guard = RESOLVE_CALLBACK.lock().unwrap();
|
|
|
|
let cb = guard.unwrap();
|
2019-12-23 20:23:55 -05:00
|
|
|
cb(context, specifier, referrer)
|
2019-12-23 18:09:03 -05:00
|
|
|
}
|
|
|
|
let r =
|
|
|
|
unsafe { v8__Module__InstantiateModule(self, context, c_cb) }.into();
|
|
|
|
drop(instantiate_guard);
|
|
|
|
r
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Evaluates the module and its dependencies.
|
|
|
|
///
|
|
|
|
/// If status is kInstantiated, run the module's code. On success, set status
|
|
|
|
/// to kEvaluated and return the completion value; on failure, set status to
|
|
|
|
/// kErrored and propagate the thrown exception (which is then also available
|
|
|
|
/// via |GetException|).
|
|
|
|
#[must_use]
|
2019-12-24 18:31:36 -05:00
|
|
|
pub fn evaluate<'sc>(
|
2019-12-23 07:12:52 -05:00
|
|
|
&mut self,
|
2019-12-24 18:31:36 -05:00
|
|
|
scope: &mut impl ToLocal<'sc>,
|
2019-12-23 07:12:52 -05:00
|
|
|
mut context: Local<Context>,
|
2019-12-24 18:31:36 -05:00
|
|
|
) -> Option<Local<'sc, Value>> {
|
|
|
|
unsafe { scope.to_local(v8__Module__Evaluate(&mut *self, &mut *context)) }
|
2019-12-20 08:56:32 -05:00
|
|
|
}
|
|
|
|
}
|