mirror of
https://github.com/ZDoom/raze-gles.git
synced 2025-01-12 11:10:39 +00:00
Lunatic: module(), require() for custom modules, bit op test from LJ homepage.
git-svn-id: https://svn.eduke32.com/eduke32@2837 1a8010ca-5511-0410-912e-c29ae57300e0
This commit is contained in:
parent
a464527f8f
commit
55d474990f
5 changed files with 161 additions and 7 deletions
53
polymer/eduke32/source/lunatic/bittest.lua
Normal file
53
polymer/eduke32/source/lunatic/bittest.lua
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
|
||||||
|
local string = require "string"
|
||||||
|
|
||||||
|
local getticks
|
||||||
|
|
||||||
|
local bit = require("bit")
|
||||||
|
local print = print
|
||||||
|
|
||||||
|
if (string.dump) then
|
||||||
|
-- stand-alone
|
||||||
|
local os = require "os"
|
||||||
|
function getticks()
|
||||||
|
return os.clock()*1000
|
||||||
|
end
|
||||||
|
else
|
||||||
|
-- embedded
|
||||||
|
getticks = gv.gethitickms
|
||||||
|
|
||||||
|
module(...)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- from http://bitop.luajit.org/api.html
|
||||||
|
|
||||||
|
local band, bxor = bit.band, bit.bxor
|
||||||
|
local rshift, rol = bit.rshift, bit.rol
|
||||||
|
|
||||||
|
local m = 1000000
|
||||||
|
|
||||||
|
function sieve()
|
||||||
|
local count = 0
|
||||||
|
local p = {}
|
||||||
|
|
||||||
|
for i=0,(m+31)/32 do p[i] = -1 end
|
||||||
|
|
||||||
|
local t = getticks()
|
||||||
|
|
||||||
|
for i=2,m do
|
||||||
|
if band(rshift(p[rshift(i, 5)], i), 1) ~= 0 then
|
||||||
|
count = count + 1
|
||||||
|
for j=i+i,m,i do
|
||||||
|
local jx = rshift(j, 5)
|
||||||
|
p[jx] = band(p[jx], rol(-2, j))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
print(string.format("Found %d primes up to %d (%.02f ms)", count, m,
|
||||||
|
getticks()-t))
|
||||||
|
end
|
||||||
|
|
||||||
|
if (string.dump) then
|
||||||
|
sieve()
|
||||||
|
end
|
|
@ -471,6 +471,13 @@ playerdata_t g_player[MAXPLAYERS];
|
||||||
]]
|
]]
|
||||||
|
|
||||||
-- functions
|
-- functions
|
||||||
|
decl[[
|
||||||
|
int32_t kopen4loadfrommod(const char *filename, char searchfirst);
|
||||||
|
int32_t kfilelength(int32_t handle);
|
||||||
|
void kclose(int32_t handle);
|
||||||
|
int32_t kread(int32_t handle, void *buffer, int32_t leng);
|
||||||
|
]]
|
||||||
|
|
||||||
ffi.cdef[[
|
ffi.cdef[[
|
||||||
int32_t ksqrt(uint32_t num);
|
int32_t ksqrt(uint32_t num);
|
||||||
]]
|
]]
|
||||||
|
@ -510,6 +517,8 @@ local allowed_modules = {
|
||||||
coroutine=coroutine, bit=bit, table=table, math=math, string=string,
|
coroutine=coroutine, bit=bit, table=table, math=math, string=string,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
local package_loaded = {}
|
||||||
|
|
||||||
for modname, themodule in pairs(allowed_modules) do
|
for modname, themodule in pairs(allowed_modules) do
|
||||||
local mt = {
|
local mt = {
|
||||||
__index = themodule,
|
__index = themodule,
|
||||||
|
@ -522,22 +531,95 @@ for modname, themodule in pairs(allowed_modules) do
|
||||||
allowed_modules[modname] = setmetatable({}, mt)
|
allowed_modules[modname] = setmetatable({}, mt)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local function check_valid_modname(modname)
|
||||||
|
if (type(modname) ~= "string") then
|
||||||
|
error("module name must be a string", 5)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- TODO: restrict valid names?
|
||||||
|
end
|
||||||
|
|
||||||
|
local function errorf(level, fmt, ...)
|
||||||
|
local errmsg = string.format(fmt, ...)
|
||||||
|
error(errmsg, level+1)
|
||||||
|
|
||||||
|
-- XXX: error(nil) propagates to C!
|
||||||
|
end
|
||||||
|
|
||||||
|
local ERRLEV = 5
|
||||||
|
|
||||||
-- The "require" function accessible to Lunatic code.
|
-- The "require" function accessible to Lunatic code.
|
||||||
-- Base modules in allowed_modules are wrapped so that they cannot be
|
-- Base modules in allowed_modules are wrapped so that they cannot be
|
||||||
-- modified, (TODO:) user modules are searched in the EDuke32 search
|
-- modified, user modules are searched in the EDuke32 search
|
||||||
-- path. Also, our require never messes with the global environment,
|
-- path. Also, our require never messes with the global environment,
|
||||||
-- it only returns the module.
|
-- it only returns the module.
|
||||||
local function our_require(modname)
|
local function our_require(modname)
|
||||||
local themodule = allowed_modules[modname]
|
check_valid_modname(modname)
|
||||||
|
|
||||||
if (themodule==nil) then
|
-- see whether it's a base module name first
|
||||||
error("NOT IMPLEMENTED")
|
if (allowed_modules[modname] ~= nil) then
|
||||||
-- need to search user modules here
|
return allowed_modules[modname]
|
||||||
end
|
end
|
||||||
|
|
||||||
return themodule
|
--- search user modules
|
||||||
|
|
||||||
|
if (package_loaded[modname] ~= nil) then
|
||||||
|
-- already loaded
|
||||||
|
return package_loaded[modname]
|
||||||
|
end
|
||||||
|
|
||||||
|
local function readintostr(fn)
|
||||||
|
-- XXX: this is pretty much the same as the code in El_RunOnce()
|
||||||
|
|
||||||
|
local fd = ffiC.kopen4loadfrommod(fn, 0) -- TODO: g_loadFromGroupOnly
|
||||||
|
if (fd < 0) then
|
||||||
|
errorf(ERRLEV, "Couldn't open file \"%s\"", fn)
|
||||||
|
end
|
||||||
|
|
||||||
|
local sz = ffiC.kfilelength(fd)
|
||||||
|
if (sz == 0) then
|
||||||
|
ffiC.kclose(fd)
|
||||||
|
errorf(ERRLEV, "Didn't load module \"%s\": zero-length file", fn)
|
||||||
|
end
|
||||||
|
|
||||||
|
if (sz < 0) then
|
||||||
|
ffi.kclose(fd)
|
||||||
|
error("INTERNAL ERROR: kfilelength() returned negative length", 5)
|
||||||
|
end
|
||||||
|
|
||||||
|
local str = ffi.new("char [?]", sz) -- XXX: what does it do on out of mem?
|
||||||
|
local readlen = ffiC.kread(fd, str, sz)
|
||||||
|
|
||||||
|
ffiC.kclose(fd); fd=-1
|
||||||
|
|
||||||
|
if (readlen ~= sz) then
|
||||||
|
errorf(ERRLEV, "INTERNAL ERROR: couldn't read \"%s\" wholly", fn)
|
||||||
|
end
|
||||||
|
|
||||||
|
return ffi.string(str, sz)
|
||||||
|
end
|
||||||
|
|
||||||
|
-- TODO: better pattern-matching (permit "", .lua, .elua ?)
|
||||||
|
local str = readintostr(modname .. ".lua")
|
||||||
|
|
||||||
|
local modfunc, errmsg = loadstring(str)
|
||||||
|
if (modfunc == nil) then
|
||||||
|
errorf(ERRLEV-1, "Couldn't load \"%s\": %s", fn, errmsg)
|
||||||
|
end
|
||||||
|
|
||||||
|
local modtab = modfunc(modname)
|
||||||
|
|
||||||
|
if (modtab ~= nil) then
|
||||||
|
if (type(modtab) ~= "table") then
|
||||||
|
errorf(ERRLEV-1, "Didn't load module \"%s\": expected table as return value", modname)
|
||||||
|
end
|
||||||
|
package_loaded[modname] = modtab
|
||||||
|
end
|
||||||
|
|
||||||
|
return package_loaded[modname]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
-- _G tweaks -- pull in only 'safe' stuff
|
-- _G tweaks -- pull in only 'safe' stuff
|
||||||
local G_ = {} -- our soon-to-be global environment
|
local G_ = {} -- our soon-to-be global environment
|
||||||
|
|
||||||
|
@ -546,7 +628,18 @@ local G_ = {} -- our soon-to-be global environment
|
||||||
-- this table, since user code could later do pairs=nil.
|
-- this table, since user code could later do pairs=nil.
|
||||||
local oG = _G
|
local oG = _G
|
||||||
|
|
||||||
|
-- replacement for "module"
|
||||||
|
local function our_module(modname)
|
||||||
|
check_valid_modname(modname)
|
||||||
|
|
||||||
|
local M = {}
|
||||||
|
package_loaded[modname] = M
|
||||||
|
-- change the environment of the function which called us:
|
||||||
|
oG.setfenv(2, M)
|
||||||
|
end
|
||||||
|
|
||||||
G_.require = our_require
|
G_.require = our_require
|
||||||
|
G_.module = our_module
|
||||||
---G_.coroutine = coroutine
|
---G_.coroutine = coroutine
|
||||||
G_.assert = assert
|
G_.assert = assert
|
||||||
G_.tostring = tostring
|
G_.tostring = tostring
|
||||||
|
|
|
@ -26,6 +26,11 @@ nextspritestat;
|
||||||
headsectbunch;
|
headsectbunch;
|
||||||
nextsectbunch;
|
nextsectbunch;
|
||||||
|
|
||||||
|
kopen4loadfrommod;
|
||||||
|
kfilelength;
|
||||||
|
kclose;
|
||||||
|
kread;
|
||||||
|
|
||||||
ksqrt;
|
ksqrt;
|
||||||
hitscan;
|
hitscan;
|
||||||
|
|
||||||
|
|
|
@ -108,7 +108,7 @@ int32_t El_RunOnce(El_State *estate, const char *fn)
|
||||||
|
|
||||||
lua_State *const L = estate->L;
|
lua_State *const L = estate->L;
|
||||||
|
|
||||||
fid = kopen4load(fn, 0);
|
fid = kopen4load(fn, 0); // TODO: g_loadFromGroupOnly, kopen4loadfrommod ?
|
||||||
|
|
||||||
if (fid < 0)
|
if (fid < 0)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -237,4 +237,7 @@ gameactor(1680, -- LIZTROOP
|
||||||
|
|
||||||
printf("EVENT_INIT = %d", gv.EVENT_INIT) -- tests default defines
|
printf("EVENT_INIT = %d", gv.EVENT_INIT) -- tests default defines
|
||||||
|
|
||||||
|
local bittest = require "bittest"
|
||||||
|
bittest.sieve()
|
||||||
|
|
||||||
print('---=== END TEST SCRIPT ===---')
|
print('---=== END TEST SCRIPT ===---')
|
||||||
|
|
Loading…
Reference in a new issue