use std::{mem::ManuallyDrop, ptr}; use mlua::prelude::*; use crate::{ data::association_names::REF_INNER, ffi::{association, bit_mask::*, FfiData}, }; mod bounds; mod flag; pub use self::{ bounds::{RefBounds, UNSIZED_BOUNDS}, flag::RefFlag, }; // Box:ref():ref() should not be able to modify, Only for external const BOX_REF_REF_FLAGS: u8 = 0; const UNINIT_REF_FLAGS: u8 = RefFlag::Uninit.value(); // | FfiRefFlag::Writable.value() // | FfiRefFlag::Readable.value() // | FfiRefFlag::Dereferenceable.value() // | FfiRefFlag::Offsetable.value() // | FfiRefFlag::Function.value(); // A referenced space. It is possible to read and write through types. // This operation is not safe. This may cause a memory error in Lua // if use it incorrectly. // If it references an area managed by Lua, // the box will remain as long as this reference is alive. pub struct RefData { ptr: ManuallyDrop>, pub flags: u8, pub boundary: RefBounds, } impl RefData { pub fn new(ptr: *mut (), flags: u8, boundary: RefBounds) -> Self { Self { ptr: ManuallyDrop::new(Box::new(ptr)), flags, boundary, } } pub fn new_uninit() -> Self { Self { ptr: ManuallyDrop::new(Box::new(ptr::null_mut())), flags: UNINIT_REF_FLAGS, boundary: UNSIZED_BOUNDS, } } // Make FfiRef from ref pub fn luaref<'lua>( lua: &'lua Lua, this: LuaAnyUserData<'lua>, ) -> LuaResult> { let target = this.borrow::()?; let luaref = lua.create_userdata(RefData::new( ptr::from_ref(&target.ptr) as *mut (), BOX_REF_REF_FLAGS, RefBounds { below: 0, above: size_of::(), }, ))?; // If the ref holds a box, make sure the new ref also holds the box by holding ref association::set(lua, REF_INNER, &luaref, &this)?; Ok(luaref) } pub unsafe fn deref(&self) -> LuaResult { u8_test(self.flags, RefFlag::Dereferenceable.value()) .then_some(()) .ok_or_else(|| LuaError::external("This pointer is not dereferenceable."))?; self.boundary .check_sized(0, size_of::()) .then_some(()) .ok_or_else(|| { LuaError::external( "Offset is out of bounds. Dereferencing pointer requires size of usize", ) })?; // FIXME flags Ok(Self::new( *self.ptr.cast::<*mut ()>(), self.flags, UNSIZED_BOUNDS, )) } pub fn is_nullptr(&self) -> bool { // * ManuallyDrop wrapper // * Box wrapper (**self.ptr) as usize == 0 } pub unsafe fn offset(&self, offset: isize) -> LuaResult { u8_test(self.flags, RefFlag::Offsetable.value()) .then_some(()) .ok_or_else(|| LuaError::external("This pointer is not offsetable."))?; // Check boundary, if exceed, return error self.boundary .check_boundary(offset) .then_some(()) .ok_or_else(|| { LuaError::external(format!( "Offset is out of bounds. high: {}, low: {}. offset got {}", self.boundary.above, self.boundary.below, offset )) })?; let boundary = self.boundary.offset(offset); // TODO Ok(Self::new( self.ptr.byte_offset(offset), self.flags, boundary, )) } } impl Drop for RefData { fn drop(&mut self) { if u8_test_not(self.flags, RefFlag::Leaked.value()) { unsafe { ManuallyDrop::drop(&mut self.ptr) }; } } } impl FfiData for RefData { fn check_boundary(&self, offset: isize, size: usize) -> bool { self.boundary.check_sized(offset, size) } unsafe fn get_pointer(&self) -> *mut () { **self.ptr } fn is_readable(&self) -> bool { u8_test(self.flags, RefFlag::Readable.value()) } fn is_writable(&self) -> bool { u8_test(self.flags, RefFlag::Writable.value()) } } impl LuaUserData for RefData { fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) { // FIXME: methods.add_function("deref", |lua, this: LuaAnyUserData| { let inner = association::get(lua, REF_INNER, &this)?; let ffiref = this.borrow::()?; let result = lua.create_userdata(unsafe { ffiref.deref()? })?; if let Some(t) = inner { // if let Some(u) = association::get(lua, regname, value) {} association::set(lua, REF_INNER, &result, &t)?; } Ok(result) }); methods.add_function("offset", |lua, (this, offset): (LuaAnyUserData, isize)| { let ffiref = unsafe { this.borrow::()?.offset(offset)? }; let userdata = lua.create_userdata(ffiref)?; // If the ref holds a box, make sure the new ref also holds the box if let Some(t) = association::get(lua, REF_INNER, &this)? { association::set(lua, REF_INNER, &userdata, t)?; } Ok(userdata) }); // FIXME: methods.add_function("ref", |lua, this: LuaAnyUserData| { let ffiref = RefData::luaref(lua, this)?; Ok(ffiref) }); methods.add_method("isNull", |_, this, ()| Ok(this.is_nullptr())); } } pub fn create_nullptr(lua: &Lua) -> LuaResult { // https://en.cppreference.com/w/cpp/types/nullptr_t lua.create_userdata(RefData::new( ptr::null_mut::<()>().cast(), 0, // usize::MAX means that nullptr is can be 'any' pointer type // We check size of inner data. give ffi.box(1):ref() as argument which typed as i32:ptr() will fail, // throw lua error UNSIZED_BOUNDS, )) }