2019-12-27 09:12:16 -05:00
|
|
|
use crate::external_references::ExternalReferences;
|
2020-06-03 07:38:34 +02:00
|
|
|
use crate::scope::data::ScopeData;
|
2020-04-20 21:18:03 +02:00
|
|
|
use crate::support::char;
|
2019-12-24 14:03:32 +01:00
|
|
|
use crate::support::int;
|
2019-12-27 09:12:16 -05:00
|
|
|
use crate::support::intptr_t;
|
2019-12-24 14:03:32 +01:00
|
|
|
use crate::Context;
|
2020-09-09 20:39:49 -04:00
|
|
|
use crate::Data;
|
2019-12-24 14:03:32 +01:00
|
|
|
use crate::Isolate;
|
|
|
|
use crate::Local;
|
2020-01-03 16:52:05 -05:00
|
|
|
use crate::OwnedIsolate;
|
2020-04-20 21:18:03 +02:00
|
|
|
|
2020-01-03 16:52:05 -05:00
|
|
|
use std::borrow::Borrow;
|
2020-04-20 21:18:03 +02:00
|
|
|
use std::convert::TryFrom;
|
2020-01-03 16:52:05 -05:00
|
|
|
use std::mem::MaybeUninit;
|
|
|
|
use std::ops::Deref;
|
2019-12-24 14:03:32 +01:00
|
|
|
|
|
|
|
extern "C" {
|
2019-12-27 09:12:16 -05:00
|
|
|
fn v8__SnapshotCreator__CONSTRUCT(
|
2020-04-13 14:43:56 +02:00
|
|
|
buf: *mut MaybeUninit<SnapshotCreator>,
|
2019-12-27 09:12:16 -05:00
|
|
|
external_references: *const intptr_t,
|
|
|
|
);
|
2020-04-13 14:43:56 +02:00
|
|
|
fn v8__SnapshotCreator__DESTRUCT(this: *mut SnapshotCreator);
|
|
|
|
fn v8__SnapshotCreator__GetIsolate(
|
2020-08-06 13:37:18 +02:00
|
|
|
this: *const SnapshotCreator,
|
2020-04-13 14:43:56 +02:00
|
|
|
) -> *mut Isolate;
|
2019-12-24 14:03:32 +01:00
|
|
|
fn v8__SnapshotCreator__CreateBlob(
|
|
|
|
this: *mut SnapshotCreator,
|
|
|
|
function_code_handling: FunctionCodeHandling,
|
2020-04-20 21:18:03 +02:00
|
|
|
) -> StartupData;
|
2019-12-24 14:03:32 +01:00
|
|
|
fn v8__SnapshotCreator__SetDefaultContext(
|
2020-04-13 14:43:56 +02:00
|
|
|
this: *mut SnapshotCreator,
|
|
|
|
context: *const Context,
|
2019-12-24 14:03:32 +01:00
|
|
|
);
|
2020-09-09 20:39:49 -04:00
|
|
|
fn v8__SnapshotCreator__AddData_to_isolate(
|
|
|
|
this: *mut SnapshotCreator,
|
|
|
|
data: *const Data,
|
|
|
|
) -> usize;
|
|
|
|
fn v8__SnapshotCreator__AddData_to_context(
|
|
|
|
this: *mut SnapshotCreator,
|
|
|
|
context: *const Context,
|
|
|
|
data: *const Data,
|
|
|
|
) -> usize;
|
2020-04-13 14:43:56 +02:00
|
|
|
fn v8__StartupData__DESTRUCT(this: *mut StartupData);
|
2019-12-24 14:03:32 +01:00
|
|
|
}
|
|
|
|
|
2020-08-06 13:37:18 +02:00
|
|
|
// TODO(piscisaureus): merge this struct with
|
|
|
|
// `isolate_create_params::raw::StartupData`.
|
2019-12-24 14:03:32 +01:00
|
|
|
#[repr(C)]
|
2020-11-18 15:17:25 +01:00
|
|
|
#[derive(Debug)]
|
2020-04-20 21:18:03 +02:00
|
|
|
pub struct StartupData {
|
|
|
|
data: *const char,
|
2019-12-25 08:14:55 -05:00
|
|
|
raw_size: int,
|
2019-12-31 02:42:39 +01:00
|
|
|
}
|
|
|
|
|
2020-04-20 21:18:03 +02:00
|
|
|
impl Deref for StartupData {
|
2019-12-25 08:14:55 -05:00
|
|
|
type Target = [u8];
|
|
|
|
fn deref(&self) -> &Self::Target {
|
2020-04-20 21:18:03 +02:00
|
|
|
let data = self.data as *const u8;
|
|
|
|
let len = usize::try_from(self.raw_size).unwrap();
|
|
|
|
unsafe { std::slice::from_raw_parts(data, len) }
|
2019-12-25 08:14:55 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-20 21:18:03 +02:00
|
|
|
impl AsRef<[u8]> for StartupData {
|
|
|
|
fn as_ref(&self) -> &[u8] {
|
|
|
|
&**self
|
2019-12-31 02:42:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-20 21:18:03 +02:00
|
|
|
impl Borrow<[u8]> for StartupData {
|
|
|
|
fn borrow(&self) -> &[u8] {
|
|
|
|
&**self
|
2019-12-25 08:14:55 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-20 21:18:03 +02:00
|
|
|
impl Drop for StartupData {
|
2019-12-25 08:14:55 -05:00
|
|
|
fn drop(&mut self) {
|
2020-04-20 21:18:03 +02:00
|
|
|
unsafe { v8__StartupData__DESTRUCT(self) }
|
2019-12-25 08:14:55 -05:00
|
|
|
}
|
2019-12-24 14:03:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)]
|
2020-11-18 15:17:25 +01:00
|
|
|
#[derive(Debug)]
|
2019-12-24 14:03:32 +01:00
|
|
|
pub enum FunctionCodeHandling {
|
|
|
|
Clear,
|
|
|
|
Keep,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Helper class to create a snapshot data blob.
|
|
|
|
#[repr(C)]
|
2020-11-18 15:17:25 +01:00
|
|
|
#[derive(Debug)]
|
2019-12-24 14:03:32 +01:00
|
|
|
pub struct SnapshotCreator([usize; 1]);
|
|
|
|
|
2019-12-27 09:12:16 -05:00
|
|
|
impl SnapshotCreator {
|
2019-12-24 14:03:32 +01:00
|
|
|
/// Create and enter an isolate, and set it up for serialization.
|
|
|
|
/// The isolate is created from scratch.
|
2019-12-27 09:12:16 -05:00
|
|
|
pub fn new(external_references: Option<&'static ExternalReferences>) -> Self {
|
2019-12-24 14:03:32 +01:00
|
|
|
let mut snapshot_creator: MaybeUninit<Self> = MaybeUninit::uninit();
|
2019-12-27 09:12:16 -05:00
|
|
|
let external_references_ptr = if let Some(er) = external_references {
|
|
|
|
er.as_ptr()
|
|
|
|
} else {
|
|
|
|
std::ptr::null()
|
|
|
|
};
|
2019-12-24 14:03:32 +01:00
|
|
|
unsafe {
|
2019-12-27 09:12:16 -05:00
|
|
|
v8__SnapshotCreator__CONSTRUCT(
|
|
|
|
&mut snapshot_creator,
|
|
|
|
external_references_ptr,
|
|
|
|
);
|
2019-12-24 14:03:32 +01:00
|
|
|
snapshot_creator.assume_init()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for SnapshotCreator {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe { v8__SnapshotCreator__DESTRUCT(self) };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SnapshotCreator {
|
|
|
|
/// Set the default context to be included in the snapshot blob.
|
|
|
|
/// The snapshot will not contain the global proxy, and we expect one or a
|
|
|
|
/// global object template to create one, to be provided upon deserialization.
|
2021-02-07 23:49:07 +01:00
|
|
|
pub fn set_default_context(&mut self, context: Local<Context>) {
|
2020-04-13 14:43:56 +02:00
|
|
|
unsafe { v8__SnapshotCreator__SetDefaultContext(self, &*context) };
|
2019-12-24 14:03:32 +01:00
|
|
|
}
|
|
|
|
|
2020-09-09 20:39:49 -04:00
|
|
|
/// Attach arbitrary `v8::Data` to the isolate snapshot, which can be
|
|
|
|
/// retrieved via `HandleScope::get_context_data_from_snapshot_once()` after
|
|
|
|
/// deserialization. This data does not survive when a new snapshot is created
|
|
|
|
/// from an existing snapshot.
|
|
|
|
pub fn add_isolate_data<T>(&mut self, data: Local<T>) -> usize
|
|
|
|
where
|
|
|
|
for<'l> Local<'l, T>: Into<Local<'l, Data>>,
|
|
|
|
{
|
|
|
|
unsafe { v8__SnapshotCreator__AddData_to_isolate(self, &*data.into()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Attach arbitrary `v8::Data` to the context snapshot, which can be
|
|
|
|
/// retrieved via `HandleScope::get_context_data_from_snapshot_once()` after
|
|
|
|
/// deserialization. This data does not survive when a new snapshot is
|
|
|
|
/// created from an existing snapshot.
|
|
|
|
pub fn add_context_data<T>(
|
|
|
|
&mut self,
|
|
|
|
context: Local<Context>,
|
|
|
|
data: Local<T>,
|
|
|
|
) -> usize
|
|
|
|
where
|
|
|
|
for<'l> Local<'l, T>: Into<Local<'l, Data>>,
|
|
|
|
{
|
|
|
|
unsafe {
|
|
|
|
v8__SnapshotCreator__AddData_to_context(self, &*context, &*data.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-24 14:03:32 +01:00
|
|
|
/// Creates a snapshot data blob.
|
|
|
|
/// This must not be called from within a handle scope.
|
|
|
|
pub fn create_blob(
|
|
|
|
&mut self,
|
|
|
|
function_code_handling: FunctionCodeHandling,
|
2020-04-20 21:18:03 +02:00
|
|
|
) -> Option<StartupData> {
|
2020-06-03 07:38:34 +02:00
|
|
|
{
|
|
|
|
let isolate = unsafe { &mut *v8__SnapshotCreator__GetIsolate(self) };
|
|
|
|
ScopeData::get_root_mut(isolate);
|
|
|
|
}
|
2019-12-25 08:14:55 -05:00
|
|
|
let blob =
|
|
|
|
unsafe { v8__SnapshotCreator__CreateBlob(self, function_code_handling) };
|
|
|
|
if blob.data.is_null() {
|
|
|
|
debug_assert!(blob.raw_size == 0);
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
debug_assert!(blob.raw_size > 0);
|
|
|
|
Some(blob)
|
|
|
|
}
|
2019-12-24 14:03:32 +01:00
|
|
|
}
|
|
|
|
|
2020-06-17 05:05:01 +02:00
|
|
|
/// This is marked unsafe because it should be called at most once per
|
|
|
|
/// snapshot creator.
|
2020-01-03 16:52:05 -05:00
|
|
|
// TODO Because the SnapshotCreator creates its own isolate, we need a way to
|
|
|
|
// get an owned handle to it. This is a questionable design which ought to be
|
|
|
|
// revisited after the libdeno integration is complete.
|
|
|
|
pub unsafe fn get_owned_isolate(&mut self) -> OwnedIsolate {
|
|
|
|
let isolate_ptr = v8__SnapshotCreator__GetIsolate(self);
|
2020-05-06 19:26:23 +02:00
|
|
|
let mut owned_isolate = OwnedIsolate::new(isolate_ptr);
|
2020-06-03 07:38:34 +02:00
|
|
|
ScopeData::new_root(&mut owned_isolate);
|
2020-05-06 19:26:23 +02:00
|
|
|
owned_isolate.create_annex(Box::new(()));
|
|
|
|
owned_isolate
|
2020-01-03 16:52:05 -05:00
|
|
|
}
|
2019-12-24 14:03:32 +01:00
|
|
|
}
|