Move fixed size types into ffi, Fix test cases (#243)

This commit is contained in:
qwreey 2024-10-21 13:19:56 +00:00
parent b31f81459f
commit 658b5ef75c
No known key found for this signature in database
GPG key ID: D28DB79297A214BD
32 changed files with 248 additions and 161 deletions

View file

@ -1,5 +1,9 @@
# `lune-std-ffi`
## Tests
See [tests/ffi](../../tests/ffi/README.md)
## Code structure
### /c
@ -83,16 +87,19 @@ Implememt type-casting for all CTypes
## TODO
Add `CTypeInfo:add(target, from1, from2, ...)` and `:sub` `:mul` `:div` `:mod` `:pow` for math operation.
- CString
> Luau cannot handle i64 or i128
- Add buffer for owned data support
Add bor band and such bit-related operation
- Add math operation.
> Luau only supports 32bit bit operations
> `CTypeInfo:add(target, from1, from2, ...)` and `:sub` `:mul` `:div` `:mod` `:pow`
> Luau cannot handle f64, i64 or i128, so we should provide math operation for it
wchar and wstring support
- Add bit operation
string(null ending) / buffer support
> Luau only supports 32bit bit operations
void support
- Add wchar and wstring support
> For windows API

View file

@ -140,15 +140,17 @@ impl FfiConvert for CArrInfo {
dst: &Ref<dyn FfiData>,
src: &Ref<dyn FfiData>,
) -> LuaResult<()> {
dst.get_inner_pointer()
.byte_offset(dst_offset)
.copy_from(src.get_inner_pointer().byte_offset(src_offset), self.get_size());
dst.get_inner_pointer().byte_offset(dst_offset).copy_from(
src.get_inner_pointer().byte_offset(src_offset),
self.get_size(),
);
Ok(())
}
}
impl LuaUserData for CArrInfo {
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_meta_field(LuaMetaMethod::Type, "CArr");
fields.add_field_method_get("size", |_, this| Ok(this.get_size()));
fields.add_field_method_get("length", |_, this| Ok(this.get_length()));
fields.add_field_function_get("inner", |lua, this: LuaAnyUserData| {

View file

@ -1,5 +1,3 @@
use std::ptr;
use libffi::middle::{Cif, Type};
use mlua::prelude::*;
@ -155,19 +153,18 @@ impl CFnInfo {
this: &LuaAnyUserData,
lua_function: LuaFunction<'lua>,
) -> LuaResult<LuaAnyUserData<'lua>> {
let closure = ClosureData::new(
ptr::from_ref(lua),
let closure_data = ClosureData::alloc(
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)?;
association::set(lua, CLOSURE_CFN, &closure_data, this)?;
association::set(lua, CLOSURE_FUNC, &closure_data, lua_function)?;
Ok(closure_userdata)
Ok(closure_data)
}
pub fn create_callable<'lua>(
@ -206,6 +203,9 @@ impl CFnInfo {
}
impl LuaUserData for CFnInfo {
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_meta_field(LuaMetaMethod::Type, "CFn");
}
fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) {
// Subtype
method_provider::provide_ptr(methods);

View file

@ -302,6 +302,8 @@ pub fn stringify(lua: &Lua, userdata: &LuaAnyUserData) -> LuaResult<String> {
CPtrInfo::stringify(lua, userdata)
} else if userdata.is::<CFnInfo>() {
CFnInfo::stringify(lua, userdata)
} else if userdata.is::<CVoidInfo>() {
CVoidInfo::stringify()
} else if let Some(name) = ctype_helper::get_name(userdata)? {
Ok(String::from(name))
} else {

View file

@ -17,7 +17,7 @@ pub use self::{
ptr_info::CPtrInfo,
struct_info::CStructInfo,
type_info::{CTypeCast, CTypeInfo},
types::{ctype_helper, export_ctypes},
types::{ctype_helper, export_c_types, export_fixed_types},
void_info::CVoidInfo,
};
@ -34,10 +34,10 @@ mod association_names {
pub const CLOSURE_CFN: &str = "__closure_cfn";
}
pub fn export(lua: &Lua) -> LuaResult<LuaTable> {
pub fn export_c(lua: &Lua) -> LuaResult<LuaTable> {
TableBuilder::new(lua)?
.with_value("void", CVoidInfo::new())?
.with_values(export_ctypes(lua)?)?
.with_values(export_c_types(lua)?)?
.with_function("struct", |lua, types: LuaTable| {
CStructInfo::from_table(lua, types)
})?

View file

@ -124,6 +124,7 @@ impl CPtrInfo {
impl LuaUserData for CPtrInfo {
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_meta_field(LuaMetaMethod::Type, "CPtr");
fields.add_field_method_get("size", |_, _| Ok(size_of::<usize>()));
fields.add_field_function_get("inner", |lua, this| {
let inner = association::get(lua, CPTR_INNER, this)?

View file

@ -182,6 +182,7 @@ impl FfiConvert for CStructInfo {
impl LuaUserData for CStructInfo {
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_meta_field(LuaMetaMethod::Type, "CStruct");
fields.add_field_method_get("size", |_, this| Ok(this.get_size()));
}
fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) {

View file

@ -91,6 +91,7 @@ where
Self: CTypeCast + FfiSignedness + FfiConvert + FfiSize,
{
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_meta_field(LuaMetaMethod::Type, "CType");
fields.add_field_method_get("size", |_, this| Ok(this.get_size()));
fields.add_meta_field(LuaMetaMethod::Type, "CType");
fields.add_field_method_get("signedness", |_, this| Ok(this.get_signedness()));

View file

@ -34,7 +34,7 @@ macro_rules! create_ctypes {
),)*])
};
}
pub fn export_ctypes(lua: &Lua) -> LuaResult<Vec<(&'static str, LuaAnyUserData)>> {
pub fn export_c_types(lua: &Lua) -> LuaResult<Vec<(&'static str, LuaAnyUserData)>> {
create_ctypes!(
lua,
// Export Compile-time known c-types
@ -55,6 +55,11 @@ pub fn export_ctypes(lua: &Lua) -> LuaResult<Vec<(&'static str, LuaAnyUserData)>
("ulong", c_ulong, Type::c_ulong()),
("longlong", c_longlong, Type::c_longlong()),
("ulonglong", c_ulonglong, Type::c_ulonglong()),
)
}
pub fn export_fixed_types(lua: &Lua) -> LuaResult<Vec<(&'static str, LuaAnyUserData)>> {
create_ctypes!(
lua,
// Export Source-time known c-types (fixed)
("u8", u8, Type::u8()),
("u16", u16, Type::u16()),
@ -70,10 +75,8 @@ pub fn export_ctypes(lua: &Lua) -> LuaResult<Vec<(&'static str, LuaAnyUserData)>
("f32", f32, Type::f32()),
("usize", usize, Type::usize()),
("isize", isize, Type::isize()),
// TODO: c_float and c_double sometime can be half and single,
// TODO: but libffi-rs doesn't support it. need work-around or drop support
("float", f32, Type::f32()),
("double", f64, Type::f64()),
("f32", f32, Type::f32()),
("f64", f64, Type::f64()),
)
}

View file

@ -17,6 +17,7 @@ impl FfiSize for CVoidInfo {
0
}
}
impl CVoidInfo {
pub fn new() -> Self {
Self()
@ -24,6 +25,17 @@ impl CVoidInfo {
pub fn get_middle_type() -> Type {
Type::void()
}
pub fn stringify() -> LuaResult<String> {
Ok(String::from("CVoid"))
}
}
impl LuaUserData for CVoidInfo {}
impl LuaUserData for CVoidInfo {
fn add_fields<'lua, F: LuaUserDataFields<'lua, Self>>(fields: &mut F) {
fields.add_meta_field(LuaMetaMethod::Type, "CVoid");
}
fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) {
method_provider::provide_to_string(methods);
method_provider::provide_ptr(methods);
}
}

View file

@ -59,6 +59,14 @@ impl CallableData {
let arg_ref = arg_value.borrow::<RefData>()?;
// unsafe {
// let argp = arg_ref.get_inner_pointer();
// let fnr = transmute::<*mut c_void, unsafe extern "C" fn(i32, i32) -> i32>(
// *argp.cast::<*mut c_void>(),
// );
// dbg!(fnr(1, 2));
// }
arg_list.push(arg_ref.get_inner_pointer().cast::<c_void>());
}

View file

@ -1,14 +1,18 @@
use core::ffi::c_void;
use std::{borrow::Borrow, ptr};
use std::{borrow::Borrow, mem::transmute, ptr};
use libffi::{
low::{closure_alloc, closure_free, ffi_cif, CodePtr},
low::{closure_alloc, closure_free, ffi_cif},
raw::{ffi_closure, ffi_prep_closure_loc},
};
use mlua::prelude::*;
use super::ref_data::{RefBounds, RefData, RefFlag};
use super::{
association_names::CLSOURE_REF_INNER,
ref_data::{RefBounds, RefData, RefFlag, UNSIZED_BOUNDS},
};
use crate::ffi::{
association,
libffi_helper::{ffi_status_assert, SIZE_OF_POINTER},
FfiArg, FfiData, FfiResult,
};
@ -16,7 +20,7 @@ use crate::ffi::{
pub struct ClosureData {
lua: *const Lua,
closure: *mut ffi_closure,
code: CodePtr,
code: *mut c_void,
arg_info_list: Vec<FfiArg>,
result_info: FfiResult,
func: LuaRegistryKey,
@ -32,6 +36,7 @@ impl Drop for ClosureData {
const RESULT_REF_FLAGS: u8 =
RefFlag::Leaked.value() | RefFlag::Writable.value() | RefFlag::Offsetable.value();
const CLOSURE_REF_FLAGS: u8 = RefFlag::Function.value();
unsafe extern "C" fn callback(
cif: *mut ffi_cif,
@ -45,7 +50,7 @@ unsafe extern "C" fn callback(
let len = (*cif).nargs as usize;
let mut args = Vec::<LuaValue>::with_capacity(len + 1);
dbg!("before result");
dbg!("before result", closure_data.result_info.size);
// Push result pointer (ref)
args.push(LuaValue::UserData(
@ -81,48 +86,66 @@ unsafe extern "C" fn callback(
.unwrap()
.as_function()
.unwrap()
.call::<_, ()>(args)
.call::<_, ()>(LuaMultiValue::from_vec(args))
.unwrap();
}
impl ClosureData {
pub fn new(
lua: *const Lua,
pub fn alloc(
lua: &Lua,
cif: *mut ffi_cif,
arg_info_list: Vec<FfiArg>,
result_info: FfiResult,
func: LuaRegistryKey,
) -> LuaResult<ClosureData> {
) -> LuaResult<LuaAnyUserData> {
let (closure, code) = closure_alloc();
let code = code.as_mut_ptr();
let closure_data = ClosureData {
lua,
dbg!(result_info.size);
let closure_data = lua.create_userdata(ClosureData {
lua: ptr::from_ref(lua),
closure,
code,
arg_info_list,
result_info,
func,
};
})?;
dbg!(unsafe {
closure_data
.to_pointer()
.cast::<ClosureData>()
.as_ref()
.unwrap()
.result_info
.size
});
ffi_status_assert(unsafe {
ffi_prep_closure_loc(
closure,
cif,
Some(callback),
ptr::from_ref(&closure_data).cast::<c_void>().cast_mut(),
code.as_mut_ptr(),
closure_data.to_pointer().cast_mut(),
code,
)
})?;
unsafe {
// let argp = closure_data.borrow::<ClosureData>()?.get_inner_pointer();
let fnr = transmute::<*mut c_void, unsafe extern "C" fn(i32, i32) -> i32>(code);
dbg!(fnr(1, 2));
}
Ok(closure_data)
}
}
impl FfiData for ClosureData {
unsafe fn get_inner_pointer(&self) -> *mut () {
ptr::from_ref(&self.code.as_mut_ptr())
.cast_mut()
.cast::<()>()
ptr::from_ref(&self.code).cast_mut().cast::<()>()
// self.code.cast::<()>()
}
fn check_inner_boundary(&self, offset: isize, size: usize) -> bool {
(offset as usize) + size <= SIZE_OF_POINTER
@ -137,6 +160,23 @@ impl FfiData for ClosureData {
impl LuaUserData for ClosureData {
fn add_methods<'lua, M: LuaUserDataMethods<'lua, Self>>(methods: &mut M) {
// methods.add_function("ref", function);
methods.add_function("ref", |lua, this: LuaAnyUserData| {
let ref_data = lua.create_userdata(RefData::new(
unsafe { this.borrow::<ClosureData>()?.get_inner_pointer() },
CLOSURE_REF_FLAGS,
UNSIZED_BOUNDS,
))?;
unsafe {
let mut b = this.borrow_mut::<ClosureData>()?;
b.lua = ptr::from_ref(lua);
let argp = b.get_inner_pointer();
let fnr = transmute::<*mut c_void, unsafe extern "C" fn(i32, i32) -> i32>(
*argp.cast::<*mut c_void>(),
);
dbg!(fnr(1, 2));
}
association::set(lua, CLSOURE_REF_INNER, &ref_data, &this)?;
Ok(ref_data)
});
}
}

View file

@ -14,7 +14,7 @@ pub use self::{
callable_data::CallableData,
closure_data::ClosureData,
lib_data::LibData,
ref_data::{create_nullptr, RefBounds, RefData, RefFlag},
ref_data::{create_nullref, RefBounds, RefData, RefFlag},
};
use crate::ffi::FfiData;
@ -22,6 +22,7 @@ use crate::ffi::FfiData;
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";
}
pub trait GetFfiData {

View file

@ -7,7 +7,6 @@ pub enum RefFlag {
Writable,
Offsetable,
Function,
Uninit,
}
impl RefFlag {
pub const fn value(&self) -> u8 {
@ -18,7 +17,6 @@ impl RefFlag {
Self::Readable => U8_MASK4,
Self::Offsetable => U8_MASK5,
Self::Function => U8_MASK6,
Self::Uninit => U8_MASK7,
}
}
}

View file

@ -17,7 +17,6 @@ pub use self::{
// Box:ref():ref() should not be able to modify, Only for external
const BOX_REF_REF_FLAGS: u8 = 0;
const UNINIT_REF_FLAGS: u8 = RefFlag::Uninit.value();
// | FfiRefFlag::Writable.value()
// | FfiRefFlag::Readable.value()
// | FfiRefFlag::Dereferenceable.value()
@ -45,14 +44,6 @@ impl RefData {
}
}
pub fn new_uninit() -> Self {
Self {
ptr: ManuallyDrop::new(Box::new(ptr::null_mut())),
flags: UNINIT_REF_FLAGS,
boundary: UNSIZED_BOUNDS,
}
}
// Make FfiRef from ref
pub fn luaref<'lua>(
lua: &'lua Lua,
@ -185,14 +176,11 @@ impl LuaUserData for RefData {
}
}
pub fn create_nullptr(lua: &Lua) -> LuaResult<LuaAnyUserData> {
// https://en.cppreference.com/w/cpp/types/nullptr_t
pub fn create_nullref(lua: &Lua) -> LuaResult<LuaAnyUserData> {
lua.create_userdata(RefData::new(
ptr::null_mut::<()>().cast(),
0,
// usize::MAX means that nullptr is can be 'any' pointer type
// We check size of inner data. give ffi.box(1):ref() as argument which typed as i32:ptr() will fail,
// throw lua error
UNSIZED_BOUNDS,
))
}

View file

@ -8,8 +8,8 @@ mod data;
mod ffi;
use crate::{
c::export as c_export,
data::{create_nullptr, BoxData, LibData, RefData},
c::{export_c, export_fixed_types},
data::{create_nullref, BoxData, LibData},
};
/**
@ -21,12 +21,12 @@ use crate::{
*/
pub fn module(lua: &Lua) -> LuaResult<LuaTable> {
let result = TableBuilder::new(lua)?
.with_function("nullRef", |lua, ()| create_nullptr(lua))?
.with_values(export_fixed_types(lua)?)?
.with_function("nullRef", |lua, ()| create_nullref(lua))?
.with_function("box", |_lua, size: usize| Ok(BoxData::new(size)))?
.with_function("open", |_lua, name: String| LibData::new(name))?
.with_function("uninitRef", |_lua, ()| Ok(RefData::new_uninit()))?
.with_function("isInteger", |_lua, num: LuaValue| Ok(num.is_integer()))?
.with_value("c", c_export(lua)?)?;
.with_value("c", export_c(lua)?)?;
#[cfg(debug_assertions)]
let result = result.with_function("debugAssociation", |lua, str: String| {

36
tests/ffi/README.md Normal file
View file

@ -0,0 +1,36 @@
# tests/ffi
## Requirements
gcc for library compiling (for external-\*)
## Results
**External tests**
- [x] tests/ffi/external-math
- [x] tests/ffi/external-pointer
- [x] tests/ffi/external-print
- [x] tests/ffi/external-struct
- [ ] tests/ffi/external-closure
> failed (segfault)
**Luau-side**
- [ ] tests/ffi/pretty-print :white_check_mark:
> need box, ref test
- [x] tests/ffi/isInteger
- [ ] tests/ffi/into-boundary
> need assertion
- [ ] tests/ffi/from-boundary
> need assertion
- [ ] tests/ffi/cast
> need assertion

View file

@ -1,20 +0,0 @@
--!nocheck
--!nolint
local ffi = require("@lune/ffi")
local box = ffi.box(ffi.u8:ptr().size)
local ref = box:ref()
ffi.u8:ptr():into(box, ref)
local wt = setmetatable({}, { __mode = "v" })
wt[1] = box
wt[2] = ref
box = nil
ref = nil
collectgarbage("collect")
assert(wt[1] == nil and wt[2] == nil, "Box - ref recursion GC test failed")

View file

@ -1,6 +1,6 @@
local ffi = require("@lune/ffi")
local testdir = "./tests/ffi/external_closure"
local testdir = "./tests/ffi/external-closure"
local compile = require("../utility/compile")
compile(`{testdir}/lib.c`, `{testdir}/lib.so`)
@ -19,6 +19,7 @@ local function test_closure()
local result_box = ffi.box(ffi.c.int.size)
closure_test_callable(result_box, callback_closure:ref())
print(callback_closure)
end
test_closure()

View file

@ -1,12 +1,12 @@
#include<stdio.h>
typedef int (*lua_callback_t)(int a, int b);
typedef int (*lua_callback_t)(int, int);
int closure_test(lua_callback_t callback) {
printf("%p\n", callback);
printf("%d\n", (*callback)(12, 24));
printf("%d\n", callback(12, 24));
return (*callback)(12, 24) * 2;
return callback(12, 24) * 2;
}
int closure(int a, int b) {

View file

@ -1,7 +1,7 @@
local ffi = require("@lune/ffi")
local c = ffi.c
local testdir = "./tests/ffi/external_math"
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`)

View file

@ -1,7 +1,7 @@
local ffi = require("@lune/ffi")
local c = ffi.c
local testdir = "./tests/ffi/external_pointer"
local testdir = "./tests/ffi/external-pointer"
local compile = require("../utility/compile")
compile(`{testdir}/lib.c`, `{testdir}/lib.so`)
local lib = ffi.open(`{testdir}/lib.so`)

View file

@ -1,18 +1,17 @@
local ffi = require("@lune/ffi")
local c = ffi.c
local testdir = "./tests/ffi/external_print"
local testdir = "./tests/ffi/external-print"
local compile = require("../utility/compile")
compile(`{testdir}/lib.c`, `{testdir}/lib.so`)
local lib = ffi.open(`{testdir}/lib.so`)
local function test_hello_world()
local hello_world_info = ffi.fnInfo({}, ffi.void)
local hello_world_info = c.fn({}, c.void)
local hello_world_callable = hello_world_info:callable(lib:find("hello_world"))
hello_world_callable:call(nil)
hello_world_callable(nil)
end
test_hello_world()

View file

@ -0,0 +1,28 @@
local ffi = require("@lune/ffi")
local c = ffi.c
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 = c.struct({ c.int, c.int:ptr() })
local ResultStruct = c.struct({ c.int, c.int })
local AB = c.fn({ ArgStruct }, ResultStruct)
local AB_callable = AB:callable(lib:find("AB"))
local resultBox = ffi.box(ResultStruct.size)
local b = c.int:box(200)
local arg = ArgStruct:box({ 100, b:ref() })
AB_callable(resultBox, arg:ref())
local result = ResultStruct:readData(resultBox)
assert(result[1] == 300, `AB failed. result expected 300, got {result[1]}`)
assert(result[2] == 20000, `AB failed. result expected 300, got {result[2]}`)
end
test_AB()

View file

@ -1,30 +0,0 @@
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.structInfo({ ffi.int, ffi.int:ptrInfo() })
local ResultStruct = ffi.structInfo({ ffi.int, ffi.int })
local AB = ffi.fnInfo({ ArgStruct }, ResultStruct)
local AB_caller = AB:callable(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:readData(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

@ -1,29 +1,32 @@
local ffi = require("@lune/ffi")
local c = ffi.c
assert(typeof(ffi.int) == "CType")
assert(tostring(ffi.int) == "int")
assert(typeof(c.int) :: string == "CType")
assert(tostring(c.int) == "int")
assert(typeof(ffi.int:ptr()) == "CPtr")
assert(tostring(ffi.int:ptr()) == "int")
assert(tostring(ffi.int:arr(5):ptr()) == " <CArr( int, length = 5 )> ")
assert(typeof(c.int:ptr()) :: string == "CPtr")
assert(tostring(c.int:ptr()) == "int")
assert(tostring(c.int:arr(5):ptr()) == " <CArr( int, length = 5 )> ")
assert(typeof(ffi.int:arr(5)) == "CArr")
assert(tostring(ffi.int:arr(5)) == " int, length = 5 ")
assert(tostring(ffi.int:ptr():arr(5)) == " <CPtr(int)>, length = 5 ")
assert(typeof(c.int:arr(5)) :: string == "CArr")
assert(tostring(c.int:arr(5)) == " int, length = 5 ")
assert(tostring(c.int:ptr():arr(5)) == " <CPtr(int)>, length = 5 ")
assert(typeof(ffi.funcInfo({ ffi.int }, ffi.int)) == "CFunc")
assert(tostring(ffi.funcInfo({ ffi.int }, ffi.int)) == " (int) -> int ")
assert(tostring(ffi.funcInfo({ ffi.int, ffi.double }, ffi.int)) == " (int, double) -> int ")
assert(tostring(ffi.funcInfo({ ffi.int:ptr() }, ffi.int)) == " (<CPtr(int)>) -> int ")
assert(tostring(ffi.funcInfo({ ffi.int }, ffi.int:ptr())) == " (int) -> <CPtr(int)> ")
assert(tostring(ffi.funcInfo({ ffi.int:ptr() }, ffi.int:ptr())) == " (<CPtr(int)>) -> <CPtr(int)> ")
assert(typeof(c.fn({ c.int }, c.int)) :: string == "CFn")
assert(tostring(c.fn({ c.int }, c.int)) == " (int) -> int ")
assert(tostring(c.fn({ c.int, c.double }, c.int)) == " (int, double) -> int ")
assert(tostring(c.fn({ c.int:ptr() }, c.int)) == " (<CPtr(int)>) -> int ")
assert(tostring(c.fn({ c.int }, c.int:ptr())) == " (int) -> <CPtr(int)> ")
assert(tostring(c.fn({ c.int:ptr() }, c.int:ptr())) == " (<CPtr(int)>) -> <CPtr(int)> ")
assert(
tostring(ffi.funcInfo({ ffi.int:ptr(), ffi.int:ptr() }, ffi.int:ptr()))
tostring(c.fn({ c.int:ptr(), c.int:ptr() }, c.int:ptr()))
== " (<CPtr(int)>, <CPtr(int)>) -> <CPtr(int)> "
)
assert(typeof(ffi.structInfo({ ffi.int, ffi.char })) == "CStruct")
assert(typeof(c.struct({ c.int, c.char })) :: string == "CStruct")
assert(
tostring(ffi.structInfo({ ffi.int, ffi.char:ptr() }))
== ` int, <CPtr(char)>, size = {ffi.structInfo({ ffi.int, ffi.char:ptr() }).size} `
tostring(c.struct({ c.int, c.char:ptr() }))
== ` int, <CPtr(char)>, size = {c.struct({ c.int, c.char:ptr() }).size} `
)
-- FIXME: add box, ref pretty-print test

View file

@ -44,7 +44,7 @@ export type CArrInfo<T, R> = {
writeData: (self: CArrInfo<T, R>, target: (Ref|Box), value: { T }, offset: number?) -> (),
copyData: (self: CArrInfo<T, R>, dst: (Ref|Box), src: (Ref|Box), dst_offset: number?, src_offset: number?) -> (),
offset: (self: CArrInfo<T, R>, offset: number) -> number,
offset: (self: CArrInfo<T, R>, index: number) -> number,
}
export type CFnInfo = {
@ -53,12 +53,18 @@ export type CFnInfo = {
}
export type CStructInfo = {
size: number,
arr: (self: CStructInfo, len: number) -> CArrInfo<CStructInfo, {any}>,
ptr: (self: CStructInfo) -> CPtrInfo<CStructInfo>,
box: (self: CStructInfo, table: { any }) -> Box,
readData: (self: CStructInfo, target: (Ref|Box), offset: number?) -> { any },
writeData: (self: CStructInfo, target: (Ref|Box), table: { any }, offset: number?) -> (),
copyData: (self: CStructInfo, dst: (Ref|Box), src: (Ref|Box), dst_offset: number?, src_offset: number?) -> (),
offset: (self: CStructInfo, index: number) -> number,
field: (self: CStructInfo, index: number) -> CTypes,
}
export type CVoidInfo = {
@ -163,21 +169,6 @@ export type Closure = {
local c = {}
c.u8 = {} :: u8
c.u16 = {} :: u16
c.u32 = {} :: u32
c.u64 = {} :: u64
c.u128 = {} :: u128
c.i8 = {} :: i8
c.i16 = {} :: i16
c.i32 = {} :: i32
c.i64 = {} :: i64
c.i128 = {} :: i128
c.f32 = {} :: f32
c.f64 = {} :: f64
c.usize = {} :: usize
c.isize = {} :: isize
c.char = {} :: char
c.float = {} :: float
c.double = {} :: double
@ -206,6 +197,21 @@ local ffi = {}
ffi.c = c
ffi.u8 = {} :: u8
ffi.u16 = {} :: u16
ffi.u32 = {} :: u32
ffi.u64 = {} :: u64
ffi.u128 = {} :: u128
ffi.i8 = {} :: i8
ffi.i16 = {} :: i16
ffi.i32 = {} :: i32
ffi.i64 = {} :: i64
ffi.i128 = {} :: i128
ffi.f32 = {} :: f32
ffi.f64 = {} :: f64
ffi.usize = {} :: usize
ffi.isize = {} :: isize
function ffi.nullRef(): Ref
return nil :: any
end