luajit对垃圾收集的段错误

时间:2013-12-17 10:46:53

标签: lua garbage-collection luajit

我遇到了使用ffi.metatype定义的用户数据的问题。当对象如果垃圾收集,我得到一个段错误。这是代码。

ffi = require("ffi")
srate = 48000

ffi.cdef[[
typedef struct sin_state {
    float _now;   // time of last eval
    float _last;  // result of last eval
    float _freq;  // frequency
    float _gain;  // gain
    float _phase; // phase
    float _w;     // angular speed in radians/sample
    float _p;     // update parameter
    float _y[3];  // last 3 steps
} sin_t;
]]

sin = {}

sin.mt = {
    __call = function(s,now)
        if s._now < now then
            s._now = now
            s._phase = (s._phase + s._w) % 2 * math.pi
            s._y[1] = s._p * s._y[2] - s._y[3]
            s._y[3] = s._y[2]
            s._y[2] = s._y[1]
            s._last = s._y[1] * s._gain
        end
        return s._last 
    end,

    __index = function(s,k)
        return s["_"..k]
    end,

    __newindex = function(s,k,v)
        -- update parameter
        s["_"..k] = v
        -- update internal state
        sin.update(s)
    end
}

function sin.update(s)
    s._w = s._freq * 2 * math.pi / srate
    s._p = 2 * math.cos(s._w)
    s._y[3] = math.sin(-2 * s._w + s._phase)
    s._y[2] = math.sin(-1 * s._w + s._phase)
    s._y[1] = s._p * s._y[2] - s._y[3]
    s._last = s._y[1] * s._gain
end

sin.state = ffi.metatype("sin_t",sin.mt)

function sin.new(t)
    local t = t or {}
    local params = {
        _freq = t.freq or 440,
        _gain = t.gain or 1,
        _phase = t.phase or 0,
    }
    local s = sin.state(params)
    sin.update(s)
    return s
end

s = sin.new()
print(s(1))
s = nil
print("garbage collecting")
collectgarbage()
print("done")

跑步:

% luajit test.lua
6.6929342068534e-09
garbage collecting
[1]    20398 segmentation fault (core dumped)  luajit test.lua

我错过了什么吗?我应该定义__gc元方法吗?如果是这样,我应该把它放在那里?

1 个答案:

答案 0 :(得分:4)

好的,像往常一样,我在发布后几分钟就发现了这个错误。

我搞砸了_y数组的索引。不知怎的,我的印象是luaJIT的FFI库正在进行索引转换(从1开始而不是0),但事实并非如此。

我只是将1减去所有_y[X]并且工作正常。