0
0
Fork 0
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:
Bert Belder 2020-09-07 19:49:49 +02:00
parent 8513c8fa69
commit b31dbc89dc
No known key found for this signature in database
GPG key ID: 7A77887B2E2ED461
3 changed files with 62 additions and 25 deletions

View file

@ -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(()),
}

View file

@ -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> {

View file

@ -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;
}
}