1

我正在努力将 libcurl 包装在 luajit ffi 中。我的终结器没有被调用。

local ffi = require("ffi")

ffi.cdef [[
  typedef struct{} CURL;
  CURL * curl_easy_init();
  void curl_easy_cleanup(CURL *);
]]

local CURL_lib = ffi.load("../lib/libcurl.so")
local CURL_CTX

local CURL_CTX_mt = {
  __gc = function()  print "finalizing"; CURL_lib.curl_easy_cleanup(CURL_CTX); end
}

ffi.metatype("CURL", CURL_CTX_mt)

CURL_CTX = ffi.new("CURL[1]")
CURL_CTX = CURL_lib.curl_easy_init();
print "done"

我在这里想念什么?:D

BTW CURL 被定义为typedef void CURL;我确定我尝试这样做的方式不够干净。有什么建议吗?

尤里卡!:自我回答——如果有任何明显的问题,仍然对评论感兴趣。

4

2 回答 2

3

luajit 中的编译时间类型必须是structs(或者unions我认为)如果您将上下文类型初始化为指针,它不再是struct. 因此,这里发生了概念不匹配。因此,要解决问题,请将 a 添加void *到您的struct中,将 挂在metatypestruct,然后将void *用于库上下文。

local ffi = require("ffi")

ffi.cdef [[
  typedef struct { void * ctx; } curl;
  curl * curl_easy_init();
  void curl_easy_cleanup(curl *);
]]

local curl_lib = ffi.load("../lib/libcurl.so")
local curl

local curl_mt = {
  __gc = function()  curl_lib.curl_easy_cleanup(curl.ctx); end
}

local curl_proto = ffi.metatype("curl", curl_mt)

curl = curl_proto(nil)
curl.ctx = curl_lib.curl_easy_init();
于 2012-02-22T14:13:29.917 回答
1

或者您可以像这样在返回的指针上注册终结器:

local ffi = require("ffi")

ffi.cdef [[
    typedef struct{} CURL;
    CURL * curl_easy_init();
    void curl_easy_cleanup(CURL *);
]]

local CURL_lib = ffi.load("../lib/libcurl.so")

local function curl_pointer_finalizer(pointer)
    print "finalizing"
    CURL_lib.curl_easy_cleanup(pointer)
end

local function curl_easy_init()
    return ffi.gc(CURL_lib.curl_easy_init(), curl_pointer_finalizer)
end

local CURL_CTX = curl_easy_init() -- wrapper func
print "done"
于 2014-01-28T23:21:20.163 回答