mirror of
https://github.com/denoland/rusty_v8.git
synced 2025-01-11 08:34:01 -05:00
Simplify 'Isolate::get/set_slot()' and use optimized hasher (#461)
This commit is contained in:
parent
8513c8fa69
commit
b31dbc89dc
3 changed files with 62 additions and 25 deletions
|
@ -3,6 +3,7 @@ use crate::isolate_create_params::raw;
|
|||
use crate::isolate_create_params::CreateParams;
|
||||
use crate::promise::PromiseRejectMessage;
|
||||
use crate::scope::data::ScopeData;
|
||||
use crate::support::BuildTypeIdHasher;
|
||||
use crate::support::Opaque;
|
||||
use crate::Context;
|
||||
use crate::Function;
|
||||
|
@ -17,7 +18,7 @@ use crate::Value;
|
|||
|
||||
use std::any::Any;
|
||||
use std::any::TypeId;
|
||||
use std::cell::{Ref, RefCell, RefMut};
|
||||
|
||||
use std::collections::HashMap;
|
||||
use std::ffi::c_void;
|
||||
use std::mem::MaybeUninit;
|
||||
|
@ -311,25 +312,18 @@ impl Isolate {
|
|||
};
|
||||
}
|
||||
|
||||
/// Get mutable reference to embedder data.
|
||||
pub fn get_slot_mut<T: 'static>(&self) -> Option<RefMut<T>> {
|
||||
let cell = self.get_annex().slots.get(&TypeId::of::<T>())?;
|
||||
let ref_mut = cell.try_borrow_mut().ok()?;
|
||||
let ref_mut = RefMut::map(ref_mut, |box_any| {
|
||||
let mut_any = &mut **box_any;
|
||||
Any::downcast_mut::<T>(mut_any).unwrap()
|
||||
});
|
||||
Some(ref_mut)
|
||||
/// Get a reference to embedder data added with `set_slot()`.
|
||||
pub fn get_slot<T: 'static>(&self) -> Option<&T> {
|
||||
let b = self.get_annex().slots.get(&TypeId::of::<T>())?;
|
||||
let r = Any::downcast_ref::<T>(&**b).unwrap();
|
||||
Some(r)
|
||||
}
|
||||
|
||||
/// Get reference to embedder data.
|
||||
pub fn get_slot<T: 'static>(&self) -> Option<Ref<T>> {
|
||||
let cell = self.get_annex().slots.get(&TypeId::of::<T>())?;
|
||||
let r = cell.try_borrow().ok()?;
|
||||
Some(Ref::map(r, |box_any| {
|
||||
let a = &**box_any;
|
||||
Any::downcast_ref::<T>(a).unwrap()
|
||||
}))
|
||||
/// Get a mutable reference to embedder data added with `set_slot()`.
|
||||
pub fn get_slot_mut<T: 'static>(&mut self) -> Option<&mut T> {
|
||||
let b = self.get_annex_mut().slots.get_mut(&TypeId::of::<T>())?;
|
||||
let r = Any::downcast_mut::<T>(&mut **b).unwrap();
|
||||
Some(r)
|
||||
}
|
||||
|
||||
/// Use with Isolate::get_slot and Isolate::get_slot_mut to associate state
|
||||
|
@ -347,7 +341,7 @@ impl Isolate {
|
|||
self
|
||||
.get_annex_mut()
|
||||
.slots
|
||||
.insert(Any::type_id(&value), RefCell::new(Box::new(value)))
|
||||
.insert(Any::type_id(&value), Box::new(value))
|
||||
.is_none()
|
||||
}
|
||||
|
||||
|
@ -548,7 +542,7 @@ impl Isolate {
|
|||
|
||||
pub(crate) struct IsolateAnnex {
|
||||
create_param_allocations: Box<dyn Any>,
|
||||
slots: HashMap<TypeId, RefCell<Box<dyn Any>>>,
|
||||
slots: HashMap<TypeId, Box<dyn Any>, BuildTypeIdHasher>,
|
||||
// The `isolate` and `isolate_mutex` fields are there so an `IsolateHandle`
|
||||
// (which may outlive the isolate itself) can determine whether the isolate
|
||||
// is still alive, and if so, get a reference to it. Safety rules:
|
||||
|
@ -567,7 +561,7 @@ impl IsolateAnnex {
|
|||
) -> Self {
|
||||
Self {
|
||||
create_param_allocations,
|
||||
slots: HashMap::new(),
|
||||
slots: HashMap::default(),
|
||||
isolate,
|
||||
isolate_mutex: Mutex::new(()),
|
||||
}
|
||||
|
|
|
@ -1,11 +1,14 @@
|
|||
use std::any::type_name;
|
||||
use std::any::Any;
|
||||
use std::any::TypeId;
|
||||
use std::borrow::Borrow;
|
||||
use std::borrow::BorrowMut;
|
||||
use std::convert::identity;
|
||||
use std::convert::AsMut;
|
||||
use std::convert::AsRef;
|
||||
use std::convert::TryFrom;
|
||||
use std::hash::BuildHasher;
|
||||
use std::hash::Hasher;
|
||||
use std::marker::PhantomData;
|
||||
use std::mem::align_of;
|
||||
use std::mem::forget;
|
||||
|
@ -506,6 +509,46 @@ impl<F> FieldOffset<F> {
|
|||
}
|
||||
}
|
||||
|
||||
/// A special hasher that is optimized for hashing `std::any::TypeId` values.
|
||||
/// It can't be used for anything else.
|
||||
#[derive(Clone, Default)]
|
||||
pub(crate) struct TypeIdHasher {
|
||||
state: Option<u64>,
|
||||
}
|
||||
|
||||
impl Hasher for TypeIdHasher {
|
||||
fn write(&mut self, _bytes: &[u8]) {
|
||||
panic!("TypeIdHasher::write() called unexpectedly");
|
||||
}
|
||||
|
||||
fn write_u64(&mut self, value: u64) {
|
||||
// `TypeId` values are actually 64-bit values which themselves come out of
|
||||
// some hash function, so it's unnecessary to shuffle their bits further.
|
||||
assert_eq!(size_of::<TypeId>(), size_of::<u64>());
|
||||
assert_eq!(align_of::<TypeId>(), size_of::<u64>());
|
||||
let prev_state = self.state.replace(value);
|
||||
assert_eq!(prev_state, None);
|
||||
}
|
||||
|
||||
fn finish(&self) -> u64 {
|
||||
self.state.unwrap()
|
||||
}
|
||||
}
|
||||
|
||||
/// Factory for instances of `TypeIdHasher`. This is the type that one would
|
||||
/// pass to the constructor of some map/set type in order to make it use
|
||||
/// `TypeIdHasher` instead of the default hasher implementation.
|
||||
#[derive(Copy, Clone, Default)]
|
||||
pub(crate) struct BuildTypeIdHasher;
|
||||
|
||||
impl BuildHasher for BuildTypeIdHasher {
|
||||
type Hasher = TypeIdHasher;
|
||||
|
||||
fn build_hasher(&self) -> Self::Hasher {
|
||||
Default::default()
|
||||
}
|
||||
}
|
||||
|
||||
#[repr(C)]
|
||||
#[derive(Default)]
|
||||
pub struct Maybe<T> {
|
||||
|
|
|
@ -53,8 +53,8 @@ impl CoreIsolate {
|
|||
s.i
|
||||
}
|
||||
|
||||
fn set_i(&self, i: usize) {
|
||||
let mut s = self.0.get_slot_mut::<CoreIsolateState>().unwrap();
|
||||
fn set_i(&mut self, i: usize) {
|
||||
let s = self.0.get_slot_mut::<CoreIsolateState>().unwrap();
|
||||
s.i = i;
|
||||
}
|
||||
}
|
||||
|
@ -102,8 +102,8 @@ impl EsIsolate {
|
|||
state.x
|
||||
}
|
||||
|
||||
fn set_x(&self, x: bool) {
|
||||
let mut state = self.0.get_slot_mut::<EsIsolateState>().unwrap();
|
||||
fn set_x(&mut self, x: bool) {
|
||||
let state = self.0.get_slot_mut::<EsIsolateState>().unwrap();
|
||||
state.x = x;
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue