Implement call (#243)

This commit is contained in:
qwreey 2024-10-14 09:37:21 +00:00
parent 7ce5be248f
commit 46dd185c6f
No known key found for this signature in database
GPG key ID: D28DB79297A214BD
23 changed files with 271 additions and 113 deletions

5
.gitignore vendored
View file

@ -22,6 +22,11 @@ luneDocs.json
luneTypes.d.luau luneTypes.d.luau
# Files generated by runtime or build scripts # Files generated by runtime or build scripts
scripts/brick_color.rs scripts/brick_color.rs
scripts/font_enum_map.rs scripts/font_enum_map.rs
scripts/physical_properties_enum_map.rs scripts/physical_properties_enum_map.rs
# Files generated by tests
/tests/ffi/**/*.so

View file

@ -5,13 +5,14 @@ use mlua::prelude::*;
use super::{ use super::{
association_names::CARR_INNER, association_names::CARR_INNER,
c_helper::{get_conv, get_ensured_size, libffi_type_from_userdata, pretty_format_userdata}, c_helper::{get_conv, libffi_type_from_userdata, pretty_format_userdata},
CPtr, CPtr,
}; };
use crate::ffi::{ use crate::ffi::{
ffi_association::{get_association, set_association}, ffi_association::{get_association, set_association},
FfiBox, GetNativeData, NativeConvert, NativeData, NativeSize, FfiBox, GetNativeData, NativeConvert, NativeData, NativeSize,
}; };
use crate::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,

View file

@ -1,20 +1,17 @@
use std::ptr; use std::ptr;
use libffi::low::ffi_cif;
use libffi::middle::{Cif, Type}; use libffi::middle::{Cif, Type};
use mlua::prelude::*; use mlua::prelude::*;
use super::c_helper::{get_size, get_userdata};
use super::{ use super::{
association_names::{CALLABLE_CFN, CALLABLE_REF, CFN_ARGS, CFN_RESULT}, association_names::{CALLABLE_CFN, CALLABLE_REF, CFN_ARGS, CFN_RESULT},
c_helper::{ c_helper::{get_conv, libffi_type_from_userdata, libffi_type_list_from_table},
get_conv, get_conv_list_from_table, libffi_type_from_userdata, libffi_type_list_from_table,
},
}; };
use crate::ffi::bit_mask::u8_test_not;
use crate::ffi::{ use crate::ffi::{
ffi_association::set_association, FfiClosure, NativeArgInfo, NativeConvert, NativeResultInfo, bit_mask::u8_test_not, ffi_association::set_association, FfiCallable, FfiRef, FfiRefFlag,
NativeArgInfo, NativeData, NativeResultInfo,
}; };
use crate::ffi::{FfiCallable, FfiRef, FfiRefFlag, NativeData};
// cfn is a type declaration for a function. // cfn is a type declaration for a function.
// Basically, when calling an external function, this type declaration // Basically, when calling an external function, this type declaration
@ -33,7 +30,7 @@ use crate::ffi::{FfiCallable, FfiRef, FfiRefFlag, NativeData};
// moved to a Lua function or vice versa. // moved to a Lua function or vice versa.
pub struct CFn { pub struct CFn {
cif: *mut ffi_cif, cif: Cif,
arg_info_list: Vec<NativeArgInfo>, arg_info_list: Vec<NativeArgInfo>,
result_info: NativeResultInfo, result_info: NativeResultInfo,
} }
@ -46,37 +43,44 @@ impl CFn {
ret: Type, ret: Type,
arg_info_list: Vec<NativeArgInfo>, arg_info_list: Vec<NativeArgInfo>,
result_info: NativeResultInfo, result_info: NativeResultInfo,
) -> Self { ) -> LuaResult<Self> {
Self { // let cif = ;
cif: Cif::new(args.clone(), ret.clone()).as_raw_ptr(),
Ok(Self {
cif: Cif::new(args.clone(), ret.clone()),
arg_info_list, arg_info_list,
result_info, result_info,
} })
} }
pub fn new_from_lua_table<'lua>( pub fn new_from_lua_table<'lua>(
lua: &'lua Lua, lua: &'lua Lua,
args: LuaTable, arg_table: LuaTable,
ret: LuaAnyUserData, ret: LuaAnyUserData,
) -> LuaResult<LuaAnyUserData<'lua>> { ) -> LuaResult<LuaAnyUserData<'lua>> {
let args_types = libffi_type_list_from_table(lua, &args)?; let args_types = libffi_type_list_from_table(lua, &arg_table)?;
let ret_type = libffi_type_from_userdata(lua, &ret)?; let ret_type = libffi_type_from_userdata(lua, &ret)?;
let mut arg_info_list = Vec::<NativeArgInfo>::with_capacity(args.raw_len()); let arg_len = arg_table.raw_len();
for conv in unsafe { get_conv_list_from_table(&args)? } { let mut arg_info_list = Vec::<NativeArgInfo>::with_capacity(arg_len);
arg_info_list.push(NativeArgInfo { conv }) for index in 0..arg_len {
let userdata = get_userdata(arg_table.raw_get(index + 1)?)?;
arg_info_list.push(NativeArgInfo {
conv: unsafe { get_conv(&userdata)? },
size: get_size(&userdata)?,
});
} }
let result_info = NativeResultInfo { let result_info = NativeResultInfo {
conv: unsafe { get_conv(&ret)? }, conv: unsafe { get_conv(&ret)? },
size: get_size(&ret)?,
}; };
let cfn = let cfn =
lua.create_userdata(Self::new(args_types, ret_type, arg_info_list, result_info))?; lua.create_userdata(Self::new(args_types, ret_type, arg_info_list, result_info)?)?;
// Create association to hold argument and result type // Create association to hold argument and result type
set_association(lua, CFN_ARGS, &cfn, args)?; set_association(lua, CFN_ARGS, &cfn, arg_table)?;
set_association(lua, CFN_ARGS, &cfn, ret)?; set_association(lua, CFN_RESULT, &cfn, ret)?;
Ok(cfn) Ok(cfn)
} }
@ -88,22 +92,22 @@ impl LuaUserData for CFn {
// lua.create_userdata(FfiClosure::new(this.cif, userdata)) // lua.create_userdata(FfiClosure::new(this.cif, userdata))
// }) // })
methods.add_function( methods.add_function(
"func", "caller",
|lua, (cfn, function_ref): (LuaAnyUserData, LuaAnyUserData)| { |lua, (cfn, function_ref): (LuaAnyUserData, LuaAnyUserData)| {
let this = cfn.borrow::<CFn>()?; let this = cfn.borrow::<CFn>()?;
if !function_ref.is::<FfiRef>() { if !function_ref.is::<FfiRef>() {
return Err(LuaError::external("")); return Err(LuaError::external("argument 0 must be ffiref"));
} }
let ffi_ref = function_ref.borrow::<FfiRef>()?; let ffi_ref = function_ref.borrow::<FfiRef>()?;
if u8_test_not(ffi_ref.flags, FfiRefFlag::Function.value()) { if u8_test_not(ffi_ref.flags, FfiRefFlag::Function.value()) {
return Err(LuaError::external("")); return Err(LuaError::external("not a function ref"));
} }
let callable = lua.create_userdata(unsafe { let callable = lua.create_userdata(unsafe {
FfiCallable::new( FfiCallable::new(
this.cif, this.cif.as_raw_ptr(),
ptr::from_ref(&this.arg_info_list), ptr::from_ref(&this.arg_info_list),
ptr::from_ref(&this.result_info), ptr::from_ref(&this.result_info),
ffi_ref.get_pointer(0), ffi_ref.get_pointer(0),

View file

@ -1,17 +1,26 @@
#![allow(clippy::inline_always)] #![allow(clippy::inline_always)]
use std::ptr::{self, null_mut}; use libffi::middle::Type;
use libffi::{low, middle::Type, raw};
use lune_utils::fmt::{pretty_format_value, ValueFormatConfig}; use lune_utils::fmt::{pretty_format_value, ValueFormatConfig};
use mlua::prelude::*; use mlua::prelude::*;
use super::{ use super::{
association_names::CTYPE_STATIC, types::get_ctype_conv, CArr, CPtr, CStruct, CTypeStatic, association_names::CTYPE_STATIC,
}; types::{get_ctype_conv, get_ctype_size},
use crate::ffi::{ CArr, CPtr, CStruct, CTypeStatic,
ffi_association::get_association, NativeConvert, NativeSignedness, NativeSize, FFI_STATUS_NAMES,
}; };
use crate::ffi::{ffi_association::get_association, NativeConvert, NativeSize};
pub fn get_userdata(value: LuaValue) -> LuaResult<LuaAnyUserData> {
if let LuaValue::UserData(field_type) = value {
Ok(field_type)
} else {
Err(LuaError::external(format!(
"Unexpected field. CStruct, CType or CArr is required for element but got {}",
pretty_format_value(&value, &ValueFormatConfig::new())
)))
}
}
// Get the NativeConvert handle from the type UserData // Get the NativeConvert handle from the type UserData
// this is intended to avoid lookup userdata and lua table every time. (eg: struct) // this is intended to avoid lookup userdata and lua table every time. (eg: struct)
@ -33,30 +42,21 @@ pub unsafe fn get_conv_list_from_table(
for i in 0..len { for i in 0..len {
let value: LuaValue = table.raw_get(i + 1)?; let value: LuaValue = table.raw_get(i + 1)?;
conv_list.push(get_conv(&get_userdata(value)?)?);
if let LuaValue::UserData(field_type) = value {
conv_list.push(get_conv(&field_type)?);
} else {
return Err(LuaError::external(format!(
"Unexpected field. CStruct, CType or CArr is required for element but got {}",
pretty_format_value(&value, &ValueFormatConfig::new())
)));
}
} }
Ok(conv_list) Ok(conv_list)
} }
// #[inline(always)] pub fn get_size(this: &LuaAnyUserData) -> LuaResult<usize> {
// pub fn type_size_from_userdata(this: &LuaAnyUserData) -> LuaResult<usize> { if this.is::<CStruct>() {
// if this.is::<CStruct>() { Ok(this.borrow::<CStruct>()?.get_size())
// Ok(this.borrow::<CStruct>()?.get_size()) } else if this.is::<CArr>() {
// } else if this.is::<CArr>() { Ok(this.borrow::<CArr>()?.get_size())
// Ok(this.borrow::<CArr>()?.get_size()) } else {
// } else { get_ctype_size(this)
// ctype_size_from_userdata(this) }
// } }
// }
// get Vec<libffi_type> from table(array) of c-type userdata // get Vec<libffi_type> from table(array) of c-type userdata
pub fn libffi_type_list_from_table(lua: &Lua, table: &LuaTable) -> LuaResult<Vec<Type>> { pub fn libffi_type_list_from_table(lua: &Lua, table: &LuaTable) -> LuaResult<Vec<Type>> {
@ -162,26 +162,3 @@ pub fn pretty_format_userdata(lua: &Lua, userdata: &LuaAnyUserData) -> LuaResult
)) ))
} }
} }
// Ensure sizeof c-type (raw::libffi_type)
// See: http://www.chiark.greenend.org.uk/doc/libffi-dev/html/Size-and-Alignment.html
pub fn get_ensured_size(ffi_type: *mut raw::ffi_type) -> LuaResult<usize> {
let mut cif = low::ffi_cif::default();
let result = unsafe {
raw::ffi_prep_cif(
ptr::from_mut(&mut cif),
raw::ffi_abi_FFI_DEFAULT_ABI,
0,
ffi_type,
null_mut(),
)
};
if result != raw::ffi_status_FFI_OK {
return Err(LuaError::external(format!(
"ffi_get_struct_offsets failed. expected result {}, got {}",
FFI_STATUS_NAMES[0], FFI_STATUS_NAMES[result as usize]
)));
}
unsafe { Ok((*ffi_type).size) }
}

View file

@ -4,3 +4,5 @@
// but separated it for clarity. // but separated it for clarity.
// This also allows operations such as ffi.string:intoBox(). // This also allows operations such as ffi.string:intoBox().
// (Write a string to an already existing box) // (Write a string to an already existing box)
// FIXME: use buffer instead?

View file

@ -7,11 +7,12 @@ use lune_utils::fmt::{pretty_format_value, ValueFormatConfig};
use mlua::prelude::*; use mlua::prelude::*;
use num::cast::AsPrimitive; use num::cast::AsPrimitive;
use super::{association_names::CTYPE_STATIC, c_helper::get_ensured_size, CArr, CPtr}; use super::{association_names::CTYPE_STATIC, CArr, CPtr};
use crate::ffi::{ use crate::ffi::{
ffi_association::set_association, native_num_cast, FfiBox, GetNativeData, NativeConvert, ffi_association::set_association, native_num_cast, FfiBox, GetNativeData, NativeConvert,
NativeData, NativeSignedness, NativeSize, NativeData, NativeSignedness, NativeSize,
}; };
use crate::libffi_helper::get_ensured_size;
// We can't get a CType<T> through mlua, something like // We can't get a CType<T> through mlua, something like
// .is::<CType<dyn Any>> will fail. // .is::<CType<dyn Any>> will fail.
@ -106,11 +107,9 @@ where
libffi_type: Type, libffi_type: Type,
name: Option<&'static str>, name: Option<&'static str>,
) -> LuaResult<LuaAnyUserData<'lua>> { ) -> LuaResult<LuaAnyUserData<'lua>> {
// let libffi_cfi = Cif::new(vec![libffi_type.clone()], Type::void());
let size = get_ensured_size(libffi_type.as_raw_ptr())?; let size = get_ensured_size(libffi_type.as_raw_ptr())?;
let ctype = Self { let ctype = Self {
// libffi_cif: libffi_cfi,
libffi_type, libffi_type,
size, size,
name, name,

View file

@ -8,7 +8,7 @@ use mlua::prelude::*;
use num::cast::AsPrimitive; use num::cast::AsPrimitive;
use super::{CType, CTypeCast}; use super::{CType, CTypeCast};
use crate::ffi::{NativeConvert, NativeData}; use crate::ffi::{NativeConvert, NativeData, NativeSize};
pub mod f32; pub mod f32;
pub mod f64; pub mod f64;
@ -145,3 +145,18 @@ macro_rules! define_get_ctype_conv {
pub unsafe fn get_ctype_conv(userdata: &LuaAnyUserData) -> LuaResult<*const dyn NativeConvert> { pub unsafe fn get_ctype_conv(userdata: &LuaAnyUserData) -> LuaResult<*const dyn NativeConvert> {
define_get_ctype_conv!(userdata, u8, u16, u32, u64, u128, i8, i16, i32, i64, i128, f32, f64) define_get_ctype_conv!(userdata, u8, u16, u32, u64, u128, i8, i16, i32, i64, i128, f32, f64)
} }
macro_rules! define_get_ctype_size {
($userdata:ident, $f:ty, $( $c:ty ),*) => {
if $userdata.is::<CType<$f>>() {
Ok($userdata.borrow::<CType<$f>>()?.get_size())
}$( else if $userdata.is::<CType<$c>>() {
Ok($userdata.borrow::<CType<$c>>()?.get_size())
})* else {
Err(LuaError::external("Unexpected type"))
}
};
}
pub fn get_ctype_size(userdata: &LuaAnyUserData) -> LuaResult<usize> {
define_get_ctype_size!(userdata, u8, u16, u32, u64, u128, i8, i16, i32, i64, i128, f32, f64)
}

View file

@ -128,7 +128,7 @@ impl NativeData for FfiBox {
if offset < 0 { if offset < 0 {
return false; return false;
} }
self.size() > ((offset as usize) + size) self.size() - (offset as usize) >= size
} }
unsafe fn get_pointer(&self, offset: isize) -> *mut () { unsafe fn get_pointer(&self, offset: isize) -> *mut () {
self.data self.data

View file

@ -1,12 +1,9 @@
use core::ffi::c_void; use core::ffi::c_void;
use std::cell::Ref; use std::cell::Ref;
// use std::ptr;
use libffi::{ use libffi::{
// low::{closure_alloc, ffi_cif, CodePtr, RawCallback},
low::{ffi_cif, CodePtr}, low::{ffi_cif, CodePtr},
// middle::Cif, raw::ffi_call,
// raw::ffi_prep_closure_loc,
}; };
use mlua::prelude::*; use mlua::prelude::*;
@ -14,37 +11,64 @@ use super::{GetNativeData, NativeArgInfo, NativeData, NativeResultInfo};
pub struct FfiCallable { pub struct FfiCallable {
cif: *mut ffi_cif, cif: *mut ffi_cif,
arg_info: *const Vec<NativeArgInfo>, arg_info_list: *const Vec<NativeArgInfo>,
result_info: *const NativeResultInfo, result_info: *const NativeResultInfo,
code: CodePtr, code: CodePtr,
// Caching for better performance
result_size: usize,
} }
impl FfiCallable { impl FfiCallable {
pub unsafe fn new( pub unsafe fn new(
cif: *mut ffi_cif, cif: *mut ffi_cif,
arg_info: *const Vec<NativeArgInfo>, arg_info_list: *const Vec<NativeArgInfo>,
result_info: *const NativeResultInfo, result_info: *const NativeResultInfo,
function_pointer: *const (), function_pointer: *const (),
) -> Self { ) -> Self {
let result_size = (*(*result_info).conv).get_size();
Self { Self {
cif, cif,
arg_info, arg_info_list,
result_info, result_info,
code: CodePtr::from_ptr(function_pointer.cast::<c_void>()), code: CodePtr::from_ptr(function_pointer.cast::<c_void>()),
result_size,
} }
} }
pub unsafe fn call(&self, result: &Ref<dyn NativeData>, args: LuaMultiValue) -> LuaResult<()> { pub unsafe fn call(&self, result: &Ref<dyn NativeData>, args: LuaMultiValue) -> LuaResult<()> {
result result
.check_boundary(0, self.result_size) .check_boundary(0, self.result_info.as_ref().unwrap().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
let arg_info_list = self.arg_info_list.as_ref().unwrap();
let mut arg_list = Vec::<*mut c_void>::with_capacity(arg_info_list.len());
for index in 0..arg_info_list.len() {
let arg_info = arg_info_list.get(index).unwrap();
let arg = args
.get(index)
.ok_or_else(|| LuaError::external(format!("argument {index} required")))?;
let arg_pointer = if let LuaValue::UserData(userdata) = arg {
let data_handle = userdata.get_data_handle()?;
data_handle
.check_boundary(0, arg_info.size)
.then_some(())
.ok_or_else(|| {
LuaError::external(format!("argument {index} boundary check failed"))
})?;
data_handle.get_pointer(0)
} else {
return Err(LuaError::external("unimpl"));
};
arg_list.push(arg_pointer.cast::<c_void>());
}
ffi_call(
self.cif,
Some(*self.code.as_safe_fun()),
result.get_pointer(0).cast::<c_void>(),
arg_list.as_mut_ptr(),
);
Ok(())
} }
} }
@ -63,5 +87,6 @@ impl LuaUserData for FfiCallable {
unsafe { this.call(&result.clone().get_data_handle()?, args) } unsafe { this.call(&result.clone().get_data_handle()?, args) }
}, },
); );
// ref, leak ..?
} }
} }

View file

@ -1,6 +1,6 @@
use core::ffi::c_void; use core::ffi::c_void;
use dlopen2::symbor::Library; use dlopen2::raw::Library;
use mlua::prelude::*; use mlua::prelude::*;
use super::{ use super::{
@ -41,12 +41,17 @@ impl FfiLib {
let lib = this.borrow::<FfiLib>()?; let lib = this.borrow::<FfiLib>()?;
let sym = unsafe { let sym = unsafe {
lib.0 lib.0
.symbol::<*mut c_void>(name.as_str()) .symbol::<*const c_void>(name.as_str())
.map_err(|err| LuaError::external(format!("{err}")))? .map_err(|err| LuaError::external(format!("{err}")))?
}; };
let ptr = sym.cast::<()>().cast_mut();
let luasym = // unsafe {
lua.create_userdata(FfiRef::new((*sym).cast(), LIB_REF_FLAGS, UNSIZED_BOUNDS))?; // let f = transmute::<*mut (), unsafe extern "C" fn(i32, i32) -> i32>(ptr);
// dbg!(f(1, 2));
// }
let luasym = lua.create_userdata(FfiRef::new(ptr, LIB_REF_FLAGS, UNSIZED_BOUNDS))?;
set_association(lua, SYM_INNER, &luasym, &this)?; set_association(lua, SYM_INNER, &luasym, &this)?;
@ -57,8 +62,7 @@ impl FfiLib {
impl LuaUserData for FfiLib { impl LuaUserData for FfiLib {
fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) { fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) {
methods.add_function("find", |lua, (this, name): (LuaAnyUserData, String)| { methods.add_function("find", |lua, (this, name): (LuaAnyUserData, String)| {
let luasym = FfiLib::get_sym(lua, this, name)?; FfiLib::get_sym(lua, this, name)
Ok(luasym)
}); });
} }
} }

View file

@ -11,5 +11,6 @@ pub enum NativeArgType {
pub struct NativeArgInfo { pub struct NativeArgInfo {
pub conv: *const dyn NativeConvert, pub conv: *const dyn NativeConvert,
pub size: usize,
// pub kind: NativeArgType, // pub kind: NativeArgType,
} }

View file

@ -4,13 +4,10 @@ use std::cell::Ref;
use mlua::prelude::*; use mlua::prelude::*;
use super::{NativeData, NativeSize}; use super::NativeData;
// Handle native data, provide type conversion between luavalue and native types // Handle native data, provide type conversion between luavalue and native types
pub trait NativeConvert pub trait NativeConvert {
where
Self: NativeSize,
{
// Convert luavalue into data, then write into ptr // Convert luavalue into data, then write into ptr
unsafe fn luavalue_into<'lua>( unsafe fn luavalue_into<'lua>(
&self, &self,

View file

@ -15,9 +15,7 @@ pub trait NativeSignedness {
} }
pub use self::{ pub use self::{
arg::FfiArgRefOption,
arg::NativeArgInfo, arg::NativeArgInfo,
arg::NativeArgType,
cast::native_num_cast, cast::native_num_cast,
convert::NativeConvert, convert::NativeConvert,
data::GetNativeData, data::GetNativeData,

View file

@ -7,5 +7,6 @@ use super::NativeConvert;
pub struct NativeResultInfo { pub struct NativeResultInfo {
pub conv: *const dyn NativeConvert, pub conv: *const dyn NativeConvert,
pub size: usize,
// kind: NativeResultType, // kind: NativeResultType,
} }

View file

@ -1,5 +1,3 @@
use std::clone;
// Memory range for ref or box data. For boundary checking // Memory range for ref or box data. For boundary checking
pub struct FfiRefBounds { pub struct FfiRefBounds {
// Indicates how much data is above the pointer // Indicates how much data is above the pointer

View file

@ -15,10 +15,9 @@ pub use self::{
ffi_closure::FfiClosure, ffi_closure::FfiClosure,
ffi_lib::FfiLib, ffi_lib::FfiLib,
ffi_native::{ ffi_native::{
native_num_cast, FfiArgRefOption, GetNativeData, NativeArgInfo, NativeArgType, native_num_cast, GetNativeData, NativeArgInfo, NativeConvert, NativeData, NativeResultInfo,
NativeConvert, NativeData, NativeResultInfo, NativeSignedness, NativeSize, NativeSignedness, NativeSize,
}, },
ffi_raw::FfiRaw,
ffi_ref::{create_nullptr, FfiRef, FfiRefFlag}, ffi_ref::{create_nullptr, FfiRef, FfiRefFlag},
}; };

View file

@ -6,6 +6,7 @@ use mlua::prelude::*;
mod c; mod c;
mod ffi; mod ffi;
mod libffi_helper;
use crate::{ use crate::{
c::{create_all_c_types, create_all_types, CFn, CStruct}, c::{create_all_c_types, create_all_types, CFn, CStruct},

View file

@ -0,0 +1,29 @@
use std::ptr::{self, null_mut};
use libffi::{low, raw};
use mlua::prelude::*;
use crate::ffi::FFI_STATUS_NAMES;
// Ensure sizeof c-type (raw::libffi_type)
// See: http://www.chiark.greenend.org.uk/doc/libffi-dev/html/Size-and-Alignment.html
pub fn get_ensured_size(ffi_type: *mut raw::ffi_type) -> LuaResult<usize> {
let mut cif = low::ffi_cif::default();
let result = unsafe {
raw::ffi_prep_cif(
ptr::from_mut(&mut cif),
raw::ffi_abi_FFI_DEFAULT_ABI,
0,
ffi_type,
null_mut(),
)
};
if result != raw::ffi_status_FFI_OK {
return Err(LuaError::external(format!(
"ffi_prep_cif failed. expected result {}, got {}",
FFI_STATUS_NAMES[0], FFI_STATUS_NAMES[result as usize]
)));
}
unsafe { Ok((*ffi_type).size) }
}

View file

@ -0,0 +1,42 @@
local ffi = require("@lune/ffi")
local testdir = "./tests/ffi/external_math"
local compile = require("../utility/compile")
compile(`{testdir}/lib.c`, `{testdir}/lib.so`)
local lib = ffi.open(`{testdir}/lib.so`)
local function test_add_int()
local add_int = ffi.fn({ ffi.int, ffi.int }, ffi.int)
local add_int_caller = add_int:caller(lib:find("add_int"))
local resultBox = ffi.box(ffi.int.size)
local arg1 = ffi.int:box(100)
local arg2 = ffi.int:box(200)
add_int_caller:call(resultBox, arg1, arg2)
local result = ffi.int:from(resultBox)
assert(result == 300, `add_int failed. result expected 300, got {result}`)
end
test_add_int()
local function test_mul_int()
local mul_int = ffi.fn({ ffi.int, ffi.int }, ffi.int)
local mul_int_caller = mul_int:caller(lib:find("mul_int"))
local resultBox = ffi.box(ffi.int.size)
local arg1 = ffi.int:box(100)
local arg2 = ffi.int:box(200)
mul_int_caller:call(resultBox, arg1, arg2)
local result = ffi.int:from(resultBox)
assert(result == 20000, `mul_int failed. result expected 20000, got {result}`)
end
test_mul_int()

View file

@ -0,0 +1,7 @@
int add_int(int a, int b) {
return a + b;
}
int mul_int(int a, int b) {
return a * b;
}

View file

@ -0,0 +1,30 @@
local ffi = require("@lune/ffi")
local testdir = "./tests/ffi/external_struct"
local compile = require("../utility/compile")
compile(`{testdir}/lib.c`, `{testdir}/lib.so`)
local lib = ffi.open(`{testdir}/lib.so`)
local function test_AB()
local ArgStruct = ffi.struct({ ffi.int, ffi.int:ptr() })
local ResultStruct = ffi.struct({ ffi.int, ffi.int })
local AB = ffi.fn({ ArgStruct }, ResultStruct)
local AB_caller = AB:caller(lib:find("AB"))
local resultBox = ffi.box(ffi.int.size)
local a = ffi.int:box(100)
local b = ffi.int:box(200)
local arg = ArgStruct:box({ a, b:leak() })
AB_caller:call(resultBox, arg)
local result = ResultStruct:from(resultBox)
assert(result[0] == 300, `AB failed. result expected 300, got {result}`)
assert(result[1] == 20000, `AB failed. result expected 300, got {result}`)
end
test_AB()

View file

@ -0,0 +1,14 @@
typedef struct {
int a;
int* b;
} ArgStruct;
typedef struct {
int sum;
int mul;
} ResultStruct;
ResultStruct AB(ArgStruct t) {
ResultStruct result = { t.a+*t.b, t.a**t.b };
return result;
}

View file

@ -0,0 +1,9 @@
local process = require("@lune/process")
local function compile(file, out)
local gcc = process.spawn("gcc", { "-shared", "-o", out, "-fPIC", file })
if not gcc.ok then
error("Failed to execute gcc command\n" .. gcc.stdout .. gcc.stderr)
end
end
return compile