mirror of
https://github.com/lune-org/lune.git
synced 2025-04-03 18:10:54 +01:00
Implement ClosureData (#243)
This commit is contained in:
parent
da30dfb3f0
commit
e19d9748e5
37 changed files with 306 additions and 216 deletions
|
@ -9,7 +9,7 @@ Define C-ABI type information and provide conversion and casting
|
||||||
- [**Struct ` CArrInfo`:**](./src/c/struct_info.rs) Represents C Array type
|
- [**Struct ` CArrInfo`:**](./src/c/struct_info.rs) Represents C Array type
|
||||||
- [**Struct ` CPtrInfo`:**](./src/c/ptr_info.rs) Represents C Pointer type
|
- [**Struct ` CPtrInfo`:**](./src/c/ptr_info.rs) Represents C Pointer type
|
||||||
- [**Struct ` CFnInfo`:**](./src/c/fn_info.rs) Represents C Function signature
|
- [**Struct ` CFnInfo`:**](./src/c/fn_info.rs) Represents C Function signature
|
||||||
provide CallableData and ClosureData creation
|
> provide CallableData and ClosureData creation
|
||||||
- [**Struct ` CStructInfo`:**](./src/c/struct_info.rs) Represents C Struct type
|
- [**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 ` CTypeInfo<T>`:**](./src/c/type_info.rs) Represents C type, extended in `/c/types`
|
||||||
|
|
||||||
|
@ -55,6 +55,11 @@ Implememt type-casting for all CTypes
|
||||||
- **Trait `FfiSignedness`**
|
- **Trait `FfiSignedness`**
|
||||||
- **Trait `FfiConvert`:** Provide read LuaValue from FfiData or write LuaValue into FfiData
|
- **Trait `FfiConvert`:** Provide read LuaValue from FfiData or write LuaValue into FfiData
|
||||||
|
|
||||||
|
**Traits:** Provide call information trait
|
||||||
|
|
||||||
|
- **Trait `FfiArg`:** Used for argument boundary checking and callback argument ref flag
|
||||||
|
- **Trait `FfiResult`:** Used for result boundary checking
|
||||||
|
|
||||||
**Trait `FfiData`:** Provide common data handle, including methods below
|
**Trait `FfiData`:** Provide common data handle, including methods below
|
||||||
|
|
||||||
- **Method `check_boundary`:** check boundary with offset and size
|
- **Method `check_boundary`:** check boundary with offset and size
|
||||||
|
@ -66,14 +71,22 @@ Implememt type-casting for all CTypes
|
||||||
|
|
||||||
**Mods:** Provide common helper functions
|
**Mods:** Provide common helper functions
|
||||||
|
|
||||||
- **`association.rs`:** GC utility, used for inner, ret and arg type holding in subtype
|
- [**Mod `association.rs`:**](./src/ffi/association.rs) GC utility, used for inner, ret and arg type holding in subtype
|
||||||
- **`bit_mask.rs`:** u8 bitfield helper
|
- [**Mod `bit_mask.rs`:**](./src/ffi/bit_mask.rs) u8 bitfield helper
|
||||||
- **`cast.rs`:** library
|
- [**Mod `cast.rs`:**](./src/ffi/cast.rs) library
|
||||||
- **Function `num_cast<From, Into>(from: FfiData, from: FfiData)`:**
|
- **Function `num_cast<From, Into>(from: FfiData, from: FfiData)`:**
|
||||||
Cast number type value inno another number type
|
Cast number type value inno another number type
|
||||||
- **`libffi_helper.rs`:**
|
- [**Mod `libffi_helper.rs`:**](./src/ffi/libffi_helper.rs)
|
||||||
- **Const `FFI_STATUS_NAMES`:** Used for ffi_status stringify
|
- **Const `FFI_STATUS_NAMES`:** Used for ffi_status stringify
|
||||||
- **Function `get_ensured_size`:** Returns ensured ffi_type size
|
- **Function `get_ensured_size`:** Returns ensured ffi_type size
|
||||||
- **Const `SIEE_OF_POINTER`:** Platform specific pointer size (Compile time known)
|
- **Const `SIEE_OF_POINTER`:** Platform specific pointer size (Compile time known)
|
||||||
|
|
||||||
## TODO
|
## TODO
|
||||||
|
|
||||||
|
Add `CTypeInfo:add(target, from1, from2, ...)` and `:sub` `:mul` `:div` `:mod` `:pow` for math operation.
|
||||||
|
|
||||||
|
> Luau cannot handle i64 or i128
|
||||||
|
|
||||||
|
Add bor band and such bit-related operation
|
||||||
|
|
||||||
|
> Luau only supports 32bit bit operations
|
||||||
|
|
|
@ -4,10 +4,7 @@ use libffi::middle::Type;
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{association_names::CARR_INNER, helper, method_provider};
|
use super::{association_names::CARR_INNER, helper, method_provider};
|
||||||
use crate::{
|
use crate::ffi::{association, libffi_helper::get_ensured_size, FfiConvert, FfiData, FfiSize};
|
||||||
data::{FfiConvert, FfiData, FfiSize},
|
|
||||||
ffi::{association, libffi_helper::get_ensured_size},
|
|
||||||
};
|
|
||||||
|
|
||||||
// This is a series of some type.
|
// This is a series of some type.
|
||||||
// It provides the final size and the offset of the index,
|
// It provides the final size and the offset of the index,
|
||||||
|
|
|
@ -4,12 +4,15 @@ use libffi::middle::{Cif, Type};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
association_names::{CALLABLE_CFN, CALLABLE_REF, CFN_ARGS, CFN_RESULT},
|
association_names::{
|
||||||
helper, method_provider,
|
CALLABLE_CFN, CALLABLE_REF, CFN_ARGS, CFN_RESULT, CLOSURE_CFN, CLOSURE_FUNC,
|
||||||
|
},
|
||||||
|
ctype_helper::is_ctype,
|
||||||
|
helper, method_provider, CArrInfo, CPtrInfo, CStructInfo,
|
||||||
};
|
};
|
||||||
use crate::{
|
use crate::{
|
||||||
data::{CallableData, RefData, RefDataFlag},
|
data::{CallableData, ClosureData, RefData, RefFlag},
|
||||||
ffi::{association, bit_mask::*, FfiArgInfo, FfiData, FfiResultInfo, FfiSignedness, FfiSize},
|
ffi::{association, bit_mask::*, FfiArg, FfiData, FfiResult, FfiSignedness, FfiSize},
|
||||||
};
|
};
|
||||||
|
|
||||||
// cfn is a type declaration for a function.
|
// cfn is a type declaration for a function.
|
||||||
|
@ -30,8 +33,8 @@ use crate::{
|
||||||
|
|
||||||
pub struct CFnInfo {
|
pub struct CFnInfo {
|
||||||
cif: Cif,
|
cif: Cif,
|
||||||
arg_info_list: Vec<FfiArgInfo>,
|
arg_info_list: Vec<FfiArg>,
|
||||||
result_info: FfiResultInfo,
|
result_info: FfiResult,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FfiSignedness for CFnInfo {
|
impl FfiSignedness for CFnInfo {
|
||||||
|
@ -45,15 +48,39 @@ impl FfiSize for CFnInfo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const CALLBACK_ARG_REF_FLAG_TYPE: u8 = RefFlag::Readable.value();
|
||||||
|
const CALLBACK_ARG_REF_FLAG_PTR: u8 = RefFlag::Dereferenceable.value() | RefFlag::Readable.value();
|
||||||
|
const CALLBACK_ARG_REF_FLAG_ARR: u8 = RefFlag::Readable.value() | RefFlag::Offsetable.value();
|
||||||
|
const CALLBACK_ARG_REF_FLAG_STRUCT: u8 = RefFlag::Readable.value() | RefFlag::Offsetable.value();
|
||||||
|
const CALLBACK_ARG_REF_FLAG_CFN: u8 = RefFlag::Function.value();
|
||||||
|
|
||||||
|
fn create_arg_info(userdata: &LuaAnyUserData) -> LuaResult<FfiArg> {
|
||||||
|
let callback_ref_flag = if is_ctype(userdata) {
|
||||||
|
CALLBACK_ARG_REF_FLAG_TYPE
|
||||||
|
} else if userdata.is::<CPtrInfo>() {
|
||||||
|
CALLBACK_ARG_REF_FLAG_PTR
|
||||||
|
} else if userdata.is::<CArrInfo>() {
|
||||||
|
CALLBACK_ARG_REF_FLAG_ARR
|
||||||
|
} else if userdata.is::<CStructInfo>() {
|
||||||
|
CALLBACK_ARG_REF_FLAG_STRUCT
|
||||||
|
} else if userdata.is::<CFnInfo>() {
|
||||||
|
CALLBACK_ARG_REF_FLAG_CFN
|
||||||
|
} else {
|
||||||
|
return Err(LuaError::external("unexpected type userdata"));
|
||||||
|
};
|
||||||
|
Ok(FfiArg {
|
||||||
|
size: helper::get_size(userdata)?,
|
||||||
|
callback_ref_flag,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
impl CFnInfo {
|
impl CFnInfo {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
args: Vec<Type>,
|
args: Vec<Type>,
|
||||||
ret: Type,
|
ret: Type,
|
||||||
arg_info_list: Vec<FfiArgInfo>,
|
arg_info_list: Vec<FfiArg>,
|
||||||
result_info: FfiResultInfo,
|
result_info: FfiResult,
|
||||||
) -> LuaResult<Self> {
|
) -> LuaResult<Self> {
|
||||||
// let cif = ;
|
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
cif: Cif::new(args.clone(), ret.clone()),
|
cif: Cif::new(args.clone(), ret.clone()),
|
||||||
arg_info_list,
|
arg_info_list,
|
||||||
|
@ -61,7 +88,7 @@ impl CFnInfo {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_from_table<'lua>(
|
pub fn from_table<'lua>(
|
||||||
lua: &'lua Lua,
|
lua: &'lua Lua,
|
||||||
arg_table: LuaTable,
|
arg_table: LuaTable,
|
||||||
ret: LuaAnyUserData,
|
ret: LuaAnyUserData,
|
||||||
|
@ -70,16 +97,12 @@ impl CFnInfo {
|
||||||
let ret_type = helper::get_middle_type(&ret)?;
|
let ret_type = helper::get_middle_type(&ret)?;
|
||||||
|
|
||||||
let arg_len = arg_table.raw_len();
|
let arg_len = arg_table.raw_len();
|
||||||
let mut arg_info_list = Vec::<FfiArgInfo>::with_capacity(arg_len);
|
let mut arg_info_list = Vec::<FfiArg>::with_capacity(arg_len);
|
||||||
for index in 0..arg_len {
|
for index in 0..arg_len {
|
||||||
let userdata = helper::get_userdata(arg_table.raw_get(index + 1)?)?;
|
let userdata = helper::get_userdata(arg_table.raw_get(index + 1)?)?;
|
||||||
arg_info_list.push(FfiArgInfo {
|
arg_info_list.push(create_arg_info(&userdata)?);
|
||||||
conv: unsafe { helper::get_conv(&userdata)? },
|
|
||||||
size: helper::get_size(&userdata)?,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
let result_info = FfiResultInfo {
|
let result_info = FfiResult {
|
||||||
conv: unsafe { helper::get_conv(&ret)? },
|
|
||||||
size: helper::get_size(&ret)?,
|
size: helper::get_size(&ret)?,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -122,6 +145,57 @@ impl CFnInfo {
|
||||||
Err(LuaError::external("failed to get inner type userdata."))
|
Err(LuaError::external("failed to get inner type userdata."))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn create_closure<'lua>(
|
||||||
|
&self,
|
||||||
|
lua: &'lua Lua,
|
||||||
|
this: &LuaAnyUserData,
|
||||||
|
lua_function: LuaFunction<'lua>,
|
||||||
|
) -> LuaResult<LuaAnyUserData<'lua>> {
|
||||||
|
let closure = ClosureData::new(
|
||||||
|
ptr::from_ref(lua),
|
||||||
|
self.cif.as_raw_ptr(),
|
||||||
|
self.arg_info_list.clone(),
|
||||||
|
self.result_info.clone(),
|
||||||
|
lua.create_registry_value(&lua_function)?,
|
||||||
|
)?;
|
||||||
|
let closure_userdata = lua.create_userdata(closure)?;
|
||||||
|
|
||||||
|
association::set(lua, CLOSURE_CFN, &closure_userdata, this)?;
|
||||||
|
association::set(lua, CLOSURE_FUNC, &closure_userdata, lua_function)?;
|
||||||
|
|
||||||
|
Ok(closure_userdata)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn create_callable<'lua>(
|
||||||
|
&self,
|
||||||
|
lua: &'lua Lua,
|
||||||
|
this: &LuaAnyUserData,
|
||||||
|
target_ref: &LuaAnyUserData,
|
||||||
|
) -> LuaResult<LuaAnyUserData<'lua>> {
|
||||||
|
if !target_ref.is::<RefData>() {
|
||||||
|
return Err(LuaError::external("argument 0 must be ffiref"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let ffi_ref = target_ref.borrow::<RefData>()?;
|
||||||
|
if u8_test_not(ffi_ref.flags, RefFlag::Function.value()) {
|
||||||
|
return Err(LuaError::external("not a function ref"));
|
||||||
|
}
|
||||||
|
|
||||||
|
let callable = lua.create_userdata(unsafe {
|
||||||
|
CallableData::new(
|
||||||
|
self.cif.as_raw_ptr(),
|
||||||
|
self.arg_info_list.clone(),
|
||||||
|
self.result_info.clone(),
|
||||||
|
ffi_ref.get_pointer(),
|
||||||
|
)
|
||||||
|
})?;
|
||||||
|
|
||||||
|
association::set(lua, CALLABLE_CFN, &callable, this)?;
|
||||||
|
association::set(lua, CALLABLE_REF, &callable, target_ref)?;
|
||||||
|
|
||||||
|
Ok(callable)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl LuaUserData for CFnInfo {
|
impl LuaUserData for CFnInfo {
|
||||||
|
@ -134,36 +208,18 @@ impl LuaUserData for CFnInfo {
|
||||||
method_provider::provide_to_string(methods);
|
method_provider::provide_to_string(methods);
|
||||||
|
|
||||||
// Realize
|
// Realize
|
||||||
// methods.add_method("closure", |lua, this, func: LuaFunction| {
|
methods.add_function(
|
||||||
// lua.create_userdata(FfiClosure::new(this.cif, userdata))
|
"closure",
|
||||||
// })
|
|lua, (cfn, func): (LuaAnyUserData, LuaFunction)| {
|
||||||
|
let this = cfn.borrow::<CFnInfo>()?;
|
||||||
|
this.create_closure(lua, cfn.as_ref(), func)
|
||||||
|
},
|
||||||
|
);
|
||||||
methods.add_function(
|
methods.add_function(
|
||||||
"callable",
|
"callable",
|
||||||
|lua, (cfn, function_ref): (LuaAnyUserData, LuaAnyUserData)| {
|
|lua, (cfn, target): (LuaAnyUserData, LuaAnyUserData)| {
|
||||||
let this = cfn.borrow::<CFnInfo>()?;
|
let this = cfn.borrow::<CFnInfo>()?;
|
||||||
|
this.create_callable(lua, cfn.as_ref(), &target)
|
||||||
if !function_ref.is::<RefData>() {
|
|
||||||
return Err(LuaError::external("argument 0 must be ffiref"));
|
|
||||||
}
|
|
||||||
|
|
||||||
let ffi_ref = function_ref.borrow::<RefData>()?;
|
|
||||||
if u8_test_not(ffi_ref.flags, RefDataFlag::Function.value()) {
|
|
||||||
return Err(LuaError::external("not a function ref"));
|
|
||||||
}
|
|
||||||
|
|
||||||
let callable = lua.create_userdata(unsafe {
|
|
||||||
CallableData::new(
|
|
||||||
this.cif.as_raw_ptr(),
|
|
||||||
ptr::from_ref(&this.arg_info_list),
|
|
||||||
ptr::from_ref(&this.result_info),
|
|
||||||
ffi_ref.get_pointer(),
|
|
||||||
)
|
|
||||||
})?;
|
|
||||||
|
|
||||||
association::set(lua, CALLABLE_CFN, &callable, cfn.clone())?;
|
|
||||||
association::set(lua, CALLABLE_REF, &callable, function_ref.clone())?;
|
|
||||||
|
|
||||||
Ok(callable)
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,10 @@ use lune_utils::fmt::{pretty_format_value, ValueFormatConfig};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{ctype_helper, CArrInfo, CFnInfo, CPtrInfo, CStructInfo};
|
use super::{ctype_helper, CArrInfo, CFnInfo, CPtrInfo, CStructInfo};
|
||||||
use crate::data::{BoxData, FfiConvert, FfiSize, GetFfiData};
|
use crate::{
|
||||||
|
data::{BoxData, GetFfiData},
|
||||||
|
ffi::{FfiConvert, FfiSize},
|
||||||
|
};
|
||||||
|
|
||||||
pub mod method_provider {
|
pub mod method_provider {
|
||||||
use super::*;
|
use super::*;
|
||||||
|
|
|
@ -25,4 +25,6 @@ mod association_names {
|
||||||
pub const CFN_ARGS: &str = "__cfn_args";
|
pub const CFN_ARGS: &str = "__cfn_args";
|
||||||
pub const CALLABLE_REF: &str = "__callable_ref";
|
pub const CALLABLE_REF: &str = "__callable_ref";
|
||||||
pub const CALLABLE_CFN: &str = "__callable_cfn";
|
pub const CALLABLE_CFN: &str = "__callable_cfn";
|
||||||
|
pub const CLOSURE_FUNC: &str = "__closure_func";
|
||||||
|
pub const CLOSURE_CFN: &str = "__closure_cfn";
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,8 +5,10 @@ use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{association_names::CPTR_INNER, ctype_helper, helper, method_provider};
|
use super::{association_names::CPTR_INNER, ctype_helper, helper, method_provider};
|
||||||
use crate::{
|
use crate::{
|
||||||
data::{FfiConvert, FfiData, FfiSignedness, FfiSize, RefData},
|
data::RefData,
|
||||||
ffi::{association, libffi_helper::SIEE_OF_POINTER},
|
ffi::{
|
||||||
|
association, libffi_helper::SIZE_OF_POINTER, FfiConvert, FfiData, FfiSignedness, FfiSize,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct CPtrInfo {
|
pub struct CPtrInfo {
|
||||||
|
@ -20,7 +22,7 @@ impl FfiSignedness for CPtrInfo {
|
||||||
}
|
}
|
||||||
impl FfiSize for CPtrInfo {
|
impl FfiSize for CPtrInfo {
|
||||||
fn get_size(&self) -> usize {
|
fn get_size(&self) -> usize {
|
||||||
SIEE_OF_POINTER
|
SIZE_OF_POINTER
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
impl FfiConvert for CPtrInfo {
|
impl FfiConvert for CPtrInfo {
|
||||||
|
|
|
@ -4,9 +4,8 @@ use libffi::{low, middle::Type, raw};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{association_names::CSTRUCT_INNER, helper, method_provider};
|
use super::{association_names::CSTRUCT_INNER, helper, method_provider};
|
||||||
use crate::{
|
use crate::ffi::{
|
||||||
data::{FfiConvert, FfiData, FfiSignedness, FfiSize},
|
association, libffi_helper::FFI_STATUS_NAMES, FfiConvert, FfiData, FfiSignedness, FfiSize,
|
||||||
ffi::{association, libffi_helper::FFI_STATUS_NAMES},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct CStructInfo {
|
pub struct CStructInfo {
|
||||||
|
|
|
@ -8,8 +8,8 @@ use mlua::prelude::*;
|
||||||
|
|
||||||
use super::method_provider;
|
use super::method_provider;
|
||||||
use crate::{
|
use crate::{
|
||||||
data::{FfiConvert, FfiData, FfiSignedness, FfiSize, GetFfiData},
|
data::GetFfiData,
|
||||||
ffi::libffi_helper::get_ensured_size,
|
ffi::{libffi_helper::get_ensured_size, FfiConvert, FfiData, FfiSignedness, FfiSize},
|
||||||
};
|
};
|
||||||
|
|
||||||
// Cast native data
|
// Cast native data
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<f32> {
|
impl FfiSignedness for CTypeInfo<f32> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<f64> {
|
impl FfiSignedness for CTypeInfo<f64> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<i128> {
|
impl FfiSignedness for CTypeInfo<i128> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<i16> {
|
impl FfiSignedness for CTypeInfo<i16> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<i32> {
|
impl FfiSignedness for CTypeInfo<i32> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<i64> {
|
impl FfiSignedness for CTypeInfo<i64> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<i8> {
|
impl FfiSignedness for CTypeInfo<i8> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<isize> {
|
impl FfiSignedness for CTypeInfo<isize> {
|
||||||
|
|
|
@ -8,7 +8,7 @@ use mlua::prelude::*;
|
||||||
use num::cast::AsPrimitive;
|
use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use super::{CTypeCast, CTypeInfo};
|
use super::{CTypeCast, CTypeInfo};
|
||||||
use crate::data::{num_cast, FfiConvert, FfiData, FfiSize};
|
use crate::ffi::{num_cast, FfiConvert, FfiData, FfiSize};
|
||||||
|
|
||||||
pub mod f32;
|
pub mod f32;
|
||||||
pub mod f64;
|
pub mod f64;
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<u128> {
|
impl FfiSignedness for CTypeInfo<u128> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<u16> {
|
impl FfiSignedness for CTypeInfo<u16> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<u32> {
|
impl FfiSignedness for CTypeInfo<u32> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<u64> {
|
impl FfiSignedness for CTypeInfo<u64> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<u8> {
|
impl FfiSignedness for CTypeInfo<u8> {
|
||||||
|
|
|
@ -5,7 +5,7 @@ use num::cast::AsPrimitive;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
c::type_info::CTypeInfo,
|
c::type_info::CTypeInfo,
|
||||||
data::{FfiConvert, FfiData, FfiSignedness},
|
ffi::{FfiConvert, FfiData, FfiSignedness},
|
||||||
};
|
};
|
||||||
|
|
||||||
impl FfiSignedness for CTypeInfo<usize> {
|
impl FfiSignedness for CTypeInfo<usize> {
|
||||||
|
|
|
@ -1,10 +1,10 @@
|
||||||
use crate::ffi::bit_mask::*;
|
use crate::ffi::bit_mask::*;
|
||||||
|
|
||||||
pub enum BoxDataFlag {
|
pub enum BoxFlag {
|
||||||
Leaked,
|
Leaked,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl BoxDataFlag {
|
impl BoxFlag {
|
||||||
pub const fn value(&self) -> u8 {
|
pub const fn value(&self) -> u8 {
|
||||||
match self {
|
match self {
|
||||||
Self::Leaked => U8_MASK2,
|
Self::Leaked => U8_MASK2,
|
||||||
|
|
|
@ -3,17 +3,17 @@ use std::{alloc, alloc::Layout, boxed::Box, mem::ManuallyDrop, ptr};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
data::{association_names::REF_INNER, RefData, RefDataBounds, RefDataFlag},
|
data::{association_names::REF_INNER, RefBounds, RefData, RefFlag},
|
||||||
ffi::{association, bit_mask::*, FfiData},
|
ffi::{association, bit_mask::*, FfiData},
|
||||||
};
|
};
|
||||||
|
|
||||||
mod flag;
|
mod flag;
|
||||||
|
|
||||||
pub use self::flag::BoxDataFlag;
|
pub use self::flag::BoxFlag;
|
||||||
|
|
||||||
// Ref which created by lua should not be dereferenceable,
|
// Ref which created by lua should not be dereferenceable,
|
||||||
const BOX_REF_FLAGS: u8 =
|
const BOX_REF_FLAGS: u8 =
|
||||||
RefDataFlag::Readable.value() | RefDataFlag::Writable.value() | RefDataFlag::Offsetable.value();
|
RefFlag::Readable.value() | RefFlag::Writable.value() | RefFlag::Offsetable.value();
|
||||||
|
|
||||||
// It is an untyped, sized memory area that Lua can manage.
|
// It is an untyped, sized memory area that Lua can manage.
|
||||||
// This area is safe within Lua. Operations have their boundaries checked.
|
// This area is safe within Lua. Operations have their boundaries checked.
|
||||||
|
@ -63,7 +63,7 @@ impl BoxData {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn leak(&mut self) {
|
pub fn leak(&mut self) {
|
||||||
self.flags = u8_set(self.flags, BoxDataFlag::Leaked.value(), true);
|
self.flags = u8_set(self.flags, BoxFlag::Leaked.value(), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make FfiRef from box, with boundary checking
|
// Make FfiRef from box, with boundary checking
|
||||||
|
@ -73,7 +73,7 @@ impl BoxData {
|
||||||
offset: Option<isize>,
|
offset: Option<isize>,
|
||||||
) -> LuaResult<LuaAnyUserData<'lua>> {
|
) -> LuaResult<LuaAnyUserData<'lua>> {
|
||||||
let target = this.borrow::<BoxData>()?;
|
let target = this.borrow::<BoxData>()?;
|
||||||
let mut bounds = RefDataBounds::new(0, target.size());
|
let mut bounds = RefBounds::new(0, target.size());
|
||||||
let mut ptr = unsafe { target.get_pointer() };
|
let mut ptr = unsafe { target.get_pointer() };
|
||||||
|
|
||||||
// Calculate offset
|
// Calculate offset
|
||||||
|
@ -114,7 +114,7 @@ impl BoxData {
|
||||||
|
|
||||||
impl Drop for BoxData {
|
impl Drop for BoxData {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
if u8_test_not(self.flags, BoxDataFlag::Leaked.value()) {
|
if u8_test_not(self.flags, BoxFlag::Leaked.value()) {
|
||||||
unsafe { self.drop() };
|
unsafe { self.drop() };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,20 +7,21 @@ use libffi::{
|
||||||
};
|
};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{FfiArgInfo, FfiData, FfiResultInfo, GetFfiData};
|
use super::{FfiData, GetFfiData};
|
||||||
|
use crate::ffi::{FfiArg, FfiResult};
|
||||||
|
|
||||||
pub struct CallableData {
|
pub struct CallableData {
|
||||||
cif: *mut ffi_cif,
|
cif: *mut ffi_cif,
|
||||||
arg_info_list: *const Vec<FfiArgInfo>,
|
arg_info_list: Vec<FfiArg>,
|
||||||
result_info: *const FfiResultInfo,
|
result_info: FfiResult,
|
||||||
code: CodePtr,
|
code: CodePtr,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CallableData {
|
impl CallableData {
|
||||||
pub unsafe fn new(
|
pub unsafe fn new(
|
||||||
cif: *mut ffi_cif,
|
cif: *mut ffi_cif,
|
||||||
arg_info_list: *const Vec<FfiArgInfo>,
|
arg_info_list: Vec<FfiArg>,
|
||||||
result_info: *const FfiResultInfo,
|
result_info: FfiResult,
|
||||||
function_pointer: *const (),
|
function_pointer: *const (),
|
||||||
) -> Self {
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
|
@ -35,16 +36,15 @@ impl CallableData {
|
||||||
|
|
||||||
pub unsafe fn call(&self, result: &Ref<dyn FfiData>, args: LuaMultiValue) -> LuaResult<()> {
|
pub unsafe fn call(&self, result: &Ref<dyn FfiData>, args: LuaMultiValue) -> LuaResult<()> {
|
||||||
result
|
result
|
||||||
.check_boundary(0, self.result_info.as_ref().unwrap().size)
|
.check_boundary(0, self.result_info.size)
|
||||||
.then_some(())
|
.then_some(())
|
||||||
.ok_or_else(|| LuaError::external("result boundary check failed"))?;
|
.ok_or_else(|| LuaError::external("result boundary check failed"))?;
|
||||||
|
|
||||||
// cache Vec => unable to create async call but no allocation
|
// cache Vec => unable to create async call but no allocation
|
||||||
let arg_info_list = self.arg_info_list.as_ref().unwrap();
|
let mut arg_list = Vec::<*mut c_void>::with_capacity(self.arg_info_list.len());
|
||||||
let mut arg_list = Vec::<*mut c_void>::with_capacity(arg_info_list.len());
|
|
||||||
|
|
||||||
for index in 0..arg_info_list.len() {
|
for index in 0..self.arg_info_list.len() {
|
||||||
let arg_info = arg_info_list.get(index).unwrap();
|
let arg_info = self.arg_info_list.get(index).unwrap();
|
||||||
let arg = args
|
let arg = args
|
||||||
.get(index)
|
.get(index)
|
||||||
.ok_or_else(|| LuaError::external(format!("argument {index} required")))?;
|
.ok_or_else(|| LuaError::external(format!("argument {index} required")))?;
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use core::ffi::c_void;
|
use core::ffi::c_void;
|
||||||
use std::ptr;
|
use std::{borrow::Borrow, ptr};
|
||||||
|
|
||||||
use libffi::{
|
use libffi::{
|
||||||
low::{closure_alloc, closure_free, ffi_cif, CodePtr},
|
low::{closure_alloc, closure_free, ffi_cif, CodePtr},
|
||||||
|
@ -7,19 +7,22 @@ use libffi::{
|
||||||
};
|
};
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{
|
use super::ref_data::{RefBounds, RefData, RefFlag};
|
||||||
ref_data::{RefDataBounds, RefDataFlag},
|
use crate::ffi::{
|
||||||
RefData,
|
libffi_helper::{FFI_STATUS_NAMES, SIZE_OF_POINTER},
|
||||||
|
FfiArg, FfiData, FfiResult,
|
||||||
};
|
};
|
||||||
use crate::ffi::libffi_helper::FFI_STATUS_NAMES;
|
|
||||||
|
|
||||||
pub struct ClosureData<'a> {
|
pub struct ClosureData {
|
||||||
|
lua: *const Lua,
|
||||||
closure: *mut ffi_closure,
|
closure: *mut ffi_closure,
|
||||||
code: CodePtr,
|
code: CodePtr,
|
||||||
userdata: CallbackUserdata<'a>,
|
arg_info_list: Vec<FfiArg>,
|
||||||
|
result_info: FfiResult,
|
||||||
|
func: LuaRegistryKey,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Drop for ClosureData<'a> {
|
impl Drop for ClosureData {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe {
|
unsafe {
|
||||||
closure_free(self.closure);
|
closure_free(self.closure);
|
||||||
|
@ -27,69 +30,82 @@ impl<'a> Drop for ClosureData<'a> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unused)]
|
const RESULT_REF_FLAGS: u8 =
|
||||||
pub struct CallbackUserdata<'a> {
|
RefFlag::Leaked.value() | RefFlag::Writable.value() | RefFlag::Offsetable.value();
|
||||||
pub func: LuaFunction<'a>,
|
|
||||||
pub lua: &'a Lua,
|
|
||||||
pub arg_ref_flags: Vec<u8>,
|
|
||||||
pub arg_ref_size: Vec<usize>,
|
|
||||||
pub result_size: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
const RESULT_REF_FLAGS: u8 = RefDataFlag::Leaked.value() | RefDataFlag::Writable.value();
|
|
||||||
|
|
||||||
unsafe extern "C" fn callback(
|
unsafe extern "C" fn callback(
|
||||||
cif: *mut ffi_cif,
|
cif: *mut ffi_cif,
|
||||||
result_pointer: *mut c_void,
|
result_pointer: *mut c_void,
|
||||||
arg_pointers: *mut *mut c_void,
|
arg_pointers: *mut *mut c_void,
|
||||||
userdata: *mut c_void,
|
closure_data: *mut c_void,
|
||||||
) {
|
) {
|
||||||
let userdata = userdata.cast::<CallbackUserdata>();
|
let closure_data = closure_data.cast::<ClosureData>().as_ref().unwrap();
|
||||||
|
let lua = closure_data.lua.as_ref().unwrap();
|
||||||
let len = (*cif).nargs as usize;
|
let len = (*cif).nargs as usize;
|
||||||
let mut args = Vec::<LuaValue>::with_capacity(len + 1);
|
let mut args = Vec::<LuaValue>::with_capacity(len + 1);
|
||||||
|
|
||||||
// Push result pointer (ref)
|
// Push result pointer (ref)
|
||||||
args.push(LuaValue::UserData(
|
args.push(LuaValue::UserData(
|
||||||
(*userdata)
|
lua.create_userdata(RefData::new(
|
||||||
.lua
|
result_pointer.cast::<()>(),
|
||||||
.create_userdata(RefData::new(
|
RESULT_REF_FLAGS,
|
||||||
result_pointer.cast::<()>(),
|
RefBounds::new(0, closure_data.result_info.size),
|
||||||
RESULT_REF_FLAGS,
|
))
|
||||||
RefDataBounds::new(0, (*userdata).result_size),
|
.unwrap(),
|
||||||
))
|
|
||||||
.unwrap(),
|
|
||||||
));
|
));
|
||||||
|
|
||||||
// Push arg pointer (ref)
|
// Push arg pointer (ref)
|
||||||
for i in 0..len {
|
for i in 0..len {
|
||||||
|
let arg_info = closure_data.arg_info_list.get(i).unwrap();
|
||||||
args.push(LuaValue::UserData(
|
args.push(LuaValue::UserData(
|
||||||
(*userdata)
|
lua.create_userdata(RefData::new(
|
||||||
.lua
|
(*arg_pointers.add(i)).cast::<()>(),
|
||||||
.create_userdata(RefData::new(
|
arg_info.callback_ref_flag,
|
||||||
(*arg_pointers.add(i)).cast::<()>(),
|
RefBounds::new(0, arg_info.size),
|
||||||
(*userdata).arg_ref_flags.get(i).unwrap().to_owned(),
|
))
|
||||||
RefDataBounds::new(0, (*userdata).arg_ref_size.get(i).unwrap().to_owned()),
|
.unwrap(),
|
||||||
))
|
|
||||||
.unwrap(),
|
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
(*userdata).func.call::<_, ()>(args).unwrap();
|
closure_data
|
||||||
|
.func
|
||||||
|
.borrow()
|
||||||
|
.into_lua(lua)
|
||||||
|
.unwrap()
|
||||||
|
.as_function()
|
||||||
|
.unwrap()
|
||||||
|
.call::<_, ()>(args)
|
||||||
|
.unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> ClosureData<'a> {
|
impl ClosureData {
|
||||||
pub unsafe fn new(
|
pub fn new(
|
||||||
|
lua: *const Lua,
|
||||||
cif: *mut ffi_cif,
|
cif: *mut ffi_cif,
|
||||||
userdata: CallbackUserdata<'a>,
|
arg_info_list: Vec<FfiArg>,
|
||||||
) -> LuaResult<ClosureData<'a>> {
|
result_info: FfiResult,
|
||||||
|
func: LuaRegistryKey,
|
||||||
|
) -> LuaResult<ClosureData> {
|
||||||
let (closure, code) = closure_alloc();
|
let (closure, code) = closure_alloc();
|
||||||
let prep_result = ffi_prep_closure_loc(
|
|
||||||
|
let closure_data = ClosureData {
|
||||||
|
lua,
|
||||||
closure,
|
closure,
|
||||||
cif,
|
code,
|
||||||
Some(callback),
|
arg_info_list,
|
||||||
ptr::from_ref(&userdata).cast::<c_void>().cast_mut(),
|
result_info,
|
||||||
code.as_mut_ptr(),
|
func,
|
||||||
);
|
};
|
||||||
|
|
||||||
|
let prep_result = unsafe {
|
||||||
|
ffi_prep_closure_loc(
|
||||||
|
closure,
|
||||||
|
cif,
|
||||||
|
Some(callback),
|
||||||
|
ptr::from_ref(&closure_data).cast::<c_void>().cast_mut(),
|
||||||
|
code.as_mut_ptr(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
if prep_result != 0 {
|
if prep_result != 0 {
|
||||||
Err(LuaError::external(format!(
|
Err(LuaError::external(format!(
|
||||||
|
@ -97,11 +113,24 @@ impl<'a> ClosureData<'a> {
|
||||||
FFI_STATUS_NAMES[0], FFI_STATUS_NAMES[prep_result as usize]
|
FFI_STATUS_NAMES[0], FFI_STATUS_NAMES[prep_result as usize]
|
||||||
)))
|
)))
|
||||||
} else {
|
} else {
|
||||||
Ok(ClosureData {
|
Ok(closure_data)
|
||||||
closure,
|
|
||||||
code,
|
|
||||||
userdata,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FfiData for ClosureData {
|
||||||
|
unsafe fn get_pointer(&self) -> *mut () {
|
||||||
|
self.code.as_mut_ptr().cast::<()>()
|
||||||
|
}
|
||||||
|
fn check_boundary(&self, offset: isize, size: usize) -> bool {
|
||||||
|
(offset as usize) + size <= SIZE_OF_POINTER
|
||||||
|
}
|
||||||
|
fn is_readable(&self) -> bool {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
fn is_writable(&self) -> bool {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LuaUserData for ClosureData {}
|
||||||
|
|
|
@ -2,15 +2,15 @@ use dlopen2::raw::Library;
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
association,
|
|
||||||
association_names::SYM_INNER,
|
association_names::SYM_INNER,
|
||||||
ref_data::{RefData, RefDataFlag, UNSIZED_BOUNDS},
|
ref_data::{RefData, RefFlag, UNSIZED_BOUNDS},
|
||||||
};
|
};
|
||||||
|
use crate::ffi::association;
|
||||||
|
|
||||||
const LIB_REF_FLAGS: u8 = RefDataFlag::Offsetable.value()
|
const LIB_REF_FLAGS: u8 = RefFlag::Offsetable.value()
|
||||||
| RefDataFlag::Readable.value()
|
| RefFlag::Readable.value()
|
||||||
| RefDataFlag::Dereferenceable.value()
|
| RefFlag::Dereferenceable.value()
|
||||||
| RefDataFlag::Function.value();
|
| RefFlag::Function.value();
|
||||||
|
|
||||||
pub struct LibData(Library);
|
pub struct LibData(Library);
|
||||||
|
|
||||||
|
|
|
@ -9,19 +9,14 @@ mod closure_data;
|
||||||
mod lib_data;
|
mod lib_data;
|
||||||
mod ref_data;
|
mod ref_data;
|
||||||
|
|
||||||
pub use crate::{
|
pub use self::{
|
||||||
data::{
|
box_data::BoxData,
|
||||||
box_data::BoxData,
|
callable_data::CallableData,
|
||||||
callable_data::CallableData,
|
closure_data::ClosureData,
|
||||||
closure_data::ClosureData,
|
lib_data::LibData,
|
||||||
lib_data::LibData,
|
ref_data::{create_nullptr, RefBounds, RefData, RefFlag},
|
||||||
ref_data::{create_nullptr, RefData, RefDataBounds, RefDataFlag},
|
|
||||||
},
|
|
||||||
ffi::{
|
|
||||||
association, num_cast, FfiArgInfo, FfiConvert, FfiData, FfiResultInfo, FfiSignedness,
|
|
||||||
FfiSize,
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
use crate::ffi::FfiData;
|
||||||
|
|
||||||
// Named registry table names
|
// Named registry table names
|
||||||
mod association_names {
|
mod association_names {
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
// Memory range for ref or box data. For boundary checking
|
// Memory range for ref or box data. For boundary checking
|
||||||
pub struct RefDataBounds {
|
pub struct RefBounds {
|
||||||
// Indicates how much data is above the pointer
|
// Indicates how much data is above the pointer
|
||||||
pub(crate) above: usize,
|
pub(crate) above: usize,
|
||||||
// Indicates how much data is below the pointer
|
// Indicates how much data is below the pointer
|
||||||
pub(crate) below: usize,
|
pub(crate) below: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const UNSIZED_BOUNDS: RefDataBounds = RefDataBounds {
|
pub const UNSIZED_BOUNDS: RefBounds = RefBounds {
|
||||||
above: usize::MAX,
|
above: usize::MAX,
|
||||||
below: usize::MAX,
|
below: usize::MAX,
|
||||||
};
|
};
|
||||||
|
|
||||||
impl RefDataBounds {
|
impl RefBounds {
|
||||||
pub fn new(above: usize, below: usize) -> Self {
|
pub fn new(above: usize, below: usize) -> Self {
|
||||||
Self { above, below }
|
Self { above, below }
|
||||||
}
|
}
|
||||||
|
@ -88,7 +88,7 @@ impl RefDataBounds {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Clone for RefDataBounds {
|
impl Clone for RefBounds {
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
Self {
|
Self {
|
||||||
above: self.above,
|
above: self.above,
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use crate::ffi::bit_mask::*;
|
use crate::ffi::bit_mask::*;
|
||||||
|
|
||||||
pub enum RefDataFlag {
|
pub enum RefFlag {
|
||||||
Leaked,
|
Leaked,
|
||||||
Dereferenceable,
|
Dereferenceable,
|
||||||
Readable,
|
Readable,
|
||||||
|
@ -9,7 +9,7 @@ pub enum RefDataFlag {
|
||||||
Function,
|
Function,
|
||||||
Uninit,
|
Uninit,
|
||||||
}
|
}
|
||||||
impl RefDataFlag {
|
impl RefFlag {
|
||||||
pub const fn value(&self) -> u8 {
|
pub const fn value(&self) -> u8 {
|
||||||
match self {
|
match self {
|
||||||
Self::Leaked => U8_MASK1,
|
Self::Leaked => U8_MASK1,
|
||||||
|
|
|
@ -11,13 +11,13 @@ mod bounds;
|
||||||
mod flag;
|
mod flag;
|
||||||
|
|
||||||
pub use self::{
|
pub use self::{
|
||||||
bounds::{RefDataBounds, UNSIZED_BOUNDS},
|
bounds::{RefBounds, UNSIZED_BOUNDS},
|
||||||
flag::RefDataFlag,
|
flag::RefFlag,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Box:ref():ref() should not be able to modify, Only for external
|
// Box:ref():ref() should not be able to modify, Only for external
|
||||||
const BOX_REF_REF_FLAGS: u8 = 0;
|
const BOX_REF_REF_FLAGS: u8 = 0;
|
||||||
const UNINIT_REF_FLAGS: u8 = RefDataFlag::Uninit.value();
|
const UNINIT_REF_FLAGS: u8 = RefFlag::Uninit.value();
|
||||||
// | FfiRefFlag::Writable.value()
|
// | FfiRefFlag::Writable.value()
|
||||||
// | FfiRefFlag::Readable.value()
|
// | FfiRefFlag::Readable.value()
|
||||||
// | FfiRefFlag::Dereferenceable.value()
|
// | FfiRefFlag::Dereferenceable.value()
|
||||||
|
@ -33,11 +33,11 @@ const UNINIT_REF_FLAGS: u8 = RefDataFlag::Uninit.value();
|
||||||
pub struct RefData {
|
pub struct RefData {
|
||||||
ptr: ManuallyDrop<Box<*mut ()>>,
|
ptr: ManuallyDrop<Box<*mut ()>>,
|
||||||
pub flags: u8,
|
pub flags: u8,
|
||||||
pub boundary: RefDataBounds,
|
pub boundary: RefBounds,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RefData {
|
impl RefData {
|
||||||
pub fn new(ptr: *mut (), flags: u8, boundary: RefDataBounds) -> Self {
|
pub fn new(ptr: *mut (), flags: u8, boundary: RefBounds) -> Self {
|
||||||
Self {
|
Self {
|
||||||
ptr: ManuallyDrop::new(Box::new(ptr)),
|
ptr: ManuallyDrop::new(Box::new(ptr)),
|
||||||
flags,
|
flags,
|
||||||
|
@ -63,7 +63,7 @@ impl RefData {
|
||||||
let luaref = lua.create_userdata(RefData::new(
|
let luaref = lua.create_userdata(RefData::new(
|
||||||
ptr::from_ref(&target.ptr) as *mut (),
|
ptr::from_ref(&target.ptr) as *mut (),
|
||||||
BOX_REF_REF_FLAGS,
|
BOX_REF_REF_FLAGS,
|
||||||
RefDataBounds {
|
RefBounds {
|
||||||
below: 0,
|
below: 0,
|
||||||
above: size_of::<usize>(),
|
above: size_of::<usize>(),
|
||||||
},
|
},
|
||||||
|
@ -76,7 +76,7 @@ impl RefData {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub unsafe fn deref(&self) -> LuaResult<Self> {
|
pub unsafe fn deref(&self) -> LuaResult<Self> {
|
||||||
u8_test(self.flags, RefDataFlag::Dereferenceable.value())
|
u8_test(self.flags, RefFlag::Dereferenceable.value())
|
||||||
.then_some(())
|
.then_some(())
|
||||||
.ok_or_else(|| LuaError::external("This pointer is not dereferenceable."))?;
|
.ok_or_else(|| LuaError::external("This pointer is not dereferenceable."))?;
|
||||||
|
|
||||||
|
@ -104,7 +104,7 @@ impl RefData {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub unsafe fn offset(&self, offset: isize) -> LuaResult<Self> {
|
pub unsafe fn offset(&self, offset: isize) -> LuaResult<Self> {
|
||||||
u8_test(self.flags, RefDataFlag::Offsetable.value())
|
u8_test(self.flags, RefFlag::Offsetable.value())
|
||||||
.then_some(())
|
.then_some(())
|
||||||
.ok_or_else(|| LuaError::external("This pointer is not offsetable."))?;
|
.ok_or_else(|| LuaError::external("This pointer is not offsetable."))?;
|
||||||
|
|
||||||
|
@ -132,7 +132,7 @@ impl RefData {
|
||||||
|
|
||||||
impl Drop for RefData {
|
impl Drop for RefData {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
if u8_test_not(self.flags, RefDataFlag::Leaked.value()) {
|
if u8_test_not(self.flags, RefFlag::Leaked.value()) {
|
||||||
unsafe { ManuallyDrop::drop(&mut self.ptr) };
|
unsafe { ManuallyDrop::drop(&mut self.ptr) };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -146,10 +146,10 @@ impl FfiData for RefData {
|
||||||
**self.ptr
|
**self.ptr
|
||||||
}
|
}
|
||||||
fn is_readable(&self) -> bool {
|
fn is_readable(&self) -> bool {
|
||||||
u8_test(self.flags, RefDataFlag::Readable.value())
|
u8_test(self.flags, RefFlag::Readable.value())
|
||||||
}
|
}
|
||||||
fn is_writable(&self) -> bool {
|
fn is_writable(&self) -> bool {
|
||||||
u8_test(self.flags, RefDataFlag::Writable.value())
|
u8_test(self.flags, RefFlag::Writable.value())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,16 +0,0 @@
|
||||||
use super::FfiConvert;
|
|
||||||
|
|
||||||
pub struct FfiArgRefOption {
|
|
||||||
pub flag: u8,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum FfiArgType {
|
|
||||||
FfiBox,
|
|
||||||
FfiRef(FfiArgRefOption),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct FfiArgInfo {
|
|
||||||
pub conv: *const dyn FfiConvert,
|
|
||||||
pub size: usize,
|
|
||||||
// pub kind: NativeArgType,
|
|
||||||
}
|
|
|
@ -26,7 +26,7 @@ pub fn get_ensured_size(ffi_type: *mut raw::ffi_type) -> LuaResult<usize> {
|
||||||
unsafe { Ok((*ffi_type).size) }
|
unsafe { Ok((*ffi_type).size) }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const SIEE_OF_POINTER: usize = size_of::<*mut ()>();
|
pub const SIZE_OF_POINTER: usize = size_of::<*mut ()>();
|
||||||
|
|
||||||
// Converts ffi status into &str
|
// Converts ffi status into &str
|
||||||
pub const FFI_STATUS_NAMES: [&str; 4] = [
|
pub const FFI_STATUS_NAMES: [&str; 4] = [
|
||||||
|
|
|
@ -2,12 +2,12 @@ use std::cell::Ref;
|
||||||
|
|
||||||
use mlua::prelude::*;
|
use mlua::prelude::*;
|
||||||
|
|
||||||
mod arg;
|
|
||||||
pub mod association;
|
pub mod association;
|
||||||
pub mod bit_mask;
|
pub mod bit_mask;
|
||||||
mod cast;
|
mod cast;
|
||||||
pub mod libffi_helper;
|
pub mod libffi_helper;
|
||||||
mod result;
|
|
||||||
|
pub use self::cast::num_cast;
|
||||||
|
|
||||||
pub trait FfiSize {
|
pub trait FfiSize {
|
||||||
fn get_size(&self) -> usize;
|
fn get_size(&self) -> usize;
|
||||||
|
@ -46,4 +46,26 @@ pub trait FfiData {
|
||||||
fn is_readable(&self) -> bool;
|
fn is_readable(&self) -> bool;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub use self::{arg::FfiArgInfo, cast::num_cast, result::FfiResultInfo};
|
pub struct FfiArg {
|
||||||
|
pub size: usize,
|
||||||
|
pub callback_ref_flag: u8,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Clone for FfiArg {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
Self {
|
||||||
|
size: self.size,
|
||||||
|
callback_ref_flag: self.callback_ref_flag,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct FfiResult {
|
||||||
|
pub size: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Clone for FfiResult {
|
||||||
|
fn clone(&self) -> Self {
|
||||||
|
Self { size: self.size }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -1,12 +0,0 @@
|
||||||
use super::FfiConvert;
|
|
||||||
|
|
||||||
// pub enum NativeResultType {
|
|
||||||
// FfiBox,
|
|
||||||
// FfiRef,
|
|
||||||
// }
|
|
||||||
|
|
||||||
pub struct FfiResultInfo {
|
|
||||||
pub conv: *const dyn FfiConvert,
|
|
||||||
pub size: usize,
|
|
||||||
// kind: NativeResultType,
|
|
||||||
}
|
|
|
@ -32,7 +32,7 @@ pub fn module(lua: &Lua) -> LuaResult<LuaTable> {
|
||||||
.with_function("uninitRef", |_lua, ()| Ok(RefData::new_uninit()))?
|
.with_function("uninitRef", |_lua, ()| Ok(RefData::new_uninit()))?
|
||||||
.with_function("isInteger", |_lua, num: LuaValue| Ok(num.is_integer()))?
|
.with_function("isInteger", |_lua, num: LuaValue| Ok(num.is_integer()))?
|
||||||
.with_function("fnInfo", |lua, (args, ret): (LuaTable, LuaAnyUserData)| {
|
.with_function("fnInfo", |lua, (args, ret): (LuaTable, LuaAnyUserData)| {
|
||||||
CFnInfo::new_from_table(lua, args, ret)
|
CFnInfo::from_table(lua, args, ret)
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
#[cfg(debug_assertions)]
|
#[cfg(debug_assertions)]
|
||||||
|
|
Loading…
Add table
Reference in a new issue