2019-10-16 15:47:34 -07:00
|
|
|
use std::marker::PhantomData;
|
|
|
|
use std::mem::size_of;
|
2019-10-17 16:46:54 -07:00
|
|
|
use std::ops::{Deref, DerefMut};
|
2019-10-16 15:47:34 -07:00
|
|
|
|
2019-10-19 17:15:23 -07:00
|
|
|
pub use std::os::raw::c_int as int;
|
|
|
|
|
2019-10-16 15:47:34 -07:00
|
|
|
pub type Opaque = [usize; 0];
|
|
|
|
|
2019-10-17 16:46:54 -07:00
|
|
|
pub trait Delete
|
|
|
|
where
|
|
|
|
Self: Sized + 'static,
|
|
|
|
{
|
|
|
|
fn delete(&'static mut self) -> ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Pointer to object allocated on the C++ heap.
|
|
|
|
#[repr(transparent)]
|
2019-10-17 17:58:31 -07:00
|
|
|
#[derive(Debug)]
|
2019-10-17 16:46:54 -07:00
|
|
|
pub struct UniquePtr<T>(Option<&'static mut T>)
|
|
|
|
where
|
|
|
|
T: Delete;
|
|
|
|
|
|
|
|
impl<T> Deref for UniquePtr<T>
|
|
|
|
where
|
|
|
|
T: Delete,
|
|
|
|
{
|
|
|
|
type Target = Option<&'static mut T>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> DerefMut for UniquePtr<T>
|
|
|
|
where
|
|
|
|
T: Delete,
|
|
|
|
{
|
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
&mut self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> Drop for UniquePtr<T>
|
|
|
|
where
|
|
|
|
T: Delete,
|
|
|
|
{
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.0.take().map(Delete::delete);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Reference to object allocated on the C++ heap, similar to UniquePtr<T>,
|
|
|
|
/// but guaranteed to never contain a nullptr.
|
|
|
|
#[repr(transparent)]
|
|
|
|
pub struct UniqueRef<T>(&'static mut T)
|
|
|
|
where
|
|
|
|
T: Delete;
|
|
|
|
|
|
|
|
impl<T> Deref for UniqueRef<T>
|
|
|
|
where
|
|
|
|
T: Delete,
|
|
|
|
{
|
|
|
|
type Target = T;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> DerefMut for UniqueRef<T>
|
|
|
|
where
|
|
|
|
T: Delete,
|
|
|
|
{
|
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> Drop for UniqueRef<T>
|
|
|
|
where
|
|
|
|
T: Delete,
|
|
|
|
{
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let ptr: &'static mut T = unsafe { std::mem::transmute_copy(self.0) };
|
|
|
|
ptr.delete();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
#[repr(transparent)]
|
|
|
|
pub struct CxxVTable(pub *const Opaque);
|
|
|
|
|
2019-10-16 15:47:34 -07:00
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
pub struct RustVTable<DynT>(pub *const Opaque, pub PhantomData<DynT>);
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct FieldOffset<F>(usize, PhantomData<F>);
|
|
|
|
|
|
|
|
unsafe impl<F> Send for FieldOffset<F> where F: Send {}
|
|
|
|
unsafe impl<F> Sync for FieldOffset<F> where F: Sync {}
|
|
|
|
impl<F> Copy for FieldOffset<F> {}
|
|
|
|
|
|
|
|
impl<F> Clone for FieldOffset<F> {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
Self(self.0, self.1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<F> FieldOffset<F> {
|
|
|
|
pub fn from_ptrs<E>(embedder_ptr: *const E, field_ptr: *const F) -> Self {
|
|
|
|
let embedder_addr = embedder_ptr as usize;
|
|
|
|
let field_addr = field_ptr as usize;
|
|
|
|
assert!(field_addr >= embedder_addr);
|
|
|
|
assert!((field_addr + size_of::<F>()) <= (embedder_addr + size_of::<E>()));
|
|
|
|
Self(embedder_addr - field_addr, PhantomData)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn to_embedder<E>(self, field: &F) -> &E {
|
|
|
|
(((field as *const _ as usize) - self.0) as *const E)
|
|
|
|
.as_ref()
|
|
|
|
.unwrap()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub unsafe fn to_embedder_mut<E>(self, field: &mut F) -> &mut E {
|
|
|
|
(((field as *mut _ as usize) - self.0) as *mut E)
|
|
|
|
.as_mut()
|
|
|
|
.unwrap()
|
|
|
|
}
|
|
|
|
}
|