mirror of
https://github.com/lune-org/lune.git
synced 2025-04-03 01:50:55 +01:00
Fix closure error (#243)
This commit is contained in:
parent
658b5ef75c
commit
5002088240
19 changed files with 78 additions and 107 deletions
|
@ -4,6 +4,25 @@
|
|||
|
||||
See [tests/ffi](../../tests/ffi/README.md)
|
||||
|
||||
## TODO
|
||||
|
||||
- CString
|
||||
|
||||
- Add buffer for owned data support
|
||||
|
||||
- Add math operation.
|
||||
|
||||
> `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
|
||||
|
||||
- Add bit operation
|
||||
|
||||
> Luau only supports 32bit bit operations
|
||||
|
||||
- Add wchar and wstring support
|
||||
|
||||
> For windows API
|
||||
|
||||
## Code structure
|
||||
|
||||
### /c
|
||||
|
@ -84,22 +103,3 @@ Implememt type-casting for all CTypes
|
|||
- **Const `FFI_STATUS_NAMES`:** Used for ffi_status stringify
|
||||
- **Function `get_ensured_size`:** Returns ensured ffi_type size
|
||||
- **Const `SIEE_OF_POINTER`:** Platform specific pointer size (Compile time known)
|
||||
|
||||
## TODO
|
||||
|
||||
- CString
|
||||
|
||||
- Add buffer for owned data support
|
||||
|
||||
- Add math operation.
|
||||
|
||||
> `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
|
||||
|
||||
- Add bit operation
|
||||
|
||||
> Luau only supports 32bit bit operations
|
||||
|
||||
- Add wchar and wstring support
|
||||
|
||||
> For windows API
|
||||
|
|
|
@ -198,7 +198,7 @@ impl CFnInfo {
|
|||
}
|
||||
|
||||
pub fn get_middle_type() -> Type {
|
||||
Type::void()
|
||||
Type::pointer()
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -32,10 +32,7 @@ impl CallableData {
|
|||
}
|
||||
}
|
||||
|
||||
// TODO? async call: if have no lua closure in arguments, fficallble can be called with async way
|
||||
|
||||
pub unsafe fn call(&self, result: LuaValue, args: LuaMultiValue) -> LuaResult<()> {
|
||||
// cache Vec => unable to create async call but no allocation
|
||||
let mut arg_list = Vec::<*mut c_void>::with_capacity(self.arg_info_list.len());
|
||||
|
||||
let result_pointer = if self.result_info.size == 0 {
|
||||
|
@ -50,7 +47,6 @@ impl CallableData {
|
|||
.cast::<c_void>();
|
||||
|
||||
for index in 0..self.arg_info_list.len() {
|
||||
// let arg_info = self.arg_info_list.get(index).unwrap();
|
||||
let arg_value = args
|
||||
.get(index)
|
||||
.ok_or_else(|| LuaError::external(format!("argument {index} required")))?
|
||||
|
@ -59,14 +55,6 @@ 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>());
|
||||
}
|
||||
|
||||
|
|
|
@ -1,5 +1,5 @@
|
|||
use core::ffi::c_void;
|
||||
use std::{borrow::Borrow, mem::transmute, ptr};
|
||||
use std::{borrow::Borrow, ptr};
|
||||
|
||||
use libffi::{
|
||||
low::{closure_alloc, closure_free, ffi_cif},
|
||||
|
@ -20,7 +20,7 @@ use crate::ffi::{
|
|||
pub struct ClosureData {
|
||||
lua: *const Lua,
|
||||
closure: *mut ffi_closure,
|
||||
code: *mut c_void,
|
||||
code: Box<*mut c_void>,
|
||||
arg_info_list: Vec<FfiArg>,
|
||||
result_info: FfiResult,
|
||||
func: LuaRegistryKey,
|
||||
|
@ -44,14 +44,11 @@ unsafe extern "C" fn callback(
|
|||
arg_pointers: *mut *mut c_void,
|
||||
closure_data: *mut c_void,
|
||||
) {
|
||||
dbg!("before ud");
|
||||
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 mut args = Vec::<LuaValue>::with_capacity(len + 1);
|
||||
|
||||
dbg!("before result", closure_data.result_info.size);
|
||||
|
||||
// Push result pointer (ref)
|
||||
args.push(LuaValue::UserData(
|
||||
lua.create_userdata(RefData::new(
|
||||
|
@ -62,8 +59,6 @@ unsafe extern "C" fn callback(
|
|||
.unwrap(),
|
||||
));
|
||||
|
||||
dbg!("before arg");
|
||||
|
||||
// Push arg pointer (ref)
|
||||
for i in 0..len {
|
||||
let arg_info = closure_data.arg_info_list.get(i).unwrap();
|
||||
|
@ -77,8 +72,6 @@ unsafe extern "C" fn callback(
|
|||
));
|
||||
}
|
||||
|
||||
dbg!("before call");
|
||||
|
||||
closure_data
|
||||
.func
|
||||
.borrow()
|
||||
|
@ -101,51 +94,36 @@ impl ClosureData {
|
|||
let (closure, code) = closure_alloc();
|
||||
let code = code.as_mut_ptr();
|
||||
|
||||
dbg!(result_info.size);
|
||||
|
||||
let closure_data = lua.create_userdata(ClosureData {
|
||||
lua: ptr::from_ref(lua),
|
||||
closure,
|
||||
code,
|
||||
code: Box::new(code),
|
||||
arg_info_list,
|
||||
result_info,
|
||||
func,
|
||||
})?;
|
||||
|
||||
dbg!(unsafe {
|
||||
closure_data
|
||||
.to_pointer()
|
||||
.cast::<ClosureData>()
|
||||
.as_ref()
|
||||
.unwrap()
|
||||
.result_info
|
||||
.size
|
||||
});
|
||||
let closure_data_ptr = ptr::from_ref(&*closure_data.borrow::<ClosureData>()?);
|
||||
|
||||
ffi_status_assert(unsafe {
|
||||
ffi_prep_closure_loc(
|
||||
closure,
|
||||
cif,
|
||||
Some(callback),
|
||||
closure_data.to_pointer().cast_mut(),
|
||||
closure_data_ptr.cast::<c_void>().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).cast_mut().cast::<()>()
|
||||
// self.code.cast::<()>()
|
||||
ptr::from_ref::<*mut c_void>(&*self.code)
|
||||
.cast::<()>()
|
||||
.cast_mut()
|
||||
}
|
||||
fn check_inner_boundary(&self, offset: isize, size: usize) -> bool {
|
||||
(offset as usize) + size <= SIZE_OF_POINTER
|
||||
|
@ -166,15 +144,6 @@ impl LuaUserData for ClosureData {
|
|||
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)
|
||||
});
|
||||
|
|
|
@ -8,29 +8,29 @@ gcc for library compiling (for external-\*)
|
|||
|
||||
**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
|
||||
- [x] [external_math](./external_math/init.luau)
|
||||
- [x] [external_pointer](./external_pointer/init.luau)
|
||||
- [x] [external_print](./external_print/init.luau)
|
||||
- [x] [external_struct](./external_struct/init.luau)
|
||||
- [ ] [external_closure](./external_closure/init.luau)
|
||||
|
||||
> failed (segfault)
|
||||
|
||||
**Luau-side**
|
||||
|
||||
- [ ] tests/ffi/pretty-print :white_check_mark:
|
||||
- [ ] [pretty_print](./pretty_print)
|
||||
|
||||
> need box, ref test
|
||||
|
||||
- [x] tests/ffi/isInteger
|
||||
- [ ] tests/ffi/into-boundary
|
||||
- [x] [isInteger](./isInteger)
|
||||
- [ ] [into_boundary](./into_boundary)
|
||||
|
||||
> need assertion
|
||||
|
||||
- [ ] tests/ffi/from-boundary
|
||||
- [ ] [from_boundary](./from_boundary)
|
||||
|
||||
> need assertion
|
||||
|
||||
- [ ] tests/ffi/cast
|
||||
- [ ] [cast](./cast)
|
||||
|
||||
> need assertion
|
||||
|
|
|
@ -1,14 +0,0 @@
|
|||
#include<stdio.h>
|
||||
|
||||
typedef int (*lua_callback_t)(int, int);
|
||||
|
||||
int closure_test(lua_callback_t callback) {
|
||||
printf("%p\n", callback);
|
||||
printf("%d\n", callback(12, 24));
|
||||
|
||||
return callback(12, 24) * 2;
|
||||
}
|
||||
|
||||
int closure(int a, int b) {
|
||||
return a+b;
|
||||
}
|
|
@ -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`)
|
||||
|
@ -15,11 +15,27 @@ local function test_closure()
|
|||
|
||||
local closure_test_info = ffi.c.fn({ callback_info }, ffi.c.int)
|
||||
|
||||
local closure_test_callable = closure_test_info:callable(lib:find("closure_test"))
|
||||
local closure_test_callable = closure_test_info:callable(lib:find("closure"))
|
||||
|
||||
local result_box = ffi.box(ffi.c.int.size)
|
||||
closure_test_callable(result_box, callback_closure:ref())
|
||||
print(callback_closure)
|
||||
local result = ffi.c.int:readData(result_box)
|
||||
assert(result == 72, `test_closure failed. result expected 20000, got {result}`)
|
||||
end
|
||||
|
||||
test_closure()
|
||||
|
||||
local function test_hello_world()
|
||||
local callback_info = ffi.c.fn({}, ffi.c.void)
|
||||
local callback_closure = callback_info:closure(function()
|
||||
print("Hello world in lua closure!")
|
||||
end)
|
||||
|
||||
local closure_test_info = ffi.c.fn({ callback_info }, ffi.c.void)
|
||||
|
||||
local closure_test_callable = closure_test_info:callable(lib:find("hello_world"))
|
||||
|
||||
closure_test_callable(nil, callback_closure:ref())
|
||||
end
|
||||
|
||||
test_hello_world()
|
12
tests/ffi/external_closure/lib.c
Normal file
12
tests/ffi/external_closure/lib.c
Normal file
|
@ -0,0 +1,12 @@
|
|||
#include<stdio.h>
|
||||
|
||||
typedef int (*lua_callback_t)(int, int);
|
||||
typedef void (*lua_hello_world_callback_t)();
|
||||
|
||||
int closure(lua_callback_t lua_closure) {
|
||||
return lua_closure(12, 24) * 2;
|
||||
}
|
||||
|
||||
void hello_world(lua_hello_world_callback_t lua_closure) {
|
||||
lua_closure();
|
||||
}
|
|
@ -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`)
|
|
@ -1,23 +1,23 @@
|
|||
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`)
|
||||
|
||||
local function test_pointer()
|
||||
local pointer_info = c.fn({ c.int:ptr() }, c.void)
|
||||
local function test_pointer_write()
|
||||
local pointer_write_info = c.fn({ c.int:ptr() }, c.void)
|
||||
|
||||
local pointer_callable = pointer_info:callable(lib:find("pointer"))
|
||||
local pointer_write_callable = pointer_write_info:callable(lib:find("pointer_write"))
|
||||
|
||||
local a = ffi.box(c.int.size)
|
||||
|
||||
pointer_callable(nil, a:ref():ref())
|
||||
pointer_write_callable(nil, a:ref():ref())
|
||||
|
||||
local result = c.int:readData(a)
|
||||
|
||||
assert(result == 123, `pointer failed. result expected 123, got {result}`)
|
||||
end
|
||||
|
||||
test_pointer()
|
||||
test_pointer_write()
|
|
@ -1,7 +1,7 @@
|
|||
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`)
|
|
@ -1,7 +1,7 @@
|
|||
local ffi = require("@lune/ffi")
|
||||
local c = ffi.c
|
||||
|
||||
local testdir = "./tests/ffi/external-struct"
|
||||
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`)
|
Loading…
Add table
Reference in a new issue