lunatic bits and pieces

git-svn-id: https://svn.eduke32.com/eduke32@2044 1a8010ca-5511-0410-912e-c29ae57300e0
This commit is contained in:
helixhorned 2011-09-25 15:11:47 +00:00
parent 04d1ebb010
commit c6f58c8dde
3 changed files with 141 additions and 30 deletions

View file

@ -3,7 +3,7 @@
local ffi = require("ffi") local ffi = require("ffi")
-- sector, wall, sprite ---- sector, wall, sprite ----
ffi.cdef[[ ffi.cdef[[
#pragma pack(push,1) #pragma pack(push,1)
typedef struct typedef struct
@ -24,7 +24,7 @@ typedef struct
typedef struct typedef struct
{ {
int32_t x, y; int32_t x, y;
int16_t point2, nextwall, nextsector, cstat; const int16_t point2, nextwall, nextsector; int16_t cstat;
int16_t picnum, overpicnum; int16_t picnum, overpicnum;
int8_t shade; int8_t shade;
uint8_t pal, xrepeat, yrepeat, xpanning, ypanning; uint8_t pal, xrepeat, yrepeat, xpanning, ypanning;
@ -52,50 +52,130 @@ walltype *wall;
spritetype *sprite; spritetype *sprite;
const int16_t numsectors, numwalls; const int16_t numsectors, numwalls;
const int16_t headspritesect[16384+1], headspritestat[1024+1];
const int16_t prevspritesect[16384], prevspritestat[16384];
const int16_t nextspritesect[16384], nextspritestat[16384];
]] ]]
-- --
---- Set up restricted access to ffi.C from lunatic. ----
local ffiC = ffi.C
local det = {} -- dummy empty table
local tmpmt = {
__index = function() error('dummy variable: read access forbidden') end,
__newindex = function() error('dummy variable: write access forbidden') end,
__metatable = true -- forbid setting the metatable
}
setmetatable(det, tmpmt)
-- GLOBAL gv: provides access to C global *scalars*
gv = {
-- all non-scalars need to be explicitly listed
-- and access to them is redirected to the dummy
-- empty table... this is somewhat ugly
sector = det,
wall = det,
sprite = det,
headspritesect = det, headspritestat = det,
prevspritesect = det, prevspritestat = det,
nextspritesect = det, nextspritestat = det,
}
local tmpmt = {
__index = ffiC,
__newindex = function() error("cannot create new fields in 'gv'") end,
__metatable = true,
}
setmetatable(gv, tmpmt)
---- indirect C array access ----
sector = {} sector = {}
local tmpmt = { local tmpmt = {
__index = function(tab, key) __index = function(tab, key)
if (key >= 0 and key < ffi.C.numsectors) then return ffi.C.sector[key] end if (key >= 0 and key < ffiC.numsectors) then return ffiC.sector[key] end
error('out-of-bounds sector[] read access') error('out-of-bounds sector[] read access')
end, end,
__newindex = function(tab, key, val) error('cannot write to sector[] structs directly') end __newindex = function(tab, key, val) error('cannot write to sector[] struct directly') end,
__metatable = true,
} }
setmetatable(sector, tmpmt) setmetatable(sector, tmpmt)
wall = {} wall = {}
local tmpmt = { local tmpmt = {
__index = function(tab, key) __index = function(tab, key)
if (key >= 0 and key < ffi.C.numwalls) then return ffi.C.wall[key] end if (key >= 0 and key < ffiC.numwalls) then return ffiC.wall[key] end
error('out-of-bounds wall[] read access') error('out-of-bounds wall[] read access')
end, end,
__newindex = function(tab, key, val) error('cannot write to wall[] structs directly') end __newindex = function(tab, key, val) error('cannot write to wall[] struct directly') end,
__metatable = true,
} }
setmetatable(wall, tmpmt) setmetatable(wall, tmpmt)
sprite = {} -- create a safe indirection for a ffi.C array
local tmpmt = { local function creategtab(ctab, maxidx, name)
__index = function(tab, key) local tab = {}
-- MAXSPRITES == 16384 local tmpmt = {
if (key >= 0 and key < 16384) then return ffi.C.sprite[key] end __index = function(tab, key)
error('out-of-bounds sprite[] read access') if (key>=0 and key < maxidx) then
end, return ctab[key]
end
error('out-of-bounds '..name..' read access')
end,
__newindex = function(tab, key, val)
error('cannot write to '..name)
end,
__metatable = true,
}
setmetatable(tab, tmpmt)
return tab
end
__newindex = function(tab, key, val) error('cannot write to sprite[] structs directly') end sprite = creategtab(ffiC.sprite, 16384, 'sprite[] struct')
} headspritesect = creategtab(ffiC.headspritesect, 16384, 'headspritesect[]')
setmetatable(sprite, tmpmt) headspritestat = creategtab(ffiC.headspritestat, 1024, 'headspritestat[]')
nextspritesect = creategtab(ffiC.nextspritesect, 16384, 'nextspritesect[]')
nextspritestat = creategtab(ffiC.nextspritestat, 16384, 'nextspritestat[]')
prevspritesect = creategtab(ffiC.prevspritesect, 16384, 'prevspritesect[]')
prevspritestat = creategtab(ffiC.prevspritestat, 16384, 'prevspritestat[]')
-- yes, this does export a couple of other stuff that users ought not see, ---- per-sector/per-statnum sprite iterators ----
-- but without the ffi.cdef declarations, they will just sit there and local function iter_spritesofsect(sect, i)
-- refuse to be accessed. if (i < 0) then
gv = ffi.C i = ffiC.headspritesect[sect]
else
i = ffiC.nextspritesect[i]
end
-- nope, this would create two numeric variables with their initial values, but if (i >= 0) then return i, i end
-- certainly not references to numsectors and numwalls like we want: end
--numsectors = ffi.C.numsectors
--numwalls = ffi.C.numwalls function spritesofsect(sect)
assert(sect >= 0 and sect < ffiC.numsectors, "passed invalid sectnum to spritesofsect iterator")
return iter_spritesofsect, sect, -1
end
local function iter_spritesofstat(stat, i)
if (i < 0) then
i = ffiC.headspritestat[stat]
else
i = ffiC.nextspritestat[i]
end
if (i >= 0) then return i, i end
end
function spritesofstat(stat)
assert(stat >= 0 and stat < 1024, "passed invalid statnum to spritesofstat iterator")
return iter_spritesofstat , stat, -1
end
-- restrict access to potentially unsafe standard Lua modules (not yet done)
os = nil
io = nil

View file

@ -5,4 +5,8 @@ sprite;
numsectors; numsectors;
numwalls; numwalls;
headspritesect; headspritestat;
prevspritesect; prevspritestat;
nextspritesect; nextspritestat;
}; };

View file

@ -2,6 +2,15 @@
print('--- ELua Test script ---') print('--- ELua Test script ---')
local function checkfail(funcstr)
local status, res = pcall(loadstring(funcstr))
if (status) then
print('ERROR: '..funcstr.." DIDN'T fail")
else
print('SUCCESS: '..funcstr.." failed: "..res)
end
end
local i local i
print('tweaking sector pals') print('tweaking sector pals')
@ -10,12 +19,30 @@ for i = 0, gv.numsectors/2 do
sector[i].ceilingpal = 2; sector[i].ceilingpal = 2;
end end
print(gv.numsectors) checkfail('gv.sprite[0].yrepeat = 100') -- direct gv array access forbidden
print(gv.numwalls) print('tweaking some sprites')
for spr in spritesofsect(307) do -- some fence sprites in E1L1
print('spr', spr)
sprite[spr].pal = 6
end
for spr in spritesofsect(236) do
print('#spr', spr)
end
print('_G contains:')
for k,v in pairs(_G) do
print(k, v)
end
checkfail('print(sprite[100000].ceilingpal)') -- oob read access
checkfail('setmetatable(sprite, {})') -- set metatable forbidden
checkfail('sector[-1].ceilingpal = 4') -- oob write access
checkfail('sector[0].wallnum = 0') -- wallnum member is read-only
checkfail('gv.numsectors = 4') -- gv.numsectors is read-only
checkfail('sector[4] = sector[6]') -- direct sector write access forbidden
print('--- end test script ---') print('--- end test script ---')
--sector[-1].ceilingpal = 4; -- this must FAIL! os = require("os")
--sector[0].wallnum = 0; -- as must this print('clk', os.clock())
--gv.numsectors = 4 -- and this
--sector[4] = sector[6] -- this is forbidden, too