mirror of
https://github.com/denoland/rusty_v8.git
synced 2025-01-13 01:22:42 -05:00
Add the 'HandleScope::with_context()' scope constructor (#410)
This commit is contained in:
parent
c13625148f
commit
acf6bfe0e4
2 changed files with 231 additions and 92 deletions
133
src/scope.rs
133
src/scope.rs
|
@ -83,6 +83,7 @@ use crate::function::FunctionCallbackInfo;
|
||||||
use crate::function::PropertyCallbackInfo;
|
use crate::function::PropertyCallbackInfo;
|
||||||
use crate::Context;
|
use crate::Context;
|
||||||
use crate::Data;
|
use crate::Data;
|
||||||
|
use crate::Handle;
|
||||||
use crate::Isolate;
|
use crate::Isolate;
|
||||||
use crate::Local;
|
use crate::Local;
|
||||||
use crate::Message;
|
use crate::Message;
|
||||||
|
@ -143,6 +144,24 @@ impl<'s> HandleScope<'s> {
|
||||||
.as_scope()
|
.as_scope()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Opens a new `HandleScope` and enters a `Context` in one step.
|
||||||
|
/// The first argument should be an `Isolate` or `OwnedIsolate`.
|
||||||
|
/// The second argument can be any handle that refers to a `Context` object;
|
||||||
|
/// usually this will be a `Global<Context>`.
|
||||||
|
pub fn with_context<
|
||||||
|
P: param::NewHandleScopeWithContext<'s>,
|
||||||
|
H: Handle<Data = Context>,
|
||||||
|
>(
|
||||||
|
param: &'s mut P,
|
||||||
|
context: H,
|
||||||
|
) -> Self {
|
||||||
|
let context_ref = context.get(param.get_isolate_mut());
|
||||||
|
param
|
||||||
|
.get_scope_data_mut()
|
||||||
|
.new_handle_scope_data_with_context(context_ref)
|
||||||
|
.as_scope()
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns the context of the currently running JavaScript, or the context
|
/// Returns the context of the currently running JavaScript, or the context
|
||||||
/// on the top of the stack if no JavaScript is running.
|
/// on the top of the stack if no JavaScript is running.
|
||||||
pub fn get_current_context(&self) -> Local<'s, Context> {
|
pub fn get_current_context(&self) -> Local<'s, Context> {
|
||||||
|
@ -645,6 +664,22 @@ mod param {
|
||||||
type NewScope = HandleScope<'s>;
|
type NewScope = HandleScope<'s>;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait NewHandleScopeWithContext<'s>: data::GetScopeData {
|
||||||
|
fn get_isolate_mut(&mut self) -> &mut Isolate;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'s> NewHandleScopeWithContext<'s> for Isolate {
|
||||||
|
fn get_isolate_mut(&mut self) -> &mut Isolate {
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'s> NewHandleScopeWithContext<'s> for OwnedIsolate {
|
||||||
|
fn get_isolate_mut(&mut self) -> &mut Isolate {
|
||||||
|
&mut *self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub trait NewEscapableHandleScope<'s, 'e: 's>: data::GetScopeData {
|
pub trait NewEscapableHandleScope<'s, 'e: 's>: data::GetScopeData {
|
||||||
type NewScope: Scope;
|
type NewScope: Scope;
|
||||||
}
|
}
|
||||||
|
@ -846,23 +881,74 @@ pub(crate) mod data {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(super) fn new_handle_scope_data(&mut self) -> &mut Self {
|
/// Implementation helper function, which creates the raw `HandleScope`, but
|
||||||
|
/// defers (maybe) entering a context to the provided callback argument.
|
||||||
|
/// This function gets called by `Self::new_handle_scope_data()` and
|
||||||
|
/// `Self::new_handle_scope_data_with_context()`.
|
||||||
|
#[inline(always)]
|
||||||
|
fn new_handle_scope_data_with<F>(&mut self, init_context_fn: F) -> &mut Self
|
||||||
|
where
|
||||||
|
F: FnOnce(
|
||||||
|
NonNull<Isolate>,
|
||||||
|
&mut Cell<Option<NonNull<Context>>>,
|
||||||
|
&mut Option<raw::ContextScope>,
|
||||||
|
),
|
||||||
|
{
|
||||||
self.new_scope_data_with(|data| {
|
self.new_scope_data_with(|data| {
|
||||||
let isolate = data.isolate;
|
let isolate = data.isolate;
|
||||||
data.scope_type_specific_data.init_with(|| {
|
data.scope_type_specific_data.init_with(|| {
|
||||||
ScopeTypeSpecificData::HandleScope {
|
ScopeTypeSpecificData::HandleScope {
|
||||||
raw_handle_scope: unsafe { raw::HandleScope::uninit() },
|
raw_handle_scope: unsafe { raw::HandleScope::uninit() },
|
||||||
|
raw_context_scope: None,
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
match &mut data.scope_type_specific_data {
|
match &mut data.scope_type_specific_data {
|
||||||
ScopeTypeSpecificData::HandleScope { raw_handle_scope } => {
|
ScopeTypeSpecificData::HandleScope {
|
||||||
|
raw_handle_scope,
|
||||||
|
raw_context_scope,
|
||||||
|
} => {
|
||||||
unsafe { raw_handle_scope.init(isolate) };
|
unsafe { raw_handle_scope.init(isolate) };
|
||||||
|
init_context_fn(isolate, &mut data.context, raw_context_scope);
|
||||||
}
|
}
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
}
|
};
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(super) fn new_handle_scope_data(&mut self) -> &mut Self {
|
||||||
|
self.new_handle_scope_data_with(|_, _, raw_context_scope| {
|
||||||
|
debug_assert!(raw_context_scope.is_none())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(super) fn new_handle_scope_data_with_context(
|
||||||
|
&mut self,
|
||||||
|
context_ref: &Context,
|
||||||
|
) -> &mut Self {
|
||||||
|
self.new_handle_scope_data_with(
|
||||||
|
move |isolate, context_data, raw_context_scope| unsafe {
|
||||||
|
let context_nn = NonNull::from(context_ref);
|
||||||
|
// Copy the `Context` reference to a new local handle to enure that it
|
||||||
|
// cannot get garbage collected until after this scope is dropped.
|
||||||
|
let local_context_ptr =
|
||||||
|
raw::v8__Local__New(isolate.as_ptr(), context_nn.cast().as_ptr())
|
||||||
|
as *const Context;
|
||||||
|
let local_context_nn =
|
||||||
|
NonNull::new_unchecked(local_context_ptr as *mut _);
|
||||||
|
let local_context = Local::from_non_null(local_context_nn);
|
||||||
|
// Initialize the `raw::ContextScope`. This enters the context too.
|
||||||
|
debug_assert!(raw_context_scope.is_none());
|
||||||
|
ptr::write(
|
||||||
|
raw_context_scope,
|
||||||
|
Some(raw::ContextScope::new(local_context)),
|
||||||
|
);
|
||||||
|
// Also store the newly created `Local<Context>` in the `Cell` that
|
||||||
|
// serves as a look-up cache for the current context.
|
||||||
|
context_data.set(Some(local_context_nn));
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
pub(super) fn new_escapable_handle_scope_data(&mut self) -> &mut Self {
|
pub(super) fn new_escapable_handle_scope_data(&mut self) -> &mut Self {
|
||||||
self.new_scope_data_with(|data| {
|
self.new_scope_data_with(|data| {
|
||||||
// Note: the `raw_escape_slot` field must be initialized _before_ the
|
// Note: the `raw_escape_slot` field must be initialized _before_ the
|
||||||
|
@ -1203,6 +1289,7 @@ pub(crate) mod data {
|
||||||
},
|
},
|
||||||
HandleScope {
|
HandleScope {
|
||||||
raw_handle_scope: raw::HandleScope,
|
raw_handle_scope: raw::HandleScope,
|
||||||
|
raw_context_scope: Option<raw::ContextScope>,
|
||||||
},
|
},
|
||||||
EscapableHandleScope {
|
EscapableHandleScope {
|
||||||
raw_handle_scope: raw::HandleScope,
|
raw_handle_scope: raw::HandleScope,
|
||||||
|
@ -1219,6 +1306,22 @@ pub(crate) mod data {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl Drop for ScopeTypeSpecificData {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
// For `HandleScope`s that also enter a `Context`, drop order matters. The
|
||||||
|
// context is stored in a `Local` handle, which is allocated in this
|
||||||
|
// scope's own private `raw::HandleScope`. When that `raw::HandleScope`
|
||||||
|
// is dropped first, we immediately lose the `Local<Context>` handle,
|
||||||
|
// which we need in order to exit `ContextScope`.
|
||||||
|
if let Self::HandleScope {
|
||||||
|
raw_context_scope, ..
|
||||||
|
} = self
|
||||||
|
{
|
||||||
|
*raw_context_scope = None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl ScopeTypeSpecificData {
|
impl ScopeTypeSpecificData {
|
||||||
pub fn is_none(&self) -> bool {
|
pub fn is_none(&self) -> bool {
|
||||||
match self {
|
match self {
|
||||||
|
@ -1272,22 +1375,21 @@ mod raw {
|
||||||
pub(super) struct Address(NonZeroUsize);
|
pub(super) struct Address(NonZeroUsize);
|
||||||
|
|
||||||
pub(super) struct ContextScope {
|
pub(super) struct ContextScope {
|
||||||
entered_context: *const Context,
|
entered_context: NonNull<Context>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ContextScope {
|
impl ContextScope {
|
||||||
pub fn new(context: Local<Context>) -> Self {
|
pub fn new(context: Local<Context>) -> Self {
|
||||||
unsafe { v8__Context__Enter(&*context) };
|
unsafe { v8__Context__Enter(&*context) };
|
||||||
Self {
|
Self {
|
||||||
entered_context: &*context,
|
entered_context: context.as_non_null(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for ContextScope {
|
impl Drop for ContextScope {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
debug_assert!(!self.entered_context.is_null());
|
unsafe { v8__Context__Exit(self.entered_context.as_ptr()) };
|
||||||
unsafe { v8__Context__Exit(self.entered_context) };
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1452,6 +1554,7 @@ mod raw {
|
||||||
mod tests {
|
mod tests {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::new_default_platform;
|
use crate::new_default_platform;
|
||||||
|
use crate::Global;
|
||||||
use crate::V8;
|
use crate::V8;
|
||||||
use std::any::type_name;
|
use std::any::type_name;
|
||||||
use std::sync::Once;
|
use std::sync::Once;
|
||||||
|
@ -1581,9 +1684,12 @@ mod tests {
|
||||||
initialize_v8();
|
initialize_v8();
|
||||||
let isolate = &mut Isolate::new(Default::default());
|
let isolate = &mut Isolate::new(Default::default());
|
||||||
AssertTypeOf(isolate).is::<OwnedIsolate>();
|
AssertTypeOf(isolate).is::<OwnedIsolate>();
|
||||||
|
let global_context: Global<Context>;
|
||||||
|
{
|
||||||
let l1_hs = &mut HandleScope::new(isolate);
|
let l1_hs = &mut HandleScope::new(isolate);
|
||||||
AssertTypeOf(l1_hs).is::<HandleScope<()>>();
|
AssertTypeOf(l1_hs).is::<HandleScope<()>>();
|
||||||
let context = Context::new(l1_hs);
|
let context = Context::new(l1_hs);
|
||||||
|
global_context = Global::new(l1_hs, context);
|
||||||
AssertTypeOf(&HandleScope::new(l1_hs)).is::<HandleScope<()>>();
|
AssertTypeOf(&HandleScope::new(l1_hs)).is::<HandleScope<()>>();
|
||||||
{
|
{
|
||||||
let l2_cxs = &mut ContextScope::new(l1_hs, context);
|
let l2_cxs = &mut ContextScope::new(l1_hs, context);
|
||||||
|
@ -1598,7 +1704,8 @@ mod tests {
|
||||||
{
|
{
|
||||||
let l2_ehs = &mut EscapableHandleScope::new(l1_hs);
|
let l2_ehs = &mut EscapableHandleScope::new(l1_hs);
|
||||||
AssertTypeOf(l2_ehs).is::<EscapableHandleScope<()>>();
|
AssertTypeOf(l2_ehs).is::<EscapableHandleScope<()>>();
|
||||||
AssertTypeOf(&HandleScope::new(l2_ehs)).is::<EscapableHandleScope<()>>();
|
AssertTypeOf(&HandleScope::new(l2_ehs))
|
||||||
|
.is::<EscapableHandleScope<()>>();
|
||||||
AssertTypeOf(&EscapableHandleScope::new(l2_ehs))
|
AssertTypeOf(&EscapableHandleScope::new(l2_ehs))
|
||||||
.is::<EscapableHandleScope<()>>();
|
.is::<EscapableHandleScope<()>>();
|
||||||
{
|
{
|
||||||
|
@ -1626,7 +1733,8 @@ mod tests {
|
||||||
AssertTypeOf(l3_tc).is::<TryCatch<EscapableHandleScope<()>>>();
|
AssertTypeOf(l3_tc).is::<TryCatch<EscapableHandleScope<()>>>();
|
||||||
AssertTypeOf(&ContextScope::new(l3_tc, context))
|
AssertTypeOf(&ContextScope::new(l3_tc, context))
|
||||||
.is::<ContextScope<EscapableHandleScope>>();
|
.is::<ContextScope<EscapableHandleScope>>();
|
||||||
AssertTypeOf(&HandleScope::new(l3_tc)).is::<EscapableHandleScope<()>>();
|
AssertTypeOf(&HandleScope::new(l3_tc))
|
||||||
|
.is::<EscapableHandleScope<()>>();
|
||||||
AssertTypeOf(&EscapableHandleScope::new(l3_tc))
|
AssertTypeOf(&EscapableHandleScope::new(l3_tc))
|
||||||
.is::<EscapableHandleScope<()>>();
|
.is::<EscapableHandleScope<()>>();
|
||||||
AssertTypeOf(&TryCatch::new(l3_tc))
|
AssertTypeOf(&TryCatch::new(l3_tc))
|
||||||
|
@ -1681,4 +1789,11 @@ mod tests {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
{
|
||||||
|
AssertTypeOf(&HandleScope::with_context(isolate, &global_context))
|
||||||
|
.is::<HandleScope>();
|
||||||
|
AssertTypeOf(&HandleScope::with_context(isolate, global_context))
|
||||||
|
.is::<HandleScope>();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -265,6 +265,30 @@ fn context_scope_param_and_context_must_share_isolate() {
|
||||||
let _context_scope_21 = &mut v8::ContextScope::new(scope2, context1);
|
let _context_scope_21 = &mut v8::ContextScope::new(scope2, context1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic(
|
||||||
|
expected = "attempt to use Handle in an Isolate that is not its host"
|
||||||
|
)]
|
||||||
|
fn handle_scope_param_and_context_must_share_isolate() {
|
||||||
|
let _setup_guard = setup();
|
||||||
|
let isolate1 = &mut v8::Isolate::new(Default::default());
|
||||||
|
let isolate2 = &mut v8::Isolate::new(Default::default());
|
||||||
|
let global_context1;
|
||||||
|
let global_context2;
|
||||||
|
{
|
||||||
|
let scope1 = &mut v8::HandleScope::new(isolate1);
|
||||||
|
let scope2 = &mut v8::HandleScope::new(isolate2);
|
||||||
|
let local_context_1 = v8::Context::new(scope1);
|
||||||
|
let local_context_2 = v8::Context::new(scope2);
|
||||||
|
global_context1 = v8::Global::new(scope1, local_context_1);
|
||||||
|
global_context2 = v8::Global::new(scope2, local_context_2);
|
||||||
|
}
|
||||||
|
let _handle_scope_12 =
|
||||||
|
&mut v8::HandleScope::with_context(isolate1, global_context2);
|
||||||
|
let _handle_scope_21 =
|
||||||
|
&mut v8::HandleScope::with_context(isolate2, global_context1);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn microtasks() {
|
fn microtasks() {
|
||||||
let _setup_guard = setup();
|
let _setup_guard = setup();
|
||||||
|
|
Loading…
Reference in a new issue