2018-05-15 18:55:55 -07:00
|
|
|
#
|
2018-01-11 02:07:37 -08:00
|
|
|
This File contains actions for making actions and compile-time actions and some helper
|
|
|
|
functions to make that easier.
|
2017-09-21 00:10:26 -07:00
|
|
|
|
2018-06-23 00:57:31 -07:00
|
|
|
lua> "NOMSU_CORE_VERSION = 1"
|
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
lua> ".."
|
2018-06-18 18:10:59 -07:00
|
|
|
nomsu.COMPILE_ACTIONS["% -> %"] = function(nomsu, tree, \%args, \%body)
|
2018-06-20 15:22:03 -07:00
|
|
|
local lua = LuaCode.Value(tree.source, "function(")
|
2018-06-18 18:10:59 -07:00
|
|
|
if AST.is_syntax_tree(\%args, "Action") then \%args = \%args:get_args() end
|
|
|
|
local lua_args = table.map(\%args, function(a) return AST.is_syntax_tree(a) and tostring(nomsu:compile(a)) or a end)
|
|
|
|
lua:concat_append(lua_args, ", ")
|
|
|
|
local body_lua = AST.is_syntax_tree(\%body) and nomsu:compile(\%body):as_statements("return ") or \%body
|
|
|
|
body_lua:remove_free_vars(lua_args)
|
|
|
|
body_lua:declare_locals()
|
|
|
|
lua:append(")\n ", body_lua, "\nend")
|
|
|
|
return lua
|
|
|
|
end
|
|
|
|
|
|
|
|
lua> ".."
|
|
|
|
nomsu.COMPILE_ACTIONS["compile as %"] = function(nomsu, tree, \%action)
|
|
|
|
local lua = LuaCode.Value(tree.source, "nomsu.COMPILE_ACTIONS[", repr(\%action.stub), "](")
|
|
|
|
local lua_args = table.map(\%action:get_args(), function(a) return nomsu:compile(a) end)
|
|
|
|
table.insert(lua_args, 1, "nomsu")
|
|
|
|
table.insert(lua_args, 2, "tree")
|
|
|
|
lua:concat_append(lua_args, ", ")
|
|
|
|
lua:append(")")
|
|
|
|
return lua
|
|
|
|
end
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
lua> ".."
|
|
|
|
nomsu.COMPILE_ACTIONS["compile % to %"] = function(nomsu, tree, \%actions, \%body)
|
|
|
|
local \%args = {"nomsu", "tree", unpack(table.map(\%actions[1]:get_args(), function(a) return tostring(nomsu:compile(a)) end))}
|
|
|
|
local lua = LuaCode(tree.source, "nomsu.COMPILE_ACTIONS[", repr(\%actions[1].stub),
|
|
|
|
"] = ", \(compile as: %args -> %body))
|
|
|
|
for i=2,#\%actions do
|
|
|
|
local alias = \%actions[i]
|
|
|
|
local \%alias_args = {"nomsu", "tree", unpack(table.map(alias:get_args(), function(a) return tostring(nomsu:compile(a)) end))}
|
|
|
|
lua:append("\nnomsu.COMPILE_ACTIONS[", repr(alias.stub), "] = ")
|
|
|
|
if utils.equivalent(\%args, \%alias_args) then
|
|
|
|
lua:append("nomsu.COMPILE_ACTIONS[", repr(\%actions[1].stub), "]")
|
2018-06-18 15:44:29 -07:00
|
|
|
else
|
2018-06-18 18:10:59 -07:00
|
|
|
lua:append("function(")
|
|
|
|
lua:concat_append(\%alias_args, ", ")
|
|
|
|
lua:append(")\n return nomsu.COMPILE_ACTIONS[", repr(\%actions[1].stub), "](")
|
|
|
|
lua:concat_append(\%args, ", ")
|
|
|
|
lua:append(")\nend")
|
2018-06-14 21:59:25 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-06-18 18:10:59 -07:00
|
|
|
return lua
|
|
|
|
end
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2018-06-23 00:57:31 -07:00
|
|
|
compile [call %fn with %args] to
|
|
|
|
lua> ".."
|
|
|
|
local lua = LuaCode.Value(tree.source, nomsu:compile(\%fn), "(")
|
|
|
|
lua:concat_append(table.map(\%args, function(a) return nomsu:compile(a) end), ", ")
|
|
|
|
lua:append(")")
|
|
|
|
return lua
|
|
|
|
|
2018-06-21 19:12:59 -07:00
|
|
|
compile [using %defs compile %body] to
|
|
|
|
lua> ".."
|
|
|
|
local lua = LuaCode(tree.source)
|
|
|
|
lua:append(
|
|
|
|
"local old_nomsu = nomsu\n",
|
|
|
|
"local nomsu = table.fork(old_nomsu, {COMPILE_ACTIONS=table.fork(old_nomsu.COMPILE_ACTIONS)})")
|
|
|
|
lua:append(nomsu:compile(\%defs))
|
|
|
|
lua:append("\n")
|
|
|
|
lua:append("local ret = nomsu:compile(tree)\n")
|
|
|
|
lua:append("return ret")
|
|
|
|
nomsu = table.fork(nomsu, {tree=\%body})
|
|
|
|
local output = nomsu:run_lua(lua)
|
|
|
|
return output
|
|
|
|
|
2018-06-18 18:10:59 -07:00
|
|
|
compile [local action %actions %body] to
|
|
|
|
lua> ".."
|
|
|
|
local fn_name = "A"..string.as_lua_id(\%actions[1].stub)
|
|
|
|
local \%args = table.map(\%actions[1]:get_args(), function(a) return tostring(nomsu:compile(a)) end)
|
|
|
|
local lua = LuaCode(tree.source, fn_name, " = ", \(compile as: %args -> %body))
|
|
|
|
lua:add_free_vars({fn_name})
|
|
|
|
for i=2,#\%actions do
|
|
|
|
local alias = \%actions[i]
|
|
|
|
local alias_name = "A"..string.as_lua_id(alias.stub)
|
|
|
|
lua:add_free_vars({alias_name})
|
|
|
|
local \%alias_args = table.map(alias:get_args(), function(a) return tostring(nomsu:compile(a)) end)
|
|
|
|
lua:append("\n", alias_name, " = ")
|
|
|
|
if utils.equivalent(\%args, \%alias_args) then
|
|
|
|
lua:append(fn_name)
|
|
|
|
else
|
|
|
|
lua:append("function(")
|
|
|
|
lua:concat_append(\%alias_args, ", ")
|
|
|
|
lua:append(")\n return ", fn_name, "(")
|
|
|
|
lua:concat_append(\%args, ", ")
|
|
|
|
lua:append(")\nend")
|
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
|
|
|
return lua
|
|
|
|
|
2018-06-18 18:10:59 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
compile [action %actions %body] to
|
|
|
|
lua> ".."
|
|
|
|
local lua = \(compile as: local action %actions %body)
|
|
|
|
lua:remove_free_vars(table.map(\%actions, function(a) return "A"..string.as_lua_id(a.stub) end))
|
|
|
|
return lua
|
|
|
|
|
2018-06-21 19:12:59 -07:00
|
|
|
compile [action %action] to
|
|
|
|
Lua value "A\(%action.stub as lua id)"
|
|
|
|
|
2018-06-18 18:10:59 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
compile [parse %actions as %body] to
|
|
|
|
lua> ".."
|
2018-06-18 15:44:29 -07:00
|
|
|
local replacements = {}
|
|
|
|
for i,arg in ipairs(\%actions[1]:get_args()) do
|
|
|
|
replacements[arg[1]] = tostring(nomsu:compile(arg))
|
|
|
|
end
|
|
|
|
local function make_tree(t)
|
|
|
|
if not AST.is_syntax_tree(t) then
|
|
|
|
return repr(t)
|
|
|
|
elseif t.type ~= 'Var' then
|
2018-06-18 18:10:59 -07:00
|
|
|
local args = {repr(tostring(t.source)), unpack(table.map(t, make_tree))}
|
2018-06-18 15:44:29 -07:00
|
|
|
return t.type.."("..table.concat(args, ", ")..")"
|
|
|
|
elseif replacements[t[1]] then
|
|
|
|
return replacements[t[1]]
|
|
|
|
else
|
2018-06-18 18:10:59 -07:00
|
|
|
return t.type.."("..repr(tostring(t.source))..", "..repr(t[1].." \\0").."..string.format('%X', __MANGLE_INDEX))"
|
2018-06-12 20:06:33 -07:00
|
|
|
end
|
2018-06-18 15:44:29 -07:00
|
|
|
end
|
2018-06-18 18:10:59 -07:00
|
|
|
local \%new_body = LuaCode(\%body.source,
|
|
|
|
"__MANGLE_INDEX = (__MANGLE_INDEX or 0) + 1",
|
|
|
|
"\nlocal tree = ", make_tree(\%body),
|
2018-06-23 00:57:31 -07:00
|
|
|
"\nlocal lua = nomsu:compile(tree); return lua")
|
|
|
|
local ret = \(compile as: compile %actions to %new_body)
|
|
|
|
return ret
|
2018-06-18 15:44:29 -07:00
|
|
|
|
2018-06-18 18:39:21 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2017-12-11 17:53:23 -08:00
|
|
|
|
2018-05-27 18:28:23 -07:00
|
|
|
compile [remove action %action] to
|
|
|
|
Lua ".."
|
2018-06-14 21:59:25 -07:00
|
|
|
A\(=lua "string.as_lua_id(\(%action.stub))") = nil
|
2017-11-01 19:59:44 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%tree as nomsu]
|
|
|
|
=lua "nomsu:tree_to_nomsu(\%tree)"
|
2018-05-16 18:12:56 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%tree as inline nomsu]
|
|
|
|
=lua "nomsu:tree_to_nomsu(\%tree, true)"
|
2018-05-16 18:12:56 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%tree as lua]
|
|
|
|
=lua "nomsu:compile(\%tree)"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%tree as lua expr]
|
|
|
|
lua> ".."
|
|
|
|
local lua = nomsu:compile(\%tree)
|
|
|
|
if not lua.is_value then
|
|
|
|
nomsu:compile_error(\%tree, "Invalid thing to convert to lua expr:\n%s")
|
|
|
|
end
|
|
|
|
return lua
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%tree as lua statements]
|
|
|
|
=lua "nomsu:compile(\%tree):as_statements()"
|
2018-06-12 13:56:15 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%tree as lua return]
|
|
|
|
=lua "nomsu:compile(\%tree):as_statements('return ')"
|
2018-06-14 21:59:25 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
action [%var as lua identifier, %var as lua id]
|
|
|
|
lua> ".."
|
|
|
|
if type(\%var) == 'string' then return string.as_lua_id(\%var)
|
|
|
|
elseif \%var.type == 'Var' then return string.as_lua_id(\%var[1])
|
|
|
|
elseif \%var.type == 'Action' then return "A"..string.as_lua_id(\%var.stub)
|
|
|
|
end
|
|
|
|
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-04-19 17:23:44 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [%tree with %t -> %replacement] to
|
|
|
|
Lua value ".."
|
|
|
|
\(%tree as lua expr):map(function(\(%t as lua expr))
|
|
|
|
\(%replacement as lua return)
|
|
|
|
end)
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-06-23 00:57:31 -07:00
|
|
|
compile [%tree with vars %v] to
|
|
|
|
Lua value ".."
|
|
|
|
\(%tree as lua expr):map(function(t)
|
|
|
|
local replacements = \(%v as lua expr)
|
|
|
|
if t.type == "Var" then
|
|
|
|
return replacements[t[1]]
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [declare locals in %code] to
|
|
|
|
Lua value "\(%code as lua expr):declare_locals()"
|
2017-12-04 17:54:52 -08:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [declare locals %locals in %code] to
|
|
|
|
Lua value "\(%code as lua expr):declare_locals(\(%locals as lua expr))"
|
2018-01-10 20:45:03 -08:00
|
|
|
|
2018-06-21 19:12:59 -07:00
|
|
|
compile [add free vars %vars to %code] to
|
|
|
|
Lua "\(%code as lua expr):add_free_vars(\(%vars as lua expr));"
|
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [remove free vars %vars from %code] to
|
|
|
|
Lua "\(%code as lua expr):remove_free_vars(\(%vars as lua expr));"
|
2018-01-11 18:51:21 -08:00
|
|
|
|
2018-06-20 15:22:03 -07:00
|
|
|
compile [%lua <-write %code, to %lua write %code] to: Lua "\(%lua as lua expr):append(\(%code as lua expr));"
|
2018-02-06 22:06:39 -08:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [quote %s] to
|
|
|
|
Lua value ".."
|
|
|
|
('"'..\(%s as lua expr):gsub("\\\\", "\\\\\\\\"):gsub("\n","\\\\n"):gsub('"', '\\\\"')..'"')
|
|
|
|
compile [type of %obj] to: Lua value "type(\(%obj as lua expr))"
|
|
|
|
|
2018-06-23 00:57:31 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2018-06-18 15:44:29 -07:00
|
|
|
|
|
|
|
compile [parse %text] to
|
|
|
|
Lua value ".."
|
|
|
|
nomsu:parse(NomsuCode("\("\(%text.source)")", \(%text as lua expr)))
|
|
|
|
|
|
|
|
compile [run %nomsu_code] to
|
|
|
|
Lua value "nomsu:run(NomsuCode(\(quote "\(%nomsu_code.source)"), \(%nomsu_code as lua expr)))"
|
|
|
|
|
|
|
|
action [run tree %tree, %tree as value]
|
|
|
|
lua> ".."
|
2018-06-23 00:57:31 -07:00
|
|
|
return nomsu:run(\%tree)
|
2018-06-14 21:59:25 -07:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [compile %block, compiled %block, %block compiled] to
|
|
|
|
Lua value "nomsu:compile(\(%block as lua))"
|
2018-01-25 17:34:49 -08:00
|
|
|
|
2018-06-18 15:44:29 -07:00
|
|
|
# Return statement is wrapped in a do..end block because Lua is unhappy if you
|
|
|
|
put code after a return statement, unless you wrap it in a block.
|
|
|
|
compile [return] to: Lua "do return; end"
|
|
|
|
compile [return %return_value] to: Lua "do return \(%return_value as lua expr); end"
|
2018-01-11 18:51:21 -08:00
|
|
|
|
|
|
|
# Literals
|
2018-06-18 15:44:29 -07:00
|
|
|
compile [yes] to: Lua value "true"
|
|
|
|
compile [no] to: Lua value "false"
|
|
|
|
compile [nothing, nil, null] to: Lua value "nil"
|
2018-06-23 00:57:31 -07:00
|
|
|
compile [Nomsu syntax version] to: Lua value "NOMSU_SYNTAX_VERSION"
|
|
|
|
compile [Nomsu compiler version] to: Lua value "NOMSU_COMPILER_VERSION"
|
|
|
|
compile [core version] to: Lua value "NOMSU_CORE_VERSION"
|
|
|
|
compile [lib version] to: Lua value "NOMSU_LIB_VERSION"
|
2018-01-11 18:51:21 -08:00
|
|
|
|
2018-06-23 00:57:31 -07:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
action [Nomsu version]
|
|
|
|
use "lib/version.nom"
|
|
|
|
return "\(Nomsu syntax version).\(Nomsu compiler version).\(core version).\(lib version)"
|