mirror of
https://github.com/ZDoom/raze-gles.git
synced 2025-01-26 00:40:56 +00:00
Lunatic translator: a first codegen ansatz.
git-svn-id: https://svn.eduke32.com/eduke32@2792 1a8010ca-5511-0410-912e-c29ae57300e0
This commit is contained in:
parent
54721d7461
commit
10de03aa7d
1 changed files with 212 additions and 67 deletions
|
@ -17,7 +17,7 @@ lpeg.setmaxstack(1024);
|
||||||
|
|
||||||
|
|
||||||
local Pat, Set, Range, Var = lpeg.P, lpeg.S, lpeg.R, lpeg.V
|
local Pat, Set, Range, Var = lpeg.P, lpeg.S, lpeg.R, lpeg.V
|
||||||
|
local POS = lpeg.Cp
|
||||||
|
|
||||||
---- All keywords pattern -- needed for CON syntax
|
---- All keywords pattern -- needed for CON syntax
|
||||||
local con = require("con_lang")
|
local con = require("con_lang")
|
||||||
|
@ -48,42 +48,87 @@ local g_filename = "???"
|
||||||
local g_directory = "" -- with trailing slash if not empty
|
local g_directory = "" -- with trailing slash if not empty
|
||||||
local g_numerrors = 0
|
local g_numerrors = 0
|
||||||
|
|
||||||
local g_ifnestlevel = 0 -- needed to cope with CONs dangling-else resolution
|
-- How many 'if' statements are following immediately each other,
|
||||||
|
-- needed to cope with CONs dangling-else resolution
|
||||||
|
local g_ifseqlevel = 0
|
||||||
|
|
||||||
|
|
||||||
|
---=== Code generation ===---
|
||||||
|
local g_file_code = {} -- a [filename]=gencode_table mapping
|
||||||
|
local g_curcode = nil -- a table of string pieces or other "gencode" tables
|
||||||
|
|
||||||
|
local g_actor_code = {} -- [actornum]=gencode_table
|
||||||
|
local g_event_code = {} -- [eventnum]=gencode_table
|
||||||
|
local g_loadactor_code = {} -- [actornum]=gencode_table
|
||||||
|
|
||||||
|
|
||||||
local function getlinecol(pos) end -- fwd-decl
|
local function getlinecol(pos) end -- fwd-decl
|
||||||
|
|
||||||
|
local function reset_codegen()
|
||||||
|
g_file_code, g_curcode = {}, nil
|
||||||
|
g_actor_code, g_event_code, g_loadactor_code = {}, {}, {}
|
||||||
|
end
|
||||||
|
|
||||||
|
local function addcode(x)
|
||||||
|
g_curcode[#g_curcode+1] = x
|
||||||
|
end
|
||||||
|
|
||||||
|
local function addcodef(fmt, ...)
|
||||||
|
addcode(string.format(fmt, ...))
|
||||||
|
end
|
||||||
|
|
||||||
|
local function on_actor_end(usertype, tsamm, codetab)
|
||||||
|
-- TODO: strength, action, move, moveflags
|
||||||
|
local tilenum = tsamm[1]
|
||||||
|
|
||||||
|
-- usertype is non-nil only for 'useractor'
|
||||||
|
addcodef("gameactor(%d,\nfunction(actori, playeri, dist)", tilenum)
|
||||||
|
g_actor_code[tilenum] = codetab
|
||||||
|
|
||||||
|
addcode(codetab)
|
||||||
|
addcode("end)")
|
||||||
|
end
|
||||||
|
|
||||||
|
----------
|
||||||
|
|
||||||
local function linecolstr(pos)
|
local function linecolstr(pos)
|
||||||
local line, col = getlinecol(pos)
|
local line, col = getlinecol(pos)
|
||||||
return string.format("%d:%d", line, col)
|
return string.format("%d:%d", line, col)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local function perrprintf(pos, fmt, ...)
|
||||||
|
printf("%s %s: error: "..fmt, g_filename, linecolstr(pos), ...)
|
||||||
|
end
|
||||||
|
|
||||||
local function errprintf(fmt, ...)
|
local function errprintf(fmt, ...)
|
||||||
if (g_lastkwpos) then
|
if (g_lastkwpos) then
|
||||||
printf("%s %s: error: "..fmt, g_filename, linecolstr(g_lastkwpos), ...)
|
perrprintf(g_lastkwpos, fmt, ...)
|
||||||
else
|
else
|
||||||
printf("%s ???: error: "..fmt, g_filename, ...)
|
printf("%s ???: error: "..fmt, g_filename, ...)
|
||||||
end
|
end
|
||||||
g_numerrors = g_numerrors+1
|
g_numerrors = g_numerrors+1
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local function pwarnprintf(pos, fmt, ...)
|
||||||
|
printf("%s %s: warning: "..fmt, g_filename, linecolstr(pos), ...)
|
||||||
|
end
|
||||||
|
|
||||||
local function warnprintf(fmt, ...)
|
local function warnprintf(fmt, ...)
|
||||||
if (g_lastkwpos) then
|
if (g_lastkwpos) then
|
||||||
printf("%s %s: warning: "..fmt, g_filename, linecolstr(g_lastkwpos), ...)
|
pwarnprintf(g_lastkwpos, fmt, ...)
|
||||||
else
|
else
|
||||||
printf("%s ???: warning: "..fmt, g_filename, ...)
|
printf("%s ???: warning: "..fmt, g_filename, ...)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
local function parse_number(numstr)
|
local function parse_number(pos, numstr)
|
||||||
local num = tonumber(numstr)
|
local num = tonumber(numstr)
|
||||||
|
|
||||||
-- TODO: print line number
|
|
||||||
if (num < -0x80000000 or num > 0xffffffff) then
|
if (num < -0x80000000 or num > 0xffffffff) then
|
||||||
errprintf("number %s out of the range of a 32-bit integer", numstr)
|
perrprintf(pos, "number %s out of the range of a 32-bit integer", numstr)
|
||||||
num = NaN
|
num = NaN
|
||||||
elseif (num >= 0x80000000 and numstr:sub(1,2):lower()~="0x") then
|
elseif (num >= 0x80000000 and numstr:sub(1,2):lower()~="0x") then
|
||||||
warnprintf("number %s converted to a negative one", numstr)
|
pwarnprintf(pos, "number %s converted to a negative one", numstr)
|
||||||
num = num-0x100000000
|
num = num-0x100000000
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -93,9 +138,9 @@ end
|
||||||
|
|
||||||
local LABEL = { MOVE=2, AI=3, ACTION=5, [2]="move", [3]="ai", [5]="action" }
|
local LABEL = { MOVE=2, AI=3, ACTION=5, [2]="move", [3]="ai", [5]="action" }
|
||||||
|
|
||||||
local MOVE_NO_ = {0,0}
|
local MOVE_NO = {0,0}
|
||||||
local ACTION_NO_ = {0,0,0,0,0}
|
local ACTION_NO = {0,0,0,0,0}
|
||||||
local LABEL_NO = { [2]=MOVE_NO_, [3]={ACTION_NO_,MOVE_NO_,0}, [5]=ACTION_NO_ }
|
local LABEL_NO = { [2]=MOVE_NO, [3]={ACTION_NO, MOVE_NO, 0}, [5]=ACTION_NO }
|
||||||
|
|
||||||
-- will contain:
|
-- will contain:
|
||||||
-- * scalar numbers: `define'd values
|
-- * scalar numbers: `define'd values
|
||||||
|
@ -105,8 +150,10 @@ local LABEL_NO = { [2]=MOVE_NO_, [3]={ACTION_NO_,MOVE_NO_,0}, [5]=ACTION_NO_ }
|
||||||
-- - action: { startframe, numframes, viewtype, incval, delay } (all <num>)
|
-- - action: { startframe, numframes, viewtype, incval, delay } (all <num>)
|
||||||
-- TODO: IDs for comparison with if*
|
-- TODO: IDs for comparison with if*
|
||||||
local g_labeldef = {}
|
local g_labeldef = {}
|
||||||
|
local TEMP_numlookups = {}
|
||||||
|
|
||||||
local function reset_labels()
|
local function reset_labels()
|
||||||
|
TEMP_numlookups = {}
|
||||||
g_labeldef = { NO=0 } -- NO is also a valid `move', `ai' or `action'
|
g_labeldef = { NO=0 } -- NO is also a valid `move', `ai' or `action'
|
||||||
|
|
||||||
for i=1,#con.labels do
|
for i=1,#con.labels do
|
||||||
|
@ -118,16 +165,16 @@ end
|
||||||
|
|
||||||
-- XXX: error reports give wrong numbers if e.g. happened in "actor" def
|
-- XXX: error reports give wrong numbers if e.g. happened in "actor" def
|
||||||
-- (since it runs when fully parsed, i.e. at "enda")
|
-- (since it runs when fully parsed, i.e. at "enda")
|
||||||
local function lookup_defined_label(maybe_minus_str, identifier)
|
local function lookup_defined_label(pos, maybe_minus_str, identifier)
|
||||||
local num = g_labeldef[identifier]
|
local num = g_labeldef[identifier]
|
||||||
|
|
||||||
if (num == nil) then
|
if (num == nil) then
|
||||||
errprintf("label \"%s\" is not defined", identifier)
|
perrprintf(pos, "label \"%s\" is not defined", identifier)
|
||||||
return -inf -- return a number for type cleanness
|
return -inf -- return a number for type cleanness
|
||||||
end
|
end
|
||||||
|
|
||||||
if (type(num) ~= "number") then
|
if (type(num) ~= "number") then
|
||||||
errprintf("label \"%s\" is not a `define'd value", identifier)
|
perrprintf(pos, "label \"%s\" is not a `define'd value", identifier)
|
||||||
return -inf
|
return -inf
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -155,10 +202,10 @@ local function do_define_label(identifier, num)
|
||||||
g_labeldef[identifier] = num
|
g_labeldef[identifier] = num
|
||||||
end
|
end
|
||||||
|
|
||||||
local function check_move_literal(num)
|
local function check_move_literal(pos, num)
|
||||||
if (num~=0 and num~=1) then
|
if (num~=0 and num~=1) then
|
||||||
errprintf("literal `move' number must be either 0 or 1")
|
perrprintf(pos, "literal `move' number must be either 0 or 1")
|
||||||
return MOVE_NO_
|
return MOVE_NO
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Both move 0 and 1 have hvel and vvel 0, but they must not compare equal
|
-- Both move 0 and 1 have hvel and vvel 0, but they must not compare equal
|
||||||
|
@ -167,16 +214,16 @@ local function check_move_literal(num)
|
||||||
return {0.1*num, 0.1*num}
|
return {0.1*num, 0.1*num}
|
||||||
end
|
end
|
||||||
|
|
||||||
local function check_action_or_ai_literal(labeltype, num)
|
local function check_action_or_ai_literal(labeltype, pos, num)
|
||||||
if (num~=0) then
|
if (num~=0) then
|
||||||
errprintf("literal `%s' number must be 0", LABEL[labeltype])
|
perrprintf(pos, "literal `%s' number must be 0", LABEL[labeltype])
|
||||||
return LABEL_NO[labeltype]
|
return LABEL_NO[labeltype]
|
||||||
end
|
end
|
||||||
|
|
||||||
return LABEL_NO[labeltype]
|
return LABEL_NO[labeltype]
|
||||||
end
|
end
|
||||||
|
|
||||||
local function lookup_composite(labeltype, identifier)
|
local function lookup_composite(labeltype, pos, identifier)
|
||||||
if (identifier=="NO") then
|
if (identifier=="NO") then
|
||||||
-- NO is a special case and is valid for move, ai, action
|
-- NO is a special case and is valid for move, ai, action
|
||||||
return LABEL_NO[labeltype]
|
return LABEL_NO[labeltype]
|
||||||
|
@ -185,16 +232,22 @@ local function lookup_composite(labeltype, identifier)
|
||||||
local val = g_labeldef[identifier]
|
local val = g_labeldef[identifier]
|
||||||
|
|
||||||
if (val == nil) then
|
if (val == nil) then
|
||||||
errprintf("label \"%s\" is not defined", identifier)
|
perrprintf(pos, "label \"%s\" is not defined", identifier)
|
||||||
return LABEL_NO[labeltype] -- return a value of the expected type for cleanness
|
return LABEL_NO[labeltype] -- return a value of the expected type for cleanness
|
||||||
end
|
end
|
||||||
|
|
||||||
if (type(val)~="table" or #val~=labeltype) then
|
if (type(val)~="table" or #val~=labeltype) then
|
||||||
errprintf("label \"%s\" is not a%s `%s' value", identifier,
|
perrprintf(pos, "label \"%s\" is not a%s `%s' value", identifier,
|
||||||
labeltype==LABEL.MOVE and "" or "n", LABEL[labeltype])
|
labeltype==LABEL.MOVE and "" or "n", LABEL[labeltype])
|
||||||
return LABEL_NO[labeltype]
|
return LABEL_NO[labeltype]
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if (TEMP_numlookups[identifier]) then
|
||||||
|
TEMP_numlookups[identifier] = TEMP_numlookups[identifier]+1
|
||||||
|
else
|
||||||
|
TEMP_numlookups[identifier] = 1
|
||||||
|
end
|
||||||
|
|
||||||
return val
|
return val
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -255,6 +308,12 @@ else
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
if (g_file_code[filename] ~= nil) then
|
||||||
|
printf("[%d] Fatal error: infinite loop including \"%s\"", filename)
|
||||||
|
g_numerrors = inf
|
||||||
|
return
|
||||||
|
end
|
||||||
|
|
||||||
local contents = fd:read("*all")
|
local contents = fd:read("*all")
|
||||||
fd:close()
|
fd:close()
|
||||||
|
|
||||||
|
@ -419,7 +478,7 @@ local alphanum = alpha + Range("09")
|
||||||
|
|
||||||
--- basic lexical elements ("tokens")
|
--- basic lexical elements ("tokens")
|
||||||
local t_maybe_minus = (Pat("-") * sp0)^-1;
|
local t_maybe_minus = (Pat("-") * sp0)^-1;
|
||||||
local t_number = lpeg.C(
|
local t_number = POS() * lpeg.C(
|
||||||
t_maybe_minus * ((Pat("0x") + "0X")*Range("09", "af", "AF")^1 + Range("09")^1)
|
t_maybe_minus * ((Pat("0x") + "0X")*Range("09", "af", "AF")^1 + Range("09")^1)
|
||||||
) / parse_number
|
) / parse_number
|
||||||
-- Valid identifier names are disjunct from keywords!
|
-- Valid identifier names are disjunct from keywords!
|
||||||
|
@ -602,10 +661,12 @@ local setperxvarcmd = -- set<actor/player>var[<idx>].<member> <var>
|
||||||
|
|
||||||
local Ci = {
|
local Ci = {
|
||||||
-- these can appear anywhere in the script
|
-- these can appear anywhere in the script
|
||||||
["break"] = cmd(),
|
["break"] = cmd() /
|
||||||
|
"do return end", -- TODO: more exact semantics
|
||||||
["return"] = cmd(),
|
["return"] = cmd(),
|
||||||
|
|
||||||
state = cmd(I),
|
state = cmd(I) /
|
||||||
|
"%1()", -- TODO: mangle names
|
||||||
|
|
||||||
--- 1. get*, set*
|
--- 1. get*, set*
|
||||||
getactor = getstructcmd,
|
getactor = getstructcmd,
|
||||||
|
@ -675,19 +736,22 @@ local Ci = {
|
||||||
getincangle = cmd(W,R,R),
|
getincangle = cmd(W,R,R),
|
||||||
|
|
||||||
--- 3. Actors
|
--- 3. Actors
|
||||||
-- These three need more attention (different kind of labels; move
|
-- These three need more attention (different kind of labels;
|
||||||
-- additionally may accept 0 or 1):
|
-- 'move' additionally may accept 0 or 1):
|
||||||
action = cmd(AC),
|
action = cmd(AC) /
|
||||||
|
function(q) return "ACTION("..(q[1] or "nil")..")" end, --TEMP
|
||||||
ai = cmd(AI),
|
ai = cmd(AI),
|
||||||
move = sp1 * t_move * (sp1 * t_define)^0,
|
move = sp1 * t_move * (sp1 * t_define)^0,
|
||||||
|
|
||||||
cactor = cmd(D),
|
cactor = cmd(D) /
|
||||||
|
"sprite[actori].tilenum=%1",
|
||||||
count = cmd(D),
|
count = cmd(D),
|
||||||
cstator = cmd(D),
|
cstator = cmd(D),
|
||||||
cstat = cmd(D),
|
cstat = cmd(D),
|
||||||
clipdist = cmd(D),
|
clipdist = cmd(D),
|
||||||
sizeto = cmd(D,D),
|
sizeto = cmd(D,D),
|
||||||
sizeat = cmd(D,D),
|
sizeat = cmd(D,D) /
|
||||||
|
"sprite[actori].xrepeat, sprite[actori].yrepeat = %1, %2",
|
||||||
strength = cmd(D),
|
strength = cmd(D),
|
||||||
addstrength = cmd(D),
|
addstrength = cmd(D),
|
||||||
spritepal = cmd(D),
|
spritepal = cmd(D),
|
||||||
|
@ -768,7 +832,8 @@ local Ci = {
|
||||||
flash = cmd(),
|
flash = cmd(),
|
||||||
getlastpal = cmd(),
|
getlastpal = cmd(),
|
||||||
insertspriteq = cmd(),
|
insertspriteq = cmd(),
|
||||||
killit = cmd(), -- exec SPECIAL HANDLING!
|
killit = cmd() /
|
||||||
|
"do return 2 end", -- exec SPECIAL HANDLING!
|
||||||
mikesnd = cmd(),
|
mikesnd = cmd(),
|
||||||
nullop = cmd(),
|
nullop = cmd(),
|
||||||
pkick = cmd(),
|
pkick = cmd(),
|
||||||
|
@ -912,7 +977,6 @@ local Ci = {
|
||||||
}
|
}
|
||||||
|
|
||||||
local Cif = {
|
local Cif = {
|
||||||
-- XXX: ai, action, move/def labels
|
|
||||||
ifai = cmd(AI),
|
ifai = cmd(AI),
|
||||||
ifaction = cmd(AC),
|
ifaction = cmd(AC),
|
||||||
ifmove = cmd(MV),
|
ifmove = cmd(MV),
|
||||||
|
@ -1031,12 +1095,14 @@ function getlinecol(pos) -- local
|
||||||
return line+1, col-1
|
return line+1, col-1
|
||||||
end
|
end
|
||||||
|
|
||||||
-- A generic trace function, prints a position together with the match content
|
-- A generic trace function, prints a position together with the match content.
|
||||||
-- A non-existing 'doit' means 'true'.
|
-- The 'doit' parameter can be used to temporarily enable/disable a particular
|
||||||
|
-- tracing function.
|
||||||
local function TraceFunc(pat, label, doit)
|
local function TraceFunc(pat, label, doit)
|
||||||
|
assert(doit ~= nil)
|
||||||
pat = Pat(pat)
|
pat = Pat(pat)
|
||||||
|
|
||||||
if (doit==nil or doit) then
|
if (doit) then
|
||||||
local function tfunc(subj, pos, a)
|
local function tfunc(subj, pos, a)
|
||||||
printf("%s:%s: %s", linecolstr(pos), label, a)
|
printf("%s:%s: %s", linecolstr(pos), label, a)
|
||||||
return true
|
return true
|
||||||
|
@ -1077,7 +1143,21 @@ local function Stmt(cmdpat) return TraceFunc(cmdpat, "st", false) end
|
||||||
--Ci["myosx"] = Temp(Ci["myosx"])
|
--Ci["myosx"] = Temp(Ci["myosx"])
|
||||||
|
|
||||||
----==== Translator continued ====----
|
----==== Translator continued ====----
|
||||||
local function after_cmd_Cmt()
|
local function after_inner_cmd_Cmt(subj, pos, ...)
|
||||||
|
local capts = {...}
|
||||||
|
|
||||||
|
if (g_numerrors == inf) then
|
||||||
|
return nil
|
||||||
|
end
|
||||||
|
|
||||||
|
if (type(capts[1])=="string" and capts[2]==nil) then
|
||||||
|
return true, capts[1].."--"
|
||||||
|
end
|
||||||
|
|
||||||
|
return true
|
||||||
|
end
|
||||||
|
|
||||||
|
local function after_cmd_Cmt(subj, pos, ...)
|
||||||
if (g_numerrors == inf) then
|
if (g_numerrors == inf) then
|
||||||
-- print("Aborting parsing...")
|
-- print("Aborting parsing...")
|
||||||
return nil -- make the match fail, bail out of parsing
|
return nil -- make the match fail, bail out of parsing
|
||||||
|
@ -1087,17 +1167,18 @@ local function after_cmd_Cmt()
|
||||||
end
|
end
|
||||||
|
|
||||||
-- attach the command names at the front!
|
-- attach the command names at the front!
|
||||||
local function attachnames(kwtab)
|
local function attachnames(kwtab, customfunc)
|
||||||
for cmdname,cmdpat in pairs(kwtab) do
|
for cmdname,cmdpat in pairs(kwtab) do
|
||||||
-- The match-time function capture at the end is so that every
|
-- The match-time function capture at the end is so that every
|
||||||
-- command acts as a barrier to captures to prevent stack overflow (and
|
-- command acts as a barrier to captures to prevent stack overflow (and
|
||||||
-- to make lpeg.match return a subject position at the end)
|
-- to make lpeg.match return a subject position at the end)
|
||||||
kwtab[cmdname] = lpeg.Cmt(Keyw(cmdname) * cmdpat, after_cmd_Cmt)
|
kwtab[cmdname] = lpeg.Cmt(Keyw(cmdname) * cmdpat,
|
||||||
|
customfunc or after_cmd_Cmt)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
attachnames(Co)
|
attachnames(Co)
|
||||||
attachnames(Ci)
|
attachnames(Ci, after_inner_cmd_Cmt)
|
||||||
attachnames(Cif)
|
attachnames(Cif)
|
||||||
|
|
||||||
|
|
||||||
|
@ -1135,28 +1216,22 @@ local function all_alt_pattern(...)
|
||||||
end
|
end
|
||||||
|
|
||||||
-- actor ORGANTIC is greeting!
|
-- actor ORGANTIC is greeting!
|
||||||
local function warn_on_lonely_else()
|
local function warn_on_lonely_else(pos)
|
||||||
warnprintf("found `else' with no `if'")
|
pwarnprintf(pos, "found `else' with no `if'")
|
||||||
end
|
end
|
||||||
|
|
||||||
function EvalEarly(pat)
|
|
||||||
-- force warnings and the like early
|
|
||||||
return lpeg.Cmt(pat, function() return true end)
|
|
||||||
end
|
|
||||||
|
|
||||||
|
|
||||||
local con_inner_command = all_alt_pattern(Ci)
|
local con_inner_command = all_alt_pattern(Ci)
|
||||||
local con_if_begs = all_alt_pattern(Cif)
|
local con_if_begs = all_alt_pattern(Cif)
|
||||||
|
|
||||||
local lone_else = ("else" * sp1)/warn_on_lonely_else
|
local lone_else = (POS() * "else" * sp1)/warn_on_lonely_else
|
||||||
|
|
||||||
local stmt_list = Var("stmt_list")
|
local stmt_list = Var("stmt_list")
|
||||||
-- possibly empty statement list:
|
-- possibly empty statement list:
|
||||||
local stmt_list_or_eps = (stmt_list * sp1)^-1
|
local stmt_list_or_eps = lpeg.Ct((stmt_list * sp1)^-1)
|
||||||
local stmt_list_nosp_or_eps = (stmt_list * (sp1 * stmt_list)^0)^-1
|
local stmt_list_nosp_or_eps = (stmt_list * (sp1 * stmt_list)^0)^-1
|
||||||
|
|
||||||
-- common to actor and useractor: <name/tilenum> [<strength> [<action> [<move> [<flags>... ]]]]
|
-- common to actor and useractor: <name/tilenum> [<strength> [<action> [<move> [<flags>... ]]]]
|
||||||
local common_actor_end = sp1 * t_define * EvalEarly(
|
local common_actor_end = sp1 * lpeg.Ct(t_define *
|
||||||
(sp1 * t_define *
|
(sp1 * t_define *
|
||||||
(sp1 * t_action *
|
(sp1 * t_action *
|
||||||
(sp1 * t_move *
|
(sp1 * t_move *
|
||||||
|
@ -1164,16 +1239,17 @@ local common_actor_end = sp1 * t_define * EvalEarly(
|
||||||
)^-1
|
)^-1
|
||||||
)^-1
|
)^-1
|
||||||
)^-1)
|
)^-1)
|
||||||
* sp1 * stmt_list_or_eps * "enda"
|
* sp1 * stmt_list_or_eps * "enda"
|
||||||
|
|
||||||
--== block delimiters (no recursion) ==--
|
--== block delimiters (no recursion) ==--
|
||||||
local Cb = {
|
local Cb = {
|
||||||
-- actor (...)
|
-- actor (...)
|
||||||
actor = common_actor_end,
|
actor = lpeg.Cc(nil) * common_actor_end / on_actor_end,
|
||||||
-- useractor <actortype> (...)
|
-- useractor <actortype> (...)
|
||||||
useractor = sp1 * t_define * common_actor_end,
|
useractor = sp1 * t_define * common_actor_end / on_actor_end,
|
||||||
-- eventloadactor <name/tilenum>
|
-- eventloadactor <name/tilenum>
|
||||||
eventloadactor = sp1 * t_define * sp1 * stmt_list_or_eps * "enda",
|
eventloadactor = lpeg.Cc(nil) * sp1 * lpeg.Ct(t_define)
|
||||||
|
* sp1 * stmt_list_or_eps * "enda" / on_actor_end,
|
||||||
|
|
||||||
onevent = sp1 * t_define * sp1 * stmt_list_or_eps * "endevent",
|
onevent = sp1 * t_define * sp1 * stmt_list_or_eps * "endevent",
|
||||||
|
|
||||||
|
@ -1196,15 +1272,23 @@ local t_broken_identifier = BadIdent(-((t_number + t_good_identifier) * (sp1 + S
|
||||||
(alphanum + Set("_/\\*")) * (alphanum + Set("_/\\*-"))^0)
|
(alphanum + Set("_/\\*")) * (alphanum + Set("_/\\*-"))^0)
|
||||||
|
|
||||||
local function begin_if_fn()
|
local function begin_if_fn()
|
||||||
g_ifnestlevel = g_ifnestlevel+1
|
g_ifseqlevel = g_ifseqlevel+1
|
||||||
|
|
||||||
|
return "if (TODO) then"
|
||||||
end
|
end
|
||||||
|
|
||||||
local function end_if_fn()
|
local function end_if_fn()
|
||||||
g_ifnestlevel = g_ifnestlevel-1
|
g_ifseqlevel = g_ifseqlevel-1
|
||||||
end
|
end
|
||||||
|
|
||||||
local function check_else_Cmt()
|
local function check_else_Cmt()
|
||||||
return (g_ifnestlevel==0) -- match an 'else' only at the outermost level
|
-- match an 'else' only at the outermost level
|
||||||
|
local good = (g_ifseqlevel==0)
|
||||||
|
if (good) then
|
||||||
|
return good, "else"
|
||||||
|
end
|
||||||
|
|
||||||
|
-- return nothing, making the Cmt fail
|
||||||
end
|
end
|
||||||
|
|
||||||
--- The final grammar!
|
--- The final grammar!
|
||||||
|
@ -1232,14 +1316,19 @@ local Grammar = Pat{
|
||||||
-- getactor [THISACTOR].y y
|
-- getactor [THISACTOR].y y
|
||||||
-- This is in need of cleanup!
|
-- This is in need of cleanup!
|
||||||
t_identifier = -NotKeyw(con.keyword * (sp1 + "[")) * lpeg.C(t_identifier_all),
|
t_identifier = -NotKeyw(con.keyword * (sp1 + "[")) * lpeg.C(t_identifier_all),
|
||||||
t_define = (lpeg.C(t_maybe_minus) * t_identifier / lookup_defined_label) + t_number,
|
t_define = (POS() * lpeg.C(t_maybe_minus) * t_identifier / lookup_defined_label) + t_number,
|
||||||
|
|
||||||
t_move = (t_identifier/function(id) lookup_composite(LABEL.MOVE, id) end) +
|
t_move =
|
||||||
t_number/check_move_literal,
|
POS()*t_identifier / function(...) return lookup_composite(LABEL.MOVE, ...) end +
|
||||||
t_ai = (t_identifier/function(id) lookup_composite(LABEL.AI, id) end) +
|
POS()*t_number / check_move_literal,
|
||||||
t_number/function(num) check_action_or_ai_literal(LABEL.AI, num) end,
|
|
||||||
t_action = (t_identifier/function(id) lookup_composite(LABEL.ACTION, id) end) +
|
t_ai =
|
||||||
t_number/function(num) check_action_or_ai_literal(LABEL.ACTION, num) end,
|
POS()*t_identifier / function(...) return lookup_composite(LABEL.AI, ...) end +
|
||||||
|
POS()*t_number / function(...) return check_action_or_ai_literal(LABEL.AI, ...) end,
|
||||||
|
|
||||||
|
t_action =
|
||||||
|
POS()*t_identifier / function(...) return lookup_composite(LABEL.ACTION, ...) end +
|
||||||
|
POS()*t_number / function(...) return check_action_or_ai_literal(LABEL.ACTION, ...) end,
|
||||||
|
|
||||||
t_arrayexp = t_identifier * arraypat * memberpat^-1,
|
t_arrayexp = t_identifier * arraypat * memberpat^-1,
|
||||||
|
|
||||||
|
@ -1253,8 +1342,10 @@ local Grammar = Pat{
|
||||||
|
|
||||||
default_block = sp1 * Keyw("default") * (sp0*":"*sp0 + sp1) * stmt_list_nosp_or_eps, -- * "break",
|
default_block = sp1 * Keyw("default") * (sp0*":"*sp0 + sp1) * stmt_list_nosp_or_eps, -- * "break",
|
||||||
|
|
||||||
if_stmt = con_if_begs/begin_if_fn * sp1 * Var("single_stmt") * Pat("")/end_if_fn
|
if_stmt = con_if_begs/begin_if_fn * sp1
|
||||||
* (sp1 * lpeg.Cmt(Pat("else"), check_else_Cmt) * sp1 * Var("single_stmt"))^-1,
|
* Var("single_stmt") * (Pat("")/end_if_fn)
|
||||||
|
* (sp1 * lpeg.Cmt(Pat("else"), check_else_Cmt) * sp1 * Var("single_stmt"))^-1
|
||||||
|
* lpeg.Cc("end"),
|
||||||
|
|
||||||
-- TODO?: SST TC has "state ... else ends"
|
-- TODO?: SST TC has "state ... else ends"
|
||||||
while_stmt = Keyw("whilevarvarn") * sp1 * t_rvar * sp1 * t_rvar * sp1 * Var("single_stmt")
|
while_stmt = Keyw("whilevarvarn") * sp1 * t_rvar * sp1 * t_rvar * sp1 * Var("single_stmt")
|
||||||
|
@ -1295,6 +1386,30 @@ local function setup_newlineidxs(contents)
|
||||||
return newlineidxs
|
return newlineidxs
|
||||||
end
|
end
|
||||||
|
|
||||||
|
---
|
||||||
|
local function do_flatten_codetab(code, intotab)
|
||||||
|
for i=1,math.huge do
|
||||||
|
local elt = code[i]
|
||||||
|
|
||||||
|
if (type(elt)=="string") then
|
||||||
|
intotab[#intotab+1] = code[i]
|
||||||
|
elseif (type(elt)=="table") then
|
||||||
|
do_flatten_codetab(elt, intotab)
|
||||||
|
else
|
||||||
|
assert(elt==nil)
|
||||||
|
return
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Return a "string buffer" table that can be table.concat'ed
|
||||||
|
-- to get the code string.
|
||||||
|
local function flatten_codetab(codetab)
|
||||||
|
local tmpcode = {}
|
||||||
|
do_flatten_codetab(codetab, tmpcode)
|
||||||
|
return tmpcode
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
---=== EXPORTED FUNCTIONS ===---
|
---=== EXPORTED FUNCTIONS ===---
|
||||||
|
|
||||||
|
@ -1302,8 +1417,11 @@ function parse(contents) -- local
|
||||||
-- save outer state
|
-- save outer state
|
||||||
local lastkw, lastkwpos, numerrors = g_lastkw, g_lastkwpos, g_numerrors
|
local lastkw, lastkwpos, numerrors = g_lastkw, g_lastkwpos, g_numerrors
|
||||||
local newlineidxs = g_newlineidxs
|
local newlineidxs = g_newlineidxs
|
||||||
|
local curcode = g_curcode
|
||||||
|
|
||||||
g_ifnestlevel = 0
|
g_ifseqlevel = 0
|
||||||
|
g_curcode = {}
|
||||||
|
g_file_code[g_filename] = g_curcode
|
||||||
|
|
||||||
-- set up new state
|
-- set up new state
|
||||||
-- TODO: pack into one "parser state" table?
|
-- TODO: pack into one "parser state" table?
|
||||||
|
@ -1346,6 +1464,7 @@ function parse(contents) -- local
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
g_curcode = curcode
|
||||||
g_recurslevel = g_recurslevel-1
|
g_recurslevel = g_recurslevel-1
|
||||||
|
|
||||||
-- restore outer state
|
-- restore outer state
|
||||||
|
@ -1357,27 +1476,53 @@ end
|
||||||
|
|
||||||
if (not _EDUKE32_LUNATIC) then
|
if (not _EDUKE32_LUNATIC) then
|
||||||
--- stand-alone
|
--- stand-alone
|
||||||
|
local debug = require("debug")
|
||||||
|
|
||||||
for argi=1,#arg do
|
for argi=1,#arg do
|
||||||
local filename = arg[argi]
|
local filename = arg[argi]
|
||||||
|
|
||||||
g_recurslevel = -1
|
g_recurslevel = -1
|
||||||
g_badids = {}
|
g_badids = {}
|
||||||
|
|
||||||
reset_labels()
|
reset_labels()
|
||||||
reset_gamedata()
|
reset_gamedata()
|
||||||
|
reset_codegen()
|
||||||
|
|
||||||
g_numerrors = 0
|
g_numerrors = 0
|
||||||
|
|
||||||
g_directory = filename:match("(.*/)") or ""
|
g_directory = filename:match("(.*/)") or ""
|
||||||
filename = filename:sub(#g_directory+1, -1)
|
filename = filename:sub(#g_directory+1, -1)
|
||||||
|
|
||||||
|
-- NOTE: xpcall isn't useful here since the traceback won't give us
|
||||||
|
-- anything inner to the lpeg.match call
|
||||||
local ok, msg = pcall(do_include_file, g_directory, filename)
|
local ok, msg = pcall(do_include_file, g_directory, filename)
|
||||||
|
|
||||||
if (not ok) then
|
if (not ok) then
|
||||||
if (g_lastkwpos ~= nil) then
|
if (g_lastkwpos ~= nil) then
|
||||||
printf("LAST KEYWORD POSITION: %s, %s", linecolstr(g_lastkwpos), g_lastkw)
|
printf("LAST KEYWORD POSITION: %s, %s", linecolstr(g_lastkwpos), g_lastkw)
|
||||||
end
|
end
|
||||||
print(msg)
|
print(msg)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
-- TEMP
|
||||||
|
local n, numl = 0, 0
|
||||||
|
local ll = {[1]=0, [2]=0, [3]=0}
|
||||||
|
for id,numlookups in pairs(TEMP_numlookups) do
|
||||||
|
numl = numl+numlookups
|
||||||
|
n = n+1
|
||||||
|
if (numlookups<=3) then
|
||||||
|
ll[numlookups] = ll[numlookups]+1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
printf("avg. lookups: %f (%d %d %d)", numl/n, ll[1],ll[2],ll[3])
|
||||||
|
|
||||||
|
--[[
|
||||||
|
for filename,codetab in pairs(g_file_code) do
|
||||||
|
io.stderr:write(string.format("-- GENERATED CODE (%s):\n", filename))
|
||||||
|
io.stderr:write(table.concat(flatten_codetab(codetab), "\n"))
|
||||||
|
io.stderr:write("\n")
|
||||||
|
end
|
||||||
|
--]]
|
||||||
end
|
end
|
||||||
else
|
else
|
||||||
--- embedded
|
--- embedded
|
||||||
|
|
Loading…
Reference in a new issue