mirror of
https://github.com/denoland/rusty_v8.git
synced 2024-12-12 02:27:50 -05:00
update
This commit is contained in:
parent
fdfa52f4bf
commit
48db2272ca
2 changed files with 207 additions and 201 deletions
3
.rustfmt.toml
Normal file
3
.rustfmt.toml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
# Copyright 2018-2019 the Deno authors. All rights reserved. MIT license.
|
||||||
|
max_width = 80
|
||||||
|
tab_spaces = 2
|
405
src/main.rs
405
src/main.rs
|
@ -1,246 +1,249 @@
|
||||||
mod channel {
|
mod channel {
|
||||||
use super::util;
|
use super::util;
|
||||||
|
|
||||||
extern "C" {
|
extern "C" {
|
||||||
// Call a method/destructor; virtual methods use C++ dynamic dispatch.
|
// Call a method/destructor; virtual methods use C++ dynamic dispatch.
|
||||||
fn Channel__DTOR(this: &mut Channel) -> ();
|
fn Channel__DTOR(this: &mut Channel) -> ();
|
||||||
fn Channel__a(this: &mut Channel) -> ();
|
fn Channel__a(this: &mut Channel) -> ();
|
||||||
fn Channel__b(this: &Channel) -> i32;
|
fn Channel__b(this: &Channel) -> i32;
|
||||||
|
|
||||||
// Call a method of a specific class implementation, bypassing dynamic
|
// Call a method of a specific class implementation, bypassing dynamic
|
||||||
// dispatch. C++ equivalent: `my_channel.Channel::a()`.
|
// dispatch. C++ equivalent: `my_channel.Channel::a()`.
|
||||||
fn Channel__Channel__a(this: &mut Channel) -> ();
|
fn Channel__Channel__a(this: &mut Channel) -> ();
|
||||||
|
|
||||||
// Constructs a special class derived from Channel that forwards all
|
// Constructs a special class derived from Channel that forwards all
|
||||||
// virtual method invocations to rust. It is assumed that this subclass
|
// virtual method invocations to rust. It is assumed that this subclass
|
||||||
// has the same size and memory layout as the class it's deriving from.
|
// has the same size and memory layout as the class it's deriving from.
|
||||||
fn Channel__OVERRIDE__CTOR(this: &mut std::mem::MaybeUninit<Channel>) -> ();
|
fn Channel__OVERRIDE__CTOR(this: &mut std::mem::MaybeUninit<Channel>)
|
||||||
|
-> ();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[repr(C)]
|
||||||
|
pub struct Channel {
|
||||||
|
_cxx_vtable: *const [usize; 0],
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(dead_code)]
|
||||||
|
impl Channel {
|
||||||
|
pub fn a(&mut self) {
|
||||||
|
unsafe { Channel__a(self) }
|
||||||
|
}
|
||||||
|
pub fn b(&self) -> i32 {
|
||||||
|
unsafe { Channel__b(self) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for Channel {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
unsafe { Channel__DTOR(self) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ChannelDefaults;
|
||||||
|
impl ChannelDefaults {
|
||||||
|
pub fn a(this: &mut Channel) {
|
||||||
|
unsafe { Channel__Channel__a(this) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait ChannelOverrides {
|
||||||
|
fn extender(&self) -> &ChannelExtender;
|
||||||
|
fn extender_mut(&mut self) -> &mut ChannelExtender;
|
||||||
|
|
||||||
|
fn a(&mut self) {
|
||||||
|
ChannelDefaults::a(self.extender_mut())
|
||||||
|
}
|
||||||
|
fn b(&self) -> i32;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ChannelExtender {
|
||||||
|
cxx_channel: Channel,
|
||||||
|
extender_offset: usize,
|
||||||
|
rust_vtable: util::RustVTable<&'static dyn ChannelOverrides>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
unsafe extern "C" fn Channel__OVERRIDE__a__DISPATCH(this: &mut Channel) {
|
||||||
|
ChannelExtender::dispatch_mut(this).a()
|
||||||
|
}
|
||||||
|
#[no_mangle]
|
||||||
|
unsafe extern "C" fn Channel__OVERRIDE__b__DISPATCH(this: &Channel) -> i32 {
|
||||||
|
ChannelExtender::dispatch(this).b()
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ChannelExtender {
|
||||||
|
fn construct_cxx_channel() -> Channel {
|
||||||
|
unsafe {
|
||||||
|
let mut buf = std::mem::MaybeUninit::<Channel>::uninit();
|
||||||
|
Channel__OVERRIDE__CTOR(&mut buf);
|
||||||
|
buf.assume_init()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[repr(C)]
|
fn get_extender_offset<T>() -> usize
|
||||||
pub struct Channel {
|
where
|
||||||
_cxx_vtable: *const [usize; 0],
|
T: ChannelOverrides,
|
||||||
|
{
|
||||||
|
let buf = std::mem::MaybeUninit::<T>::uninit();
|
||||||
|
let embedder_ptr: *const T = buf.as_ptr();
|
||||||
|
let self_ptr: *const Self = unsafe { (*embedder_ptr).extender() };
|
||||||
|
util::FieldOffset::from_ptrs(embedder_ptr, self_ptr).offset()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(dead_code)]
|
fn get_rust_vtable<T>() -> util::RustVTable<&'static dyn ChannelOverrides>
|
||||||
impl Channel {
|
where
|
||||||
pub fn a(&mut self) {
|
T: ChannelOverrides,
|
||||||
unsafe { Channel__a(self) }
|
{
|
||||||
}
|
let buf = std::mem::MaybeUninit::<T>::uninit();
|
||||||
pub fn b(&self) -> i32 {
|
let embedder_ptr = buf.as_ptr();
|
||||||
unsafe { Channel__b(self) }
|
let trait_object: *const dyn ChannelOverrides = embedder_ptr;
|
||||||
}
|
let (data_ptr, vtable): (*const T, util::RustVTable<_>) =
|
||||||
|
unsafe { std::mem::transmute(trait_object) };
|
||||||
|
assert_eq!(data_ptr, embedder_ptr);
|
||||||
|
vtable
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct ChannelDefaults;
|
pub fn new<T>() -> Self
|
||||||
impl ChannelDefaults {
|
where
|
||||||
pub fn a(this: &mut Channel) {
|
T: ChannelOverrides,
|
||||||
unsafe { Channel__Channel__a(this) }
|
{
|
||||||
}
|
Self {
|
||||||
|
cxx_channel: Self::construct_cxx_channel(),
|
||||||
|
extender_offset: Self::get_extender_offset::<T>(),
|
||||||
|
rust_vtable: Self::get_rust_vtable::<T>(),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait ChannelOverrides {
|
fn channel_offset() -> util::FieldOffset<Self, Channel> {
|
||||||
fn base(&self) -> &Override;
|
let buf = std::mem::MaybeUninit::<Self>::uninit();
|
||||||
fn base_mut(&mut self) -> &mut Override;
|
util::FieldOffset::from_ptrs(buf.as_ptr(), unsafe {
|
||||||
|
&(*buf.as_ptr()).cxx_channel
|
||||||
fn a(&mut self) {
|
})
|
||||||
ChannelDefaults::a(self.base_mut())
|
|
||||||
}
|
|
||||||
fn b(&self) -> i32;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Override {
|
fn embedder_offset(&self) -> util::FieldOffset<util::Opaque, Self> {
|
||||||
cxx_channel: Channel,
|
util::FieldOffset::<util::Opaque, Self>::from_offset(self.extender_offset)
|
||||||
base_offset: usize,
|
|
||||||
rust_vtable: util::RustVTable<&'static dyn ChannelOverrides>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[no_mangle]
|
unsafe fn dispatch(channel: &Channel) -> &dyn ChannelOverrides {
|
||||||
unsafe extern "C" fn Channel__OVERRIDE__a__DISPATCH(this: &mut Channel) {
|
let this = Self::channel_offset().to_outer(channel);
|
||||||
Override::dispatch_mut(this).a()
|
let embedder = this.embedder_offset().to_outer(this);
|
||||||
}
|
std::mem::transmute((embedder, this.rust_vtable))
|
||||||
#[no_mangle]
|
|
||||||
unsafe extern "C" fn Channel__OVERRIDE__b__DISPATCH(this: &Channel) -> i32 {
|
|
||||||
Override::dispatch(this).b()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for Channel {
|
unsafe fn dispatch_mut(channel: &mut Channel) -> &mut dyn ChannelOverrides {
|
||||||
fn drop(&mut self) {
|
let this = Self::channel_offset().to_outer_mut(channel);
|
||||||
unsafe { Channel__DTOR(self) }
|
let vtable = this.rust_vtable;
|
||||||
}
|
let embedder = this.embedder_offset().to_outer_mut(this);
|
||||||
|
std::mem::transmute((embedder, vtable))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Override {
|
impl std::ops::Deref for ChannelExtender {
|
||||||
fn construct_cxx_channel() -> Channel {
|
type Target = Channel;
|
||||||
unsafe {
|
fn deref(&self) -> &Channel {
|
||||||
let mut buf = std::mem::MaybeUninit::<Channel>::uninit();
|
&self.cxx_channel
|
||||||
Channel__OVERRIDE__CTOR(&mut buf);
|
|
||||||
buf.assume_init()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_base_offset<T>() -> usize
|
|
||||||
where
|
|
||||||
T: ChannelOverrides,
|
|
||||||
{
|
|
||||||
let buf = std::mem::MaybeUninit::<T>::uninit();
|
|
||||||
let top_ptr: *const T = buf.as_ptr();
|
|
||||||
let self_ptr: *const Self = unsafe { (*top_ptr).base() };
|
|
||||||
util::FieldOffset::from_ptrs(top_ptr, self_ptr).offset()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_rust_vtable<T>() -> util::RustVTable<&'static dyn ChannelOverrides>
|
|
||||||
where
|
|
||||||
T: ChannelOverrides,
|
|
||||||
{
|
|
||||||
let buf = std::mem::MaybeUninit::<T>::uninit();
|
|
||||||
let embedder_ptr = buf.as_ptr();
|
|
||||||
let trait_object: *const dyn ChannelOverrides = embedder_ptr;
|
|
||||||
let (data_ptr, vtable): (*const T, util::RustVTable<_>) =
|
|
||||||
unsafe { std::mem::transmute(trait_object) };
|
|
||||||
assert_eq!(data_ptr, embedder_ptr);
|
|
||||||
vtable
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn new<T>() -> Self
|
|
||||||
where
|
|
||||||
T: ChannelOverrides,
|
|
||||||
{
|
|
||||||
Self {
|
|
||||||
cxx_channel: Self::construct_cxx_channel(),
|
|
||||||
base_offset: Self::get_base_offset::<T>(),
|
|
||||||
rust_vtable: Self::get_rust_vtable::<T>(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn channel_offset() -> util::FieldOffset<Self, Channel> {
|
|
||||||
let buf = std::mem::MaybeUninit::<Self>::uninit();
|
|
||||||
util::FieldOffset::from_ptrs(buf.as_ptr(), unsafe { &(*buf.as_ptr()).cxx_channel })
|
|
||||||
}
|
|
||||||
|
|
||||||
fn embedder_offset(&self) -> util::FieldOffset<util::Opaque, Self> {
|
|
||||||
util::FieldOffset::<util::Opaque, Self>::from_offset(self.base_offset)
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe fn dispatch(channel: &Channel) -> &dyn ChannelOverrides {
|
|
||||||
let this = Self::channel_offset().to_outer(channel);
|
|
||||||
let embedder = this.embedder_offset().to_outer(this);
|
|
||||||
std::mem::transmute((embedder, this.rust_vtable))
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe fn dispatch_mut(channel: &mut Channel) -> &mut dyn ChannelOverrides {
|
|
||||||
let this = Self::channel_offset().to_outer_mut(channel);
|
|
||||||
let vtable = this.rust_vtable;
|
|
||||||
let embedder = this.embedder_offset().to_outer_mut(this);
|
|
||||||
std::mem::transmute((embedder, vtable))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl std::ops::Deref for Override {
|
impl std::ops::DerefMut for ChannelExtender {
|
||||||
type Target = Channel;
|
fn deref_mut(&mut self) -> &mut Channel {
|
||||||
fn deref(&self) -> &Channel {
|
&mut self.cxx_channel
|
||||||
&self.cxx_channel
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl std::ops::DerefMut for Override {
|
|
||||||
fn deref_mut(&mut self) -> &mut Channel {
|
|
||||||
&mut self.cxx_channel
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mod trying {
|
mod trying {
|
||||||
use super::channel::*;
|
use super::channel::*;
|
||||||
|
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
pub struct Session {
|
pub struct Session {
|
||||||
a: i32,
|
a: i32,
|
||||||
b: String,
|
b: String,
|
||||||
c: Override,
|
c: ChannelExtender,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ChannelOverrides for Session {
|
impl ChannelOverrides for Session {
|
||||||
fn base(&self) -> &Override {
|
fn extender(&self) -> &ChannelExtender {
|
||||||
&self.c
|
&self.c
|
||||||
}
|
|
||||||
fn base_mut(&mut self) -> &mut Override {
|
|
||||||
&mut self.c
|
|
||||||
}
|
|
||||||
fn a(&mut self) {
|
|
||||||
println!("Override a!");
|
|
||||||
}
|
|
||||||
fn b(&self) -> i32 {
|
|
||||||
println!("Override b!");
|
|
||||||
42
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
fn extender_mut(&mut self) -> &mut ChannelExtender {
|
||||||
|
&mut self.c
|
||||||
|
}
|
||||||
|
fn a(&mut self) {
|
||||||
|
println!("ChannelExtender a!");
|
||||||
|
}
|
||||||
|
fn b(&self) -> i32 {
|
||||||
|
println!("ChannelExtender b!");
|
||||||
|
42
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Session {
|
impl Session {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
Self {
|
Self {
|
||||||
a: 1,
|
a: 1,
|
||||||
b: "abc".to_owned(),
|
b: "abc".to_owned(),
|
||||||
c: Override::new::<Self>(),
|
c: ChannelExtender::new::<Self>(),
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mod util {
|
mod util {
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
use std::mem::size_of;
|
use std::mem::size_of;
|
||||||
|
|
||||||
pub type Opaque = [usize; 0];
|
pub type Opaque = [usize; 0];
|
||||||
|
|
||||||
#[repr(transparent)]
|
#[repr(transparent)]
|
||||||
#[derive(Copy, Clone, Debug)]
|
#[derive(Copy, Clone, Debug)]
|
||||||
pub struct RustVTable<DynT>(pub *const Opaque, pub PhantomData<DynT>);
|
pub struct RustVTable<DynT>(pub *const Opaque, pub PhantomData<DynT>);
|
||||||
|
|
||||||
#[derive(Copy, Clone, Debug)]
|
#[derive(Copy, Clone, Debug)]
|
||||||
#[repr(transparent)]
|
#[repr(transparent)]
|
||||||
pub struct FieldOffset<O, I>(isize, PhantomData<(O, I)>);
|
pub struct FieldOffset<O, I>(isize, PhantomData<(O, I)>);
|
||||||
|
|
||||||
impl<O, I> FieldOffset<O, I> {
|
impl<O, I> FieldOffset<O, I> {
|
||||||
pub fn from_ptrs(o_ptr: *const O, i_ptr: *const I) -> Self {
|
pub fn from_ptrs(o_ptr: *const O, i_ptr: *const I) -> Self {
|
||||||
let o_addr = o_ptr as usize;
|
let o_addr = o_ptr as usize;
|
||||||
let i_addr = i_ptr as usize;
|
let i_addr = i_ptr as usize;
|
||||||
assert!(i_addr >= o_addr);
|
assert!(i_addr >= o_addr);
|
||||||
assert!((i_addr + size_of::<I>()) <= (o_addr + size_of::<O>()));
|
assert!((i_addr + size_of::<I>()) <= (o_addr + size_of::<O>()));
|
||||||
let offset = (o_addr - i_addr) as isize;
|
let offset = (o_addr - i_addr) as isize;
|
||||||
assert!(offset > 0);
|
assert!(offset > 0);
|
||||||
Self(offset, PhantomData)
|
Self(offset, PhantomData)
|
||||||
}
|
|
||||||
|
|
||||||
pub fn from_offset(offset: usize) -> Self {
|
|
||||||
assert!((offset as isize) > 0);
|
|
||||||
Self(offset as isize, PhantomData)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn offset(self) -> usize {
|
|
||||||
self.0 as usize
|
|
||||||
}
|
|
||||||
|
|
||||||
fn shift<PI, PO>(ptr: *const PI, delta: isize) -> *mut PO {
|
|
||||||
(ptr as isize + delta) as *mut PO
|
|
||||||
}
|
|
||||||
pub unsafe fn to_outer<'a>(&self, inner: &'a I) -> &'a O {
|
|
||||||
Self::shift::<I, O>(inner, -self.0).as_ref().unwrap()
|
|
||||||
}
|
|
||||||
#[allow(dead_code)]
|
|
||||||
pub unsafe fn to_outer_mut<'a>(&self, inner: &'a mut I) -> &'a mut O {
|
|
||||||
Self::shift::<I, O>(inner, -self.0).as_mut().unwrap()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<O, M, I> std::ops::Add<FieldOffset<M, I>> for FieldOffset<O, M> {
|
pub fn from_offset(offset: usize) -> Self {
|
||||||
type Output = FieldOffset<O, I>;
|
assert!((offset as isize) > 0);
|
||||||
fn add(self, that: FieldOffset<M, I>) -> Self::Output {
|
Self(offset as isize, PhantomData)
|
||||||
FieldOffset::<O, I>::from_offset(self.offset() + that.offset())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn offset(self) -> usize {
|
||||||
|
self.0 as usize
|
||||||
|
}
|
||||||
|
|
||||||
|
fn shift<PI, PO>(ptr: *const PI, delta: isize) -> *mut PO {
|
||||||
|
(ptr as isize + delta) as *mut PO
|
||||||
|
}
|
||||||
|
pub unsafe fn to_outer<'a>(&self, inner: &'a I) -> &'a O {
|
||||||
|
Self::shift::<I, O>(inner, -self.0).as_ref().unwrap()
|
||||||
|
}
|
||||||
|
#[allow(dead_code)]
|
||||||
|
pub unsafe fn to_outer_mut<'a>(&self, inner: &'a mut I) -> &'a mut O {
|
||||||
|
Self::shift::<I, O>(inner, -self.0).as_mut().unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<O, M, I> std::ops::Add<FieldOffset<M, I>> for FieldOffset<O, M> {
|
||||||
|
type Output = FieldOffset<O, I>;
|
||||||
|
fn add(self, that: FieldOffset<M, I>) -> Self::Output {
|
||||||
|
FieldOffset::<O, I>::from_offset(self.offset() + that.offset())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
trying::Session::new();
|
trying::Session::new();
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue