mirror of
https://github.com/lune-org/lune.git
synced 2025-04-03 01:50:55 +01:00
Improve code quality (#243)
This commit is contained in:
parent
b503cc9f5b
commit
c3f255db7a
14 changed files with 157 additions and 142 deletions
|
@ -1,3 +1,5 @@
|
|||
<!-- markdownlint-disable MD033 -->
|
||||
|
||||
# `lune-std-ffi`
|
||||
|
||||
## Tests & Benchmarks
|
||||
|
@ -6,24 +8,18 @@ See [tests/ffi](../../tests/ffi/README.md)
|
|||
|
||||
## TODO
|
||||
|
||||
- CString
|
||||
- [CString](./src/c/string_info.rs)
|
||||
- Add buffer as owned data support
|
||||
- Add math operation for numeric types
|
||||
|
||||
> Provide related methods: `CTypeInfo:add(target, from1, from2, ...)` and `:sub` `:mul` `:div` `:mod` `:pow` `:max` `:min` `:gt` `:lt`
|
||||
> Luau cannot handle f64, i64 or i128, so we should provide math operation for it
|
||||
|
||||
- Add bit operation for box/ref
|
||||
|
||||
> Luau only supports 32bit bit operations
|
||||
|
||||
- Add wchar and wstring support
|
||||
|
||||
> For windows API support
|
||||
|
||||
- Add varargs support
|
||||
- Array argument in cfn
|
||||
- More box/ref methods
|
||||
- [More box/ref methods](./src/data/helper.rs)
|
||||
- writeString
|
||||
- readString
|
||||
- writeBase64
|
||||
|
@ -35,30 +31,53 @@ See [tests/ffi](../../tests/ffi/README.md)
|
|||
|
||||
Define C-ABI type information and provide conversion and casting
|
||||
|
||||
- [**Struct ` CArrInfo`:**](./src/c/struct_info.rs) Represents C Array type
|
||||
- [**Struct ` CPtrInfo`:**](./src/c/ptr_info.rs) Represents C Pointer type
|
||||
- [**Struct ` CFnInfo`:**](./src/c/fn_info.rs) Represents C Function signature
|
||||
**Structs:** C ABI type informations
|
||||
|
||||
- [**Struct `CArrInfo`:**](./src/c/arr_info.rs) Represents C Array type
|
||||
- [**Struct `CPtrInfo`:**](./src/c/ptr_info.rs) Represents C Pointer type
|
||||
- [**Struct `CFnInfo`:**](./src/c/fn_info.rs) Represents C Function signature
|
||||
> provide `CallableData` and `ClosureData` creator
|
||||
- [**Struct ` CStructInfo`:**](./src/c/struct_info.rs) Represents C Struct type
|
||||
- [**Struct ` CTypeInfo<T>`:**](./src/c/type_info.rs) Represents C type, extended in `/c/types`
|
||||
- [**Struct `CStructInfo`:**](./src/c/struct_info.rs) Represents C Struct type
|
||||
- [**Struct `CTypeInfo<T>`:**](./src/c/type_info.rs) Represents C type, extended in `/c/types`
|
||||
|
||||
<details><summary><a href="./src/c/helper.rs"><strong>Mod <code>helper.rs</code>: C ABI type helper</strong></a></summary>
|
||||
|
||||
- **Function `get_conv`, `get_conv_list`:**
|
||||
get `FfiConvert` from userdata (CStruct, CArr, CPtr, CTypes)
|
||||
- **Function `get_middle_type`, `get_middle_type_list`:**
|
||||
get **`libffi::middle::Type`:** from userdata (CFn, CStruct, CArr, CPtr, CTypes)
|
||||
- **Function `get_size`:**
|
||||
get size from userdata
|
||||
- **Function `has_void`:**
|
||||
check table has void type
|
||||
- **Function `stringify`:**
|
||||
stringify any type userdata
|
||||
- **Function `get_name`:**
|
||||
get type name from ctype userdata, used for pretty-print
|
||||
- **Function `is_ctype`:** check userdata is ctype
|
||||
- **Mod `method_provider`:** provide common userdata method implements
|
||||
|
||||
</details>
|
||||
|
||||
#### /c/types
|
||||
|
||||
Export fixed-size source time known types and non-fixed compile time known types
|
||||
Implememt type-casting for all CTypes
|
||||
mod.rs implememts type-casting for all CTypes
|
||||
|
||||
**Mod `ctype_helper`:**
|
||||
<details><summary><a href="./src/c/types/mod.rs"><strong>Mod <code>ctype_helper</code>:</strong></a> c type helper</summary>
|
||||
|
||||
- **Function `get_conv`:**
|
||||
get _FfiConvert_ from some ctype userdata, used for struct and array conversion
|
||||
- **Function `get_size`:**
|
||||
get size from some ctype userdata, used for call return and arguments boundary checking
|
||||
- **Function `get_name`:**
|
||||
get type name from some ctype userdata, used for pretty-print
|
||||
get `FfiConvert` from ctype userdata, used for struct and array conversion
|
||||
- **Function `get_middle_type`:**
|
||||
get **`libffi::middle::Type`:** from some ctype userdata
|
||||
get **`libffi::middle::Type`:** from ctype userdata
|
||||
- **Function `get_size`:**
|
||||
get size from ctype userdata
|
||||
- **Function `get_name`:**
|
||||
get type name from ctype userdata, used for pretty-print
|
||||
- **Function `is_ctype`:** check userdata is ctype
|
||||
|
||||
</details>
|
||||
|
||||
---
|
||||
|
||||
### /data
|
||||
|
@ -83,12 +102,21 @@ Implememt type-casting for all CTypes
|
|||
- **Trait `FfiSize`**
|
||||
- **Trait `FfiSignedness`**
|
||||
|
||||
**Structs:** Provide call information trait
|
||||
<ul><li><details><summary><strong>Trait <code>FfiConvert</code>:</strong> Provide methods for read LuaValue from FfiData or write LuaValue into FfiData</summary>
|
||||
|
||||
- **Method `value_into_data`:** set data with lua value
|
||||
- **Method `value_from_data`:** get lua value from data
|
||||
- **Method `copy_data`:** copy sized data into another data
|
||||
- **Method `stringify_data`:** stringify data with specific type
|
||||
|
||||
</details></li></ul>
|
||||
|
||||
**Structs:** Provide call information
|
||||
|
||||
- **Struct `FfiArg`:** Used for argument boundary checking and callback argument ref flag
|
||||
- **Struct `FfiResult`:** Used for result boundary checking
|
||||
|
||||
**Trait `FfiData`:** Provide common data handle, including methods below
|
||||
<details><summary><strong>Trait <code>FfiData</code>:</strong> Provide common data handle, including methods below</summary>
|
||||
|
||||
- **Method `check_inner_boundary`:** check boundary with offset and size
|
||||
- **Method `get_inner_pointer`:** returns raw pointer `*mut ()`
|
||||
|
@ -96,24 +124,23 @@ Implememt type-casting for all CTypes
|
|||
- **Method `is_readable`**
|
||||
- **Method `copy_from`** copy data from another data
|
||||
|
||||
- **Trait `FfiConvert`:** Provide methods for read LuaValue from FfiData or write LuaValue into FfiData
|
||||
</details>
|
||||
|
||||
- **Method `value_into_data`:** set data with lua value
|
||||
- **Method `value_from_data`:** get lua value from data
|
||||
- **Method `copy_data`:** copy sized data into another data
|
||||
- **Method `stringify_data`:** stringify data with specific type
|
||||
|
||||
> Note: `GetFfiData` trait in `data/mod.rs` provides `AnyUserData.get_data_handle() -> FfiData` method
|
||||
> Note: `GetFfiData` trait in `data/mod.rs` provides `(LuaValue | LuaAnyUserData).get_data_handle() -> FfiData` method
|
||||
|
||||
**Mods:** Provide common helper functions
|
||||
|
||||
- [**Mod `association.rs`:**](./src/ffi/association.rs) GC utility, used for inner, ret and arg type holding in subtype
|
||||
- [**Mod `bit_mask.rs`:**](./src/ffi/bit_mask.rs) u8 bitfield helper
|
||||
- [**Mod `cast.rs`:**](./src/ffi/cast.rs) library
|
||||
- [**Mod `cast.rs`:**](./src/ffi/cast.rs) num cast library wrapper
|
||||
- **Function `num_cast<From, Into>(from: FfiData, from: FfiData)`:**
|
||||
Cast number type value inno another number type
|
||||
- [**Mod `libffi_helper.rs`:**](./src/ffi/libffi_helper.rs)
|
||||
- **Const `FFI_STATUS_NAMES`:** Used for `ffi_status` stringify
|
||||
- **Function `get_ensured_size`:** Returns ensured `ffi_type` size
|
||||
- **Const `SIZE_OF_POINTER`:** Platform specific pointer size (Compile time known)
|
||||
- **Function `ffi_status_assert`:** Convert `ffi_status` to `LuaResult<()>`
|
||||
|
||||
<ul><li><details><summary><a href="./src/c/struct_info.rs"><strong>Mod <code>libffi_helper.rs</code>:</strong></a> libffi library helper</summary>
|
||||
|
||||
- **Const `FFI_STATUS_NAMES`:** Stringify `ffi_status`
|
||||
- **Function `get_ensured_size`:** Returns ensured size of `ffi_type`
|
||||
- **Const `SIZE_OF_POINTER`:** Platform specific pointer size (Compile time known)
|
||||
- **Function `ffi_status_assert`:** Convert `ffi_status` to `LuaResult<()>`
|
||||
|
||||
</details></li></ul>
|
||||
|
|
|
@ -11,6 +11,7 @@ use crate::{
|
|||
pub mod method_provider {
|
||||
use super::*;
|
||||
|
||||
// Implement tostring
|
||||
pub fn provide_to_string<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
M: LuaUserDataMethods<'lua, Target>,
|
||||
|
@ -20,6 +21,7 @@ pub mod method_provider {
|
|||
});
|
||||
}
|
||||
|
||||
// Implement ptr method
|
||||
pub fn provide_ptr<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
M: LuaUserDataMethods<'lua, Target>,
|
||||
|
@ -29,6 +31,7 @@ pub mod method_provider {
|
|||
});
|
||||
}
|
||||
|
||||
// Implement arr method
|
||||
pub fn provide_arr<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
M: LuaUserDataMethods<'lua, Target>,
|
||||
|
@ -38,6 +41,7 @@ pub mod method_provider {
|
|||
});
|
||||
}
|
||||
|
||||
// Implement readData method
|
||||
pub fn provide_read_data<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
Target: FfiSize + FfiConvert,
|
||||
|
@ -61,6 +65,7 @@ pub mod method_provider {
|
|||
);
|
||||
}
|
||||
|
||||
// Implement writeData method
|
||||
pub fn provide_write_data<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
Target: FfiSize + FfiConvert,
|
||||
|
@ -85,6 +90,7 @@ pub mod method_provider {
|
|||
);
|
||||
}
|
||||
|
||||
// Implement copyData method
|
||||
pub fn provide_copy_data<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
Target: FfiSize + FfiConvert,
|
||||
|
@ -125,6 +131,7 @@ pub mod method_provider {
|
|||
);
|
||||
}
|
||||
|
||||
// Implement stringifyData method
|
||||
pub fn provide_stringify_data<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
Target: FfiSize + FfiConvert,
|
||||
|
@ -138,6 +145,7 @@ pub mod method_provider {
|
|||
);
|
||||
}
|
||||
|
||||
// Implement box method
|
||||
pub fn provide_box<'lua, Target, M>(methods: &mut M)
|
||||
where
|
||||
Target: FfiSize + FfiConvert,
|
||||
|
@ -149,43 +157,9 @@ pub mod method_provider {
|
|||
Ok(result)
|
||||
});
|
||||
}
|
||||
|
||||
// FIXME: Buffer support should be part of another PR
|
||||
// pub fn provide_write_buffer<'lua, Target, M>(methods: &mut M)
|
||||
// where
|
||||
// Target: FfiSize + FfiConvert,
|
||||
// M: LuaUserDataMethods<'lua, Target>,
|
||||
// {
|
||||
// methods.add_method(
|
||||
// "writeBuffer",
|
||||
// |lua, this, (target, value, offset): (LuaValue, LuaValue, Option<isize>)| {
|
||||
// if !target.is_buffer() {
|
||||
// return Err(LuaError::external(format!(
|
||||
// "Argument target must be a buffer, got {}",
|
||||
// target.type_name()
|
||||
// )));
|
||||
// }
|
||||
|
||||
// target.to_pointer()
|
||||
// target.as_userdata().unwrap().to_pointer()
|
||||
// let offset = offset.unwrap_or(0);
|
||||
|
||||
// let data_handle = &target.get_ffi_data()?;
|
||||
// // use or functions
|
||||
// if !data_handle.check_boundary(offset, this.get_size()) {
|
||||
// return Err(LuaError::external("Out of bounds"));
|
||||
// }
|
||||
// if !data_handle.is_writable() {
|
||||
// return Err(LuaError::external("Unwritable data handle"));
|
||||
// }
|
||||
|
||||
// unsafe { this.value_into_data(lua, offset, data_handle, value) }
|
||||
// },
|
||||
// );
|
||||
// }
|
||||
}
|
||||
|
||||
pub fn get_userdata(value: LuaValue) -> LuaResult<LuaAnyUserData> {
|
||||
fn get_userdata(value: LuaValue) -> LuaResult<LuaAnyUserData> {
|
||||
if let LuaValue::UserData(field_type) = value {
|
||||
Ok(field_type)
|
||||
} else {
|
||||
|
@ -212,9 +186,9 @@ pub fn create_list<T>(
|
|||
Ok(list)
|
||||
}
|
||||
|
||||
// Get the NativeConvert handle from the ctype userData
|
||||
// Get the dynamic FfiConvert handle from the userData
|
||||
// This is intended to avoid lookup userdata and lua table every time. (eg: struct)
|
||||
// The userdata must live longer than the NativeConvert handle
|
||||
// The userdata must live longer than the FfiConvert handle
|
||||
pub unsafe fn get_conv(userdata: &LuaAnyUserData) -> LuaResult<*const dyn FfiConvert> {
|
||||
if userdata.is::<CStructInfo>() {
|
||||
Ok(userdata.to_pointer().cast::<CStructInfo>() as *const dyn FfiConvert)
|
||||
|
|
|
@ -23,7 +23,7 @@ pub use self::{
|
|||
void_info::CVoidInfo,
|
||||
};
|
||||
|
||||
// Named registry table names
|
||||
// Named registry keys
|
||||
mod association_names {
|
||||
pub const CPTR_INNER: &str = "__cptr_inner";
|
||||
pub const CARR_INNER: &str = "__carr_inner";
|
||||
|
|
|
@ -12,7 +12,8 @@ use crate::{
|
|||
ffi::{libffi_helper::get_ensured_size, FfiConvert, FfiData, FfiSignedness, FfiSize},
|
||||
};
|
||||
|
||||
// Cast native data
|
||||
// Provide type casting
|
||||
// This trait should be implemented for each types
|
||||
pub trait CTypeCast {
|
||||
#[inline(always)]
|
||||
fn cast(
|
||||
|
@ -82,7 +83,7 @@ where
|
|||
self.name
|
||||
}
|
||||
|
||||
pub fn get_type(&self) -> Type {
|
||||
pub fn get_middle_type(&self) -> Type {
|
||||
self.middle_type.clone()
|
||||
}
|
||||
}
|
||||
|
@ -113,6 +114,9 @@ where
|
|||
method_provider::provide_copy_data(methods);
|
||||
method_provider::provide_stringify_data(methods);
|
||||
|
||||
// Math
|
||||
// TODO: Math support for numeric types
|
||||
|
||||
methods.add_function(
|
||||
"cast",
|
||||
|_lua,
|
||||
|
|
|
@ -10,20 +10,20 @@ use num::cast::AsPrimitive;
|
|||
use super::{CTypeCast, CTypeInfo};
|
||||
use crate::ffi::{num_cast, FfiConvert, FfiData, FfiSize};
|
||||
|
||||
pub mod f32;
|
||||
pub mod f64;
|
||||
pub mod i128;
|
||||
pub mod i16;
|
||||
pub mod i32;
|
||||
pub mod i64;
|
||||
pub mod i8;
|
||||
pub mod isize;
|
||||
pub mod u128;
|
||||
pub mod u16;
|
||||
pub mod u32;
|
||||
pub mod u64;
|
||||
pub mod u8;
|
||||
pub mod usize;
|
||||
mod f32;
|
||||
mod f64;
|
||||
mod i128;
|
||||
mod i16;
|
||||
mod i32;
|
||||
mod i64;
|
||||
mod i8;
|
||||
mod isize;
|
||||
mod u128;
|
||||
mod u16;
|
||||
mod u32;
|
||||
mod u64;
|
||||
mod u8;
|
||||
mod usize;
|
||||
|
||||
// CType userdata export
|
||||
macro_rules! create_ctypes {
|
||||
|
@ -92,20 +92,20 @@ macro_rules! define_cast_num {
|
|||
}
|
||||
impl<From> CTypeCast for CTypeInfo<From>
|
||||
where
|
||||
From: AsPrimitive<u8>
|
||||
+ AsPrimitive<u16>
|
||||
+ AsPrimitive<u32>
|
||||
+ AsPrimitive<u64>
|
||||
+ AsPrimitive<u128>
|
||||
+ AsPrimitive<i8>
|
||||
From: AsPrimitive<f32>
|
||||
+ AsPrimitive<f64>
|
||||
+ AsPrimitive<i128>
|
||||
+ AsPrimitive<i16>
|
||||
+ AsPrimitive<i32>
|
||||
+ AsPrimitive<i64>
|
||||
+ AsPrimitive<i128>
|
||||
+ AsPrimitive<f32>
|
||||
+ AsPrimitive<f64>
|
||||
+ AsPrimitive<usize>
|
||||
+ AsPrimitive<isize>,
|
||||
+ AsPrimitive<i8>
|
||||
+ AsPrimitive<isize>
|
||||
+ AsPrimitive<u128>
|
||||
+ AsPrimitive<u16>
|
||||
+ AsPrimitive<u32>
|
||||
+ AsPrimitive<u64>
|
||||
+ AsPrimitive<u8>
|
||||
+ AsPrimitive<usize>,
|
||||
{
|
||||
fn cast(
|
||||
&self,
|
||||
|
@ -118,7 +118,7 @@ where
|
|||
) -> LuaResult<()> {
|
||||
define_cast_num!(
|
||||
From, self, from_info, into_info, from, into, from_offset, into_offset,
|
||||
u8 u16 u32 u64 u128 i8 i16 i32 i64 i128 f32 f64 usize isize
|
||||
f32 f64 i128 i16 i32 i64 i8 isize u128 u16 u32 u64 u8 usize
|
||||
)
|
||||
}
|
||||
}
|
||||
|
@ -138,7 +138,22 @@ pub mod ctype_helper {
|
|||
}
|
||||
#[inline]
|
||||
pub fn get_conv(userdata: &LuaAnyUserData) -> LuaResult<*const dyn FfiConvert> {
|
||||
define_get_conv!(userdata, u8 u16 u32 u64 u128 i8 i16 i32 i64 i128 f32 f64 usize isize)
|
||||
define_get_conv!(userdata, f32 f64 i128 i16 i32 i64 i8 isize u128 u16 u32 u64 u8 usize)
|
||||
}
|
||||
|
||||
// Get libffi_type of ctype
|
||||
macro_rules! define_get_middle_type {
|
||||
($userdata:ident, $( $rust_type:ty )*) => {
|
||||
$( if $userdata.is::<CTypeInfo<$rust_type>>() {
|
||||
Ok(Some($userdata.borrow::<CTypeInfo<$rust_type>>()?.get_middle_type()))
|
||||
} else )* {
|
||||
Ok(None)
|
||||
}
|
||||
};
|
||||
}
|
||||
#[inline]
|
||||
pub fn get_middle_type(userdata: &LuaAnyUserData) -> LuaResult<Option<Type>> {
|
||||
define_get_middle_type!(userdata, f32 f64 i128 i16 i32 i64 i8 isize u128 u16 u32 u64 u8 usize)
|
||||
}
|
||||
|
||||
// Get size of ctype (not including struct, arr, ... only CType<*>)
|
||||
|
@ -153,7 +168,7 @@ pub mod ctype_helper {
|
|||
}
|
||||
#[inline]
|
||||
pub fn get_size(userdata: &LuaAnyUserData) -> LuaResult<usize> {
|
||||
define_get_size!(userdata, u8 u16 u32 u64 u128 i8 i16 i32 i64 i128 f32 f64 usize isize)
|
||||
define_get_size!(userdata, f32 f64 i128 i16 i32 i64 i8 isize u128 u16 u32 u64 u8 usize)
|
||||
}
|
||||
|
||||
// Get name of ctype
|
||||
|
@ -168,22 +183,7 @@ pub mod ctype_helper {
|
|||
}
|
||||
#[inline]
|
||||
pub fn get_name(userdata: &LuaAnyUserData) -> LuaResult<Option<&'static str>> {
|
||||
define_get_name!(userdata, u8 u16 u32 u64 u128 i8 i16 i32 i64 i128 f32 f64 usize isize)
|
||||
}
|
||||
|
||||
// Get libffi_type of ctype
|
||||
macro_rules! define_get_middle_type {
|
||||
($userdata:ident, $( $rust_type:ty )*) => {
|
||||
$( if $userdata.is::<CTypeInfo<$rust_type>>() {
|
||||
Ok(Some($userdata.borrow::<CTypeInfo<$rust_type>>()?.get_type()))
|
||||
} else )* {
|
||||
Ok(None)
|
||||
}
|
||||
};
|
||||
}
|
||||
#[inline]
|
||||
pub fn get_middle_type(userdata: &LuaAnyUserData) -> LuaResult<Option<Type>> {
|
||||
define_get_middle_type!(userdata, u8 u16 u32 u64 u128 i8 i16 i32 i64 i128 f32 f64 usize isize)
|
||||
define_get_name!(userdata, f32 f64 i128 i16 i32 i64 i8 isize u128 u16 u32 u64 u8 usize)
|
||||
}
|
||||
|
||||
// Check whether userdata is ctype or not
|
||||
|
@ -198,6 +198,6 @@ pub mod ctype_helper {
|
|||
}
|
||||
#[inline]
|
||||
pub fn is_ctype(userdata: &LuaAnyUserData) -> bool {
|
||||
define_is_ctype!(userdata, u8 u16 u32 u64 u128 i8 i16 i32 i64 i128 f32 f64 usize isize)
|
||||
define_is_ctype!(userdata, f32 f64 i128 i16 i32 i64 i8 isize u128 u16 u32 u64 u8 usize)
|
||||
}
|
||||
}
|
||||
|
|
|
@ -7,7 +7,7 @@ pub enum BoxFlag {
|
|||
impl BoxFlag {
|
||||
pub const fn value(&self) -> u8 {
|
||||
match self {
|
||||
Self::Leaked => U8_MASK2,
|
||||
Self::Leaked => U8_MASK1,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -73,7 +73,7 @@ impl BoxData {
|
|||
|
||||
// Calculate offset
|
||||
if let Some(t) = offset {
|
||||
if !bounds.check_boundary(t) {
|
||||
if !bounds.check_offset(t) {
|
||||
return Err(LuaError::external(format!(
|
||||
"Offset out of bounds (box.size: {}, got {})",
|
||||
target.size(),
|
||||
|
@ -113,7 +113,6 @@ impl Drop for BoxData {
|
|||
}
|
||||
|
||||
impl FfiData for BoxData {
|
||||
#[inline]
|
||||
fn check_inner_boundary(&self, offset: isize, size: usize) -> bool {
|
||||
if offset < 0 {
|
||||
return false;
|
||||
|
@ -124,11 +123,9 @@ impl FfiData for BoxData {
|
|||
unsafe fn get_inner_pointer(&self) -> *mut () {
|
||||
self.data.as_ptr().cast_mut().cast::<()>()
|
||||
}
|
||||
#[inline]
|
||||
fn is_readable(&self) -> bool {
|
||||
true
|
||||
}
|
||||
#[inline]
|
||||
fn is_writable(&self) -> bool {
|
||||
true
|
||||
}
|
||||
|
|
|
@ -8,7 +8,7 @@ use libffi::{
|
|||
use mlua::prelude::*;
|
||||
|
||||
use super::{
|
||||
association_names::CLSOURE_REF_INNER,
|
||||
association_names::REF_INNER,
|
||||
ref_data::{RefBounds, RefData, RefFlag, UNSIZED_BOUNDS},
|
||||
};
|
||||
use crate::ffi::{
|
||||
|
@ -145,7 +145,7 @@ impl LuaUserData for ClosureData {
|
|||
CLOSURE_REF_FLAGS,
|
||||
UNSIZED_BOUNDS,
|
||||
))?;
|
||||
association::set(lua, CLSOURE_REF_INNER, &ref_data, &this)?;
|
||||
association::set(lua, REF_INNER, &ref_data, &this)?;
|
||||
Ok(ref_data)
|
||||
});
|
||||
}
|
||||
|
|
|
@ -38,4 +38,7 @@ pub mod method_provider {
|
|||
},
|
||||
);
|
||||
}
|
||||
|
||||
// TODO: writeString, readString, writeBase64 and readBase64 methods
|
||||
// TODO: Bit operation support
|
||||
}
|
||||
|
|
|
@ -19,11 +19,10 @@ pub use self::{
|
|||
};
|
||||
use crate::ffi::FfiData;
|
||||
|
||||
// Named registry table names
|
||||
// Named registry keys
|
||||
mod association_names {
|
||||
pub const REF_INNER: &str = "__ref_inner";
|
||||
pub const SYM_INNER: &str = "__syn_inner";
|
||||
pub const CLSOURE_REF_INNER: &str = "__closure_ref_inner";
|
||||
}
|
||||
|
||||
// Get dynamic FfiData handle from LuaValue and LuaAnyUserData
|
||||
|
|
|
@ -1,8 +1,8 @@
|
|||
// Memory boundaries
|
||||
pub struct RefBounds {
|
||||
// How much data available above
|
||||
// How much bytes available above
|
||||
pub(crate) above: usize,
|
||||
// How much data available below
|
||||
// How much bytes available below
|
||||
pub(crate) below: usize,
|
||||
}
|
||||
|
||||
|
@ -16,9 +16,9 @@ impl RefBounds {
|
|||
Self { above, below }
|
||||
}
|
||||
|
||||
// Check boundary
|
||||
// Check offset is in boundary
|
||||
#[inline]
|
||||
pub fn check_boundary(&self, offset: isize) -> bool {
|
||||
pub fn check_offset(&self, offset: isize) -> bool {
|
||||
let offset_abs = offset.unsigned_abs();
|
||||
match offset.signum() {
|
||||
-1 => self.above >= offset_abs,
|
||||
|
@ -52,8 +52,21 @@ impl RefBounds {
|
|||
end <= 0 || self.below >= end.unsigned_abs()
|
||||
}
|
||||
|
||||
// Calculate new boundaries with bounds and offset
|
||||
// Calculate new boundaries with offset
|
||||
// No boundary checking in here
|
||||
//
|
||||
// Above = 3
|
||||
// ∧ ───∧───
|
||||
// -3│ │ New above = 2
|
||||
// -2│ │
|
||||
// -1│ <────── Offset = -1
|
||||
// 0│ │
|
||||
// 1│ │
|
||||
// 2│ │
|
||||
// 3│ │ New below = 4
|
||||
// ∨ ───∨───
|
||||
// Below = 3
|
||||
//
|
||||
#[inline]
|
||||
pub fn offset(&self, offset: isize) -> Self {
|
||||
let sign = offset.signum();
|
||||
|
|
|
@ -8,6 +8,7 @@ pub enum RefFlag {
|
|||
Offsetable,
|
||||
Function,
|
||||
}
|
||||
|
||||
impl RefFlag {
|
||||
pub const fn value(&self) -> u8 {
|
||||
match self {
|
||||
|
|
|
@ -41,7 +41,7 @@ impl RefData {
|
|||
}
|
||||
}
|
||||
|
||||
// Create reference of this reference box
|
||||
// Create reference of this reference
|
||||
pub fn luaref<'lua>(
|
||||
lua: &'lua Lua,
|
||||
this: LuaAnyUserData<'lua>,
|
||||
|
@ -57,7 +57,7 @@ impl RefData {
|
|||
},
|
||||
))?;
|
||||
|
||||
// Make new reference live longer then this reference
|
||||
// Make sure new reference live longer then this reference
|
||||
association::set(lua, REF_INNER, &luaref, &this)?;
|
||||
|
||||
Ok(luaref)
|
||||
|
@ -100,7 +100,7 @@ impl RefData {
|
|||
}
|
||||
|
||||
// Check boundary
|
||||
if !self.boundary.check_boundary(offset) {
|
||||
if !self.boundary.check_offset(offset) {
|
||||
return Err(LuaError::external(format!(
|
||||
"Offset out of bounds (high: {}, low: {}, got {})",
|
||||
self.boundary.above, self.boundary.below, offset
|
||||
|
@ -130,7 +130,6 @@ impl Drop for RefData {
|
|||
}
|
||||
|
||||
impl FfiData for RefData {
|
||||
#[inline]
|
||||
fn check_inner_boundary(&self, offset: isize, size: usize) -> bool {
|
||||
self.boundary.check_sized(offset, size)
|
||||
}
|
||||
|
@ -138,11 +137,9 @@ impl FfiData for RefData {
|
|||
unsafe fn get_inner_pointer(&self) -> *mut () {
|
||||
**self.ptr
|
||||
}
|
||||
#[inline]
|
||||
fn is_readable(&self) -> bool {
|
||||
u8_test(self.flags, RefFlag::Readable.value())
|
||||
}
|
||||
#[inline]
|
||||
fn is_writable(&self) -> bool {
|
||||
u8_test(self.flags, RefFlag::Writable.value())
|
||||
}
|
||||
|
|
|
@ -23,7 +23,7 @@ pub fn get_ensured_size(ffi_type: *mut raw::ffi_type) -> LuaResult<usize> {
|
|||
}
|
||||
|
||||
// Converts ffi status into &str for formatting
|
||||
pub const FFI_STATUS_NAMES: [&str; 4] = [
|
||||
const FFI_STATUS_NAMES: [&str; 4] = [
|
||||
"ffi_status_FFI_OK",
|
||||
"ffi_status_FFI_BAD_TYPEDEF",
|
||||
"ffi_status_FFI_BAD_ABI",
|
||||
|
|
Loading…
Add table
Reference in a new issue