Lunatic translator: ai/move/action.

This exposes some problems in the default/in-the-wild CONs. As usual,
we'll have to retrofit sensible semantics :rolleyes:.

git-svn-id: https://svn.eduke32.com/eduke32@2765 1a8010ca-5511-0410-912e-c29ae57300e0
This commit is contained in:
helixhorned 2012-06-17 19:45:43 +00:00
parent 23e43886a8
commit cbb71c088a

View file

@ -91,29 +91,33 @@ local function parse_number(numstr)
end end
local LABEL_DEFINE = 1 local LABEL = { MOVE=2, AI=3, ACTION=5, [2]="move", [3]="ai", [5]="action" }
--local LABEL_STATE = 2
--local LABEL_ACTOR = 4
local LABEL_ACTION = 8
local LABEL_AI = 16
local LABEL_MOVE = 32
local MOVE_NO_ = {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_ }
-- will contain:
-- * scalar numbers: `define'd values
-- * tables of length 2, 3, 5: move, ai, action definitions (respectively)
-- - move: { hvel <num>, vvel <num> }
-- - ai: { action <label str>, move <label str, or scalar 0 or 1>, flags <num> }
-- - action: { startframe, numframes, viewtype, incval, delay } (all <num>)
local g_labeldef = {} local g_labeldef = {}
local g_labeltype = {}
local function reset_labels() local function reset_labels()
g_labeldef = { NO=0 } g_labeldef = { NO=0 } -- NO is also a valid `move', `ai' or `action'
g_labeltype = { NO=LABEL_DEFINE+LABEL_ACTION+LABEL_AI+LABEL_MOVE }
for i=1,#con.labels do for i=1,#con.labels do
for label, val in pairs(con.labels[i]) do for label, val in pairs(con.labels[i]) do
g_labeldef[label] = val g_labeldef[label] = val
g_labeltype[label] = LABEL_DEFINE
end end
end end
end end
local function lookup_defined_label(identifier) -- XXX: error reports give wrong numbers if e.g. happened in "actor" def
-- (since it runs when fully parsed, i.e. at "enda")
local function lookup_defined_label(maybe_minus_str, identifier)
local num = g_labeldef[identifier] local num = g_labeldef[identifier]
if (num == nil) then if (num == nil) then
@ -121,29 +125,28 @@ local function lookup_defined_label(identifier)
return -inf -- return a number for type cleanness return -inf -- return a number for type cleanness
end end
return num if (type(num) ~= "number") then
end errprintf("label \"%s\" is not a `define'd value", identifier)
return -inf
local function do_define_label(identifier, idornum)
-- TODO: label types
local num
if (type(idornum)=="number") then
num = idornum
else
assert(idornum ~= nil)
num = lookup_defined_label(idornum)
if (num == -inf) then
return
end
end end
local oldnum = g_labeldef[identifier] return (maybe_minus_str=="" and 1 or -1) * num
if (oldnum) then end
local function do_define_label(identifier, num)
local oldval = g_labeldef[identifier]
if (oldval) then
if (type(oldval) == "table") then
errprintf("refusing to overwrite `%s' label \"%s\" with a `define'd value",
LABEL[#oldval], identifier)
return
end
-- con.labels[...]: don't warn for wrong PROJ_ redefinitions -- con.labels[...]: don't warn for wrong PROJ_ redefinitions
if (oldnum ~= num and con.labels[2][identifier]==nil) then if (oldval ~= num and con.labels[2][identifier]==nil) then
warnprintf("label \"%s\" not redefined with new value %d (old: %d)", warnprintf("label \"%s\" not redefined with new value %d (old: %d)",
identifier, num, oldnum) identifier, num, oldval)
end end
return return
end end
@ -151,6 +154,82 @@ local function do_define_label(identifier, idornum)
g_labeldef[identifier] = num g_labeldef[identifier] = num
end end
local function check_move_literal(num)
if (num~=0 and num~=1) then
errprintf("literal `move' number must be either 0 or 1")
return MOVE_NO_
end
-- Both move 0 and 1 have hvel and vvel 0, but they must not compare equal
-- for 'ifmove'. 0.1 will be truncated to 0 when passing to the game.
-- XXX: this is still wrong, we need an ID
return {0.1*num, 0.1*num}
end
local function check_action_or_ai_literal(labeltype, num)
if (num~=0) then
errprintf("literal `%s' number must be 0", LABEL[labeltype])
return LABEL_NO[labeltype]
end
return LABEL_NO[labeltype]
end
local function lookup_composite(labeltype, identifier)
if (identifier=="NO") then
-- NO is a special case and is valid for move, ai, action
return LABEL_NO[labeltype]
end
local val = g_labeldef[identifier]
if (val == nil) then
errprintf("label \"%s\" is not defined", identifier)
return LABEL_NO[labeltype] -- return a value of the expected type for cleanness
end
if (type(val)~="table" or #val~=labeltype) then
errprintf("label \"%s\" is not a%s `%s' value", identifier,
labeltype==LABEL.MOVE and "" or "n", LABEL[labeltype])
return LABEL_NO[labeltype]
end
return val
end
local function do_define_composite(labeltype, identifier, ...)
local oldval = g_labeldef[identifier]
if (oldval) then
if (type(oldval) ~= "table" or #val~=labeltype) then
errprintf("refusing to overwrite `%s' label \"%s\" with a `%s' value",
type(oldval)=="number" and "define" or LABEL[#oldval],
identifier, LABEL[labeltype])
return
end
warnprintf("duplicate `%s' definition of \"%s\" ignored",
LABEL[labeltype], identifier)
return
end
local val = {...}
for i=#val+1,labeltype do
val[i] = LABEL_NO[labeltype][i]
end
if (labeltype == LABEL.AI) then
-- OR together the flags
for i=#val,LABEL.AI+1, -1 do
val[LABEL.AI] = bit.bor(val[LABEL.AI], val[i])
val[i] = nil
end
end
g_labeldef[identifier] = val
end
local function parse(contents) end -- fwd-decl local function parse(contents) end -- fwd-decl
local function do_include_file(dirname, filename) end local function do_include_file(dirname, filename) end
@ -169,26 +248,27 @@ else
fd, msg = io.open(dirname..filename) fd, msg = io.open(dirname..filename)
if (fd == nil) then if (fd == nil) then
printf("Fatal error: couldn't open %s", msg) printf("[%d] Fatal error: couldn't open %s", g_recurslevel, msg)
g_numerrors = inf g_numerrors = inf
return return
end end
end end
printf("%s[%d] Parsing file \"%s\"", (g_recurslevel==-1 and "\n---- ") or "",
g_recurslevel+1, dirname..filename);
local contents = fd:read("*all") local contents = fd:read("*all")
fd:close() fd:close()
if (contents == nil) then if (contents == nil) then
-- maybe that file name turned out to be a directory or other -- maybe that file name turned out to be a directory or other
-- special file accidentally -- special file accidentally
printf("Fatal error: couldn't read from \"%s\"", dirname..filename) printf("[%d] Fatal error: couldn't read from \"%s\"",
g_recurslevel, dirname..filename)
g_numerrors = inf g_numerrors = inf
return return
end end
printf("%s[%d] Parsing file \"%s\"", (g_recurslevel==-1 and "\n---- ") or "",
g_recurslevel+1, dirname..filename);
local oldfilename = g_filename local oldfilename = g_filename
g_filename = filename g_filename = filename
parse(contents) parse(contents)
@ -347,6 +427,9 @@ local t_identifier = Var("t_identifier")
-- This one matches keywords, too: -- This one matches keywords, too:
local t_identifier_all = Var("t_identifier_all") local t_identifier_all = Var("t_identifier_all")
local t_define = Var("t_define") local t_define = Var("t_define")
local t_move = Var("t_move")
local t_ai = Var("t_ai")
local t_action = Var("t_action")
-- NOTE: no chance to whitespace and double quotes in filenames: -- NOTE: no chance to whitespace and double quotes in filenames:
local t_filename = lpeg.C((anychar-Set(" \t\r\n\""))^1) local t_filename = lpeg.C((anychar-Set(" \t\r\n\""))^1)
local t_newline_term_str = match_until(anychar, newline) local t_newline_term_str = match_until(anychar, newline)
@ -379,7 +462,9 @@ local function n_defines(n) -- works well only for small n
end end
local D, R, W, I = -1, -2, -3, -4; local D, R, W, I, AC, MV, AI = -1, -2, -3, -4, -5, -6, -7
local TOKEN_PATTERN = { [D]=t_define, [R]=t_rvar, [W]=t_wvar, [I]=t_identifier,
[AC]=t_action, [MV]=t_move, [AI]=t_ai }
-- Generic command pattern, types given by varargs. -- Generic command pattern, types given by varargs.
-- The command name to be matched is attached later. -- The command name to be matched is attached later.
@ -393,18 +478,7 @@ local function cmd(...)
local vartypes = {...} local vartypes = {...}
for i=1,#vartypes do for i=1,#vartypes do
assert(vartypes[i] < 0) pat = pat * sp1 * assert(TOKEN_PATTERN[vartypes[i]])
if (vartypes[i] == D) then
pat = pat * sp1 * t_define
elseif (vartypes[i] == R) then
pat = pat * sp1 * t_rvar
elseif (vartypes[i] == W) then
pat = pat * sp1 * t_wvar
else -- I
assert(vartypes[i] == I)
pat = pat * sp1 * t_identifier
end
end end
return pat return pat
@ -463,9 +537,18 @@ local Co = {
gamearray = cmd(I,D), gamearray = cmd(I,D),
--- 5. Top level commands that are also run-time commands --- 5. Top level commands that are also run-time commands
action = sp1 * t_identifier * (sp1 * t_define)^-5 / function(id) do_define_label(id, 0) end, -- TEMP -- startframe, numframes, viewtype, incval, delay:
ai = sp1 * t_identifier * (sp1 * t_define)^0 / function(id) do_define_label(id, 0) end, -- TEMP action = sp1 * t_identifier * (sp1 * t_define)^-5 /
move = sp1 * t_identifier * (sp1 * t_define)^-2 / function(id) do_define_label(id, 0) end, -- TEMP function(...) do_define_composite(LABEL.ACTION, ...) end,
-- action, move, flags...:
ai = sp1 * t_identifier * (sp1 * t_action *
(sp1 * t_move * (sp1 * t_define)^0)^-1
)^-1 /
function(...) do_define_composite(LABEL.AI, ...) end,
move = sp1 * t_identifier * (sp1 * t_define)^-2 / -- hvel, vvel
function(...) do_define_composite(LABEL.MOVE, ...) end,
--- 6. Deprecated TLCs --- 6. Deprecated TLCs
betaname = newline_term_string, betaname = newline_term_string,
@ -593,9 +676,9 @@ local Ci = {
--- 3. Actors --- 3. Actors
-- These three need more attention (different kind of labels; move -- These three need more attention (different kind of labels; move
-- additionally may accept 0 or 1): -- additionally may accept 0 or 1):
action = cmd(D), action = cmd(AC),
ai = cmd(D), ai = cmd(AI),
move = sp1 * t_define * (sp1 * t_define)^0, move = sp1 * t_move * (sp1 * t_define)^0,
cactor = cmd(D), cactor = cmd(D),
count = cmd(D), count = cmd(D),
@ -829,9 +912,9 @@ local Ci = {
local Cif = { local Cif = {
-- XXX: ai, action, move/def labels -- XXX: ai, action, move/def labels
ifai = cmd(D), ifai = cmd(AI),
ifaction = cmd(D), ifaction = cmd(AC),
ifmove = cmd(D), ifmove = cmd(MV),
ifrnd = cmd(D), ifrnd = cmd(D),
ifpdistl = cmd(D), ifpdistl = cmd(D),
@ -1066,15 +1149,23 @@ local stmt_list = Var("stmt_list")
local stmt_list_or_eps = (stmt_list * sp1)^-1 local stmt_list_or_eps = (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 all three: <name/tilenum> [<strength> [<action> [<move> [<ai>... ]]]] -- common to actor and useractor: <name/tilenum> [<strength> [<action> [<move> [<flags>... ]]]]
local common_actor_end = sp1 * t_define * sp1 * (t_define * sp1)^0 * stmt_list_or_eps * "enda" local common_actor_end = sp1 * t_define *
(sp1 * t_define *
(sp1 * t_action *
(sp1 * t_move *
(sp1 * t_define)^0
)^-1
)^-1
)^-1
* 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 = common_actor_end,
-- eventloadactor (...) -- eventloadactor <name/tilenum>
eventloadactor = common_actor_end, eventloadactor = sp1 * t_define * sp1 * stmt_list_or_eps * "enda",
-- useractor <actortype> (...) -- useractor <actortype> (...)
useractor = sp1 * t_define * common_actor_end, useractor = sp1 * t_define * common_actor_end,
@ -1135,7 +1226,14 @@ 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 = (t_maybe_minus * t_identifier/lookup_defined_label) + t_number, -- TODO: minus t_define = (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_number/check_move_literal,
t_ai = (t_identifier/function(id) lookup_composite(LABEL.AI, id) end) +
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_number/function(num) check_action_or_ai_literal(LABEL.ACTION, num) end,
t_arrayexp = t_identifier * arraypat * memberpat^-1, t_arrayexp = t_identifier * arraypat * memberpat^-1,