-- Implementation of a bound-checked array type factory for LuaJIT 2.0 or later. -- -- Usage example: -- -- > bcarray.new("int8_t", 3, "test", "three_pigs") -- > a = ffi.new("struct { int32_t a; three_pigs p; int16_t b; }") -- > =ffi.sizeof(a) --> 12 -- > b = ffi.new("__attribute__((packed)) struct { int32_t a; three_pigs p; int16_t b; }") -- > =ffi.sizeof(b) --> 9 local ffi = require("ffi") local string = require("string") local table = require("table") local assert = assert local error = error local pairs = pairs local type = type local bcarray = {} -- Generate C decl for a sequence of <nelts> const struct members. -- For example, for 4 elements, -- "const $ _r1, _f2, _u3, _n4;" local function flatten_array(nelts, rng) local strtab = { "$ " } if (rng and rng.getu32==nil) then assert(type(rng)=="table") for i=1,#rng do strtab[i+1] = rng[i]..((i<#rng) and "," or ";") end else for i=1,nelts do local ch = 97 + (rng and (rng:getu32() % 25) or 0) -- 'a'..'z' strtab[i+1] = string.format("_%c%x%s", ch, i, (i<nelts) and "," or ";") end end return table.concat(strtab) end -- ctype = bcarray.new(basetype, numelts, showname [, typename] [, rng] [, mtadd]) -- (optional fields may be nil) -- -- <numelts>: Number of elements in array (small number) -- <showname>: The name to be shown for the derived type in error messages -- <typename>: If non-nil, the name under which the derived type is typedef'd -- <rng>: Random generator state + method :getu32(). If nil, then members are -- named _a1, _a2, ... -- It also may be a table containing member names at numeric indices 1..#rng. -- <mtadd>: A table containing functions __index and/or __newindex. They are -- called first and the bound-checking ones are tail-called then. If the -- custom __index one returns something, it is returned by the composite one. function bcarray.new(basetype, numelts, showname, typename, rng, mtadd) local eltptr_t = ffi.typeof("$ *", ffi.typeof(basetype)) local mt = { __index = function(ar, idx) if (not (idx >= 0 and idx < numelts)) then error("out-of-bounds "..showname.." read access", 2) end return ffi.cast(eltptr_t, ar)[idx] end, -- NOTE: this function will be dead code if the prefixed __newindex -- errors out unconditionally or the bcarray is declared 'const'. __newindex = function(ar, idx, val) if (not (idx >= 0 and idx < numelts)) then error("out-of-bounds "..showname.." write access", 2) end ffi.cast(eltptr_t, ar)[idx] = val end, } if (mtadd ~= nil) then local curindexf, curnewindexf = mt.__index, mt.__newindex local addindexf, addnewindexf = mtadd.__index, mtadd.__newindex if (addindexf) then -- Additional __index metamethod given. mt.__index = function(ar, idx) local sth = addindexf(ar, idx) if (sth ~= nil) then return sth end return curindexf(ar, idx) end end if (addnewindexf) then -- Additional __newindex metamethod given. mt.__newindex = function(ar, idx, val) addnewindexf(ar, idx, val) return curnewindexf(ar, idx, val) end end end local cdeclstr = "struct {"..flatten_array(numelts, rng).."}" local bcarray_t = ffi.typeof(cdeclstr, ffi.typeof(basetype)); bcarray_t = ffi.metatype(bcarray_t, mt) if (not (rng and rng.getu32==nil)) then -- When passing a member name table, it is allowed to have a different -- number of named members than array elements. assert(ffi.sizeof(bcarray_t) == ffi.sizeof(basetype)*numelts) end if (typename ~= nil) then -- Register the type name in the global namespace. assert(type(typename)=="string") ffi.cdef("typedef $ $;", bcarray_t, typename) end return bcarray_t end return bcarray