nomsu/nomsu_compiler.lua

443 lines
14 KiB
Lua
Raw Normal View History

2018-06-19 01:27:32 -07:00
local lpeg = require('lpeg')
local R, P, S
R, P, S = lpeg.R, lpeg.P, lpeg.S
2018-06-19 01:27:32 -07:00
local re = require('re')
local List, Dict, Text
do
local _obj_0 = require('containers')
List, Dict, Text = _obj_0.List, _obj_0.Dict, _obj_0.Text
end
2018-06-19 01:27:32 -07:00
local insert, remove, concat
do
local _obj_0 = table
insert, remove, concat = _obj_0.insert, _obj_0.remove, _obj_0.concat
end
local unpack = unpack or table.unpack
local match, sub, gsub, format, byte, find
2018-06-19 01:27:32 -07:00
do
local _obj_0 = string
match, sub, gsub, format, byte, find = _obj_0.match, _obj_0.sub, _obj_0.gsub, _obj_0.format, _obj_0.byte, _obj_0.find
2018-06-19 01:27:32 -07:00
end
local LuaCode, Source
2018-06-19 01:27:32 -07:00
do
local _obj_0 = require("code_obj")
LuaCode, Source = _obj_0.LuaCode, _obj_0.Source
2018-06-19 01:27:32 -07:00
end
local SyntaxTree = require("syntax_tree")
local Importer, import_to_1_from, _1_forked
do
local _obj_0 = require('importer')
Importer, import_to_1_from, _1_forked = _obj_0.Importer, _obj_0.import_to_1_from, _obj_0._1_forked
end
table.map = function(t, fn)
return setmetatable((function()
local _accum_0 = { }
local _len_0 = 1
for _, v in ipairs(t) do
_accum_0[_len_0] = fn(v)
_len_0 = _len_0 + 1
end
return _accum_0
end)(), getmetatable(t))
2018-06-19 01:27:32 -07:00
end
local pretty_error = require("pretty_errors")
local compile_error
compile_error = function(tree, err_msg, hint)
if hint == nil then
hint = nil
end
local err_str = pretty_error({
title = "Compile error",
error = err_msg,
hint = hint,
source = tree:get_source_code(),
start = tree.source.start,
stop = tree.source.stop,
filename = tree.source.filename
2018-06-19 01:27:32 -07:00
})
return error(err_str, 0)
2018-06-19 01:27:32 -07:00
end
local math_expression = re.compile([[ (([*/^+-] / [0-9]+) " ")* [*/^+-] !. ]])
local compile_math_expression
compile_math_expression = function(compile, tree, ...)
local lua = LuaCode(tree.source)
for i, tok in ipairs(tree) do
if type(tok) == 'string' then
lua:append(tok)
else
local tok_lua = compile(tok)
if tok.type == "Action" then
tok_lua:parenthesize()
end
lua:append(tok_lua)
end
if i < #tree then
lua:append(" ")
end
end
return lua
2018-09-12 15:31:59 -07:00
end
2018-06-19 01:27:32 -07:00
local MAX_LINE = 80
local compile = setmetatable({
action = Importer({
[""] = function(compile, tree, fn, ...)
local lua = LuaCode(tree.source)
local fn_lua = compile(fn)
lua:append(fn_lua)
if not (fn_lua:text():match("^%(.*%)$") or fn_lua:text():match("^[_a-zA-Z][_a-zA-Z0-9.]*$")) then
lua:parenthesize()
end
lua:append("(")
for i = 1, select('#', ...) do
if i > 1 then
lua:append(", ")
end
lua:append(compile(select(i, ...)))
end
lua:append(")")
return lua
end,
["Lua"] = function(compile, tree, code)
if code.type ~= "Text" then
return LuaCode(code.source, "LuaCode(", tostring(code.source):as_lua(), ", ", compile(code), ")")
end
local add_bit_lua
add_bit_lua = function(lua, bit_lua)
local bit_leading_len = #(bit_lua:match("^[^\n]*"))
lua:append(lua:trailing_line_len() + bit_leading_len > MAX_LINE and ",\n " or ", ")
return lua:append(bit_lua)
end
local operate_on_text
operate_on_text = function(text)
local lua = LuaCode(text.source, "LuaCode(", tostring(text.source):as_lua())
for _index_0 = 1, #text do
local bit = text[_index_0]
if type(bit) == "string" then
add_bit_lua(lua, bit:as_lua())
elseif bit.type == "Text" then
add_bit_lua(lua, operate_on_text(bit))
else
add_bit_lua(lua, compile(bit))
end
end
lua:append(")")
return lua
2018-06-19 01:27:32 -07:00
end
return operate_on_text(code)
end,
["Lua value"] = function(compile, tree, code)
return compile.action["Lua"](compile, tree, code)
2018-06-19 01:27:32 -07:00
end,
["lua >"] = function(compile, tree, code)
if code.type ~= "Text" then
return tree
2018-06-19 01:27:32 -07:00
end
local operate_on_text
operate_on_text = function(text)
local lua = LuaCode(text.source)
for _index_0 = 1, #text do
local bit = text[_index_0]
if type(bit) == "string" then
lua:append(bit)
elseif bit.type == "Text" then
lua:append(operate_on_text(bit))
else
lua:append(compile(bit))
end
end
return lua
2018-06-19 01:27:32 -07:00
end
return operate_on_text(code)
end,
["= lua"] = function(compile, tree, code)
return compile.action["lua >"](compile, tree, code)
end,
["use"] = function(compile, tree, path)
return LuaCode(tree.source, "run_file_1_in(" .. tostring(compile(path)) .. ", _ENV)")
end,
["tests"] = function(compile, tree)
return LuaCode(tree.source, "TESTS")
end,
["test"] = function(compile, tree, body)
return LuaCode(tree.source, "TESTS[" .. tostring(tostring(tree.source):as_lua()) .. "] = ", body:as_lua())
2018-09-16 16:57:14 -07:00
end,
["is jit"] = function(compile, tree, code)
return LuaCode(tree.source, "jit")
end,
["Lua version"] = function(compile, tree, code)
return LuaCode(tree.source, "_VERSION")
end,
["nomsu environment"] = function(compile, tree)
return LuaCode(tree.source, "_ENV")
2018-06-19 01:27:32 -07:00
end
})
}, {
__import = function(self, other)
import_to_1_from(self.action, other.action)
end,
__call = function(compile, tree, force_value)
if force_value == nil then
force_value = false
end
2018-07-09 19:22:40 -07:00
if tree.version then
do
local get_version = compile.action[("Nomsu version"):as_lua_id()]
if get_version then
do
local upgrade = compile.action[("1 upgraded from 2 to"):as_lua_id()]
if upgrade then
tree = upgrade(tree, tree.version, get_version())
end
end
2018-07-09 19:22:40 -07:00
end
end
end
2018-06-19 01:27:32 -07:00
local _exp_0 = tree.type
if "Action" == _exp_0 then
local stub = tree.stub
local compile_action = compile.action[stub]
if not compile_action and math_expression:match(stub) then
compile_action = compile_math_expression
end
if compile_action and not tree.target then
local args
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #tree do
local arg = tree[_index_0]
if type(arg) ~= "string" then
_accum_0[_len_0] = arg
_len_0 = _len_0 + 1
2018-06-19 01:27:32 -07:00
end
end
args = _accum_0
end
local ret = compile_action(compile, tree, unpack(args))
if ret == nil then
local info = debug.getinfo(compile_action, "S")
local filename = Source:from_string(info.source).filename
compile_error(tree, "The compile-time action here (" .. tostring(stub) .. ") failed to return any value.", "Look at the implementation of (" .. tostring(stub) .. ") in " .. tostring(filename) .. ":" .. tostring(info.linedefined) .. " and make sure it's returning something.")
end
if not (SyntaxTree:is_instance(ret)) then
return ret
end
if ret ~= tree then
return compile(ret)
2018-06-19 01:27:32 -07:00
end
end
local lua = LuaCode(tree.source)
if tree.target then
local target_lua = compile(tree.target)
local target_text = target_lua:text()
if target_text:match("^%(.*%)$") or target_text:match("^[_a-zA-Z][_a-zA-Z0-9.]*$") then
lua:append(target_lua, ":")
else
lua:append("(", target_lua, "):")
end
end
lua:append((stub):as_lua_id(), "(")
2018-06-19 01:27:32 -07:00
local args = { }
for i, tok in ipairs(tree) do
local _continue_0 = false
repeat
if type(tok) == "string" then
_continue_0 = true
break
end
local arg_lua = compile(tok, true)
2018-06-19 01:27:32 -07:00
insert(args, arg_lua)
_continue_0 = true
until true
if not _continue_0 then
break
end
end
lua:concat_append(args, ", ")
lua:append(")")
return lua
elseif "EscapedNomsu" == _exp_0 then
local lua = LuaCode(tree.source, "SyntaxTree{")
local needs_comma, i = false, 1
local as_lua
as_lua = function(x)
if type(x) == 'number' then
return tostring(x)
elseif SyntaxTree:is_instance(x) then
return compile(x)
else
return x:as_lua()
end
end
for k, v in pairs((SyntaxTree:is_instance(tree[1]) and tree[1].type == "EscapedNomsu" and tree) or tree[1]) do
if needs_comma then
lua:append(", ")
else
needs_comma = true
end
if k == i then
i = i + 1
elseif type(k) == 'string' and match(k, "[_a-zA-Z][_a-zA-Z0-9]*") then
lua:append(k, "= ")
else
lua:append("[", as_lua(k), "]= ")
end
if k == "source" then
lua:append(tostring(v):as_lua())
else
lua:append(as_lua(v))
2018-06-19 01:27:32 -07:00
end
end
lua:append("}")
return lua
2018-06-19 01:27:32 -07:00
elseif "Block" == _exp_0 then
if not force_value then
local lua = LuaCode(tree.source)
lua:concat_append((function()
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #tree do
local line = tree[_index_0]
_accum_0[_len_0] = compile(line)
_len_0 = _len_0 + 1
end
return _accum_0
end)(), "\n")
return lua
else
local lua = LuaCode(tree.source)
lua:append("((function()")
for i, line in ipairs(tree) do
lua:append("\n ", compile(line))
end
lua:append("\nend)())")
return lua
end
2018-06-19 01:27:32 -07:00
elseif "Text" == _exp_0 then
local lua = LuaCode(tree.source)
2018-06-19 01:27:32 -07:00
local string_buffer = ""
for i, bit in ipairs(tree) do
local _continue_0 = false
repeat
if type(bit) == "string" then
string_buffer = string_buffer .. bit
_continue_0 = true
break
end
if string_buffer ~= "" then
if #lua.bits > 0 then
lua:append("..")
end
lua:append(string_buffer:as_lua())
2018-06-19 01:27:32 -07:00
string_buffer = ""
end
local bit_lua = compile(bit)
2018-06-19 01:27:32 -07:00
if #lua.bits > 0 then
lua:append("..")
end
if bit.type ~= "Text" then
bit_lua = LuaCode(bit.source, "tostring(", bit_lua, ")")
2018-06-19 01:27:32 -07:00
end
lua:append(bit_lua)
_continue_0 = true
until true
if not _continue_0 then
break
end
end
if string_buffer ~= "" or #lua.bits == 0 then
if #lua.bits > 0 then
lua:append("..")
end
lua:append(string_buffer:as_lua())
2018-06-19 01:27:32 -07:00
end
if #lua.bits > 1 then
lua:parenthesize()
end
return lua
elseif "List" == _exp_0 or "Dict" == _exp_0 then
local lua = LuaCode(tree.source, tostring(tree.type) .. "{")
local i = 1
local sep = ''
while i <= #tree do
local item = tree[i]
if item.type == "Block" then
break
2018-06-19 01:27:32 -07:00
end
lua:append(sep)
if item.type == "Comment" then
lua:append(compile(item), "\n")
sep = ''
else
local item_lua = compile(item)
lua:append(item_lua)
sep = ', '
2018-06-19 01:27:32 -07:00
end
i = i + 1
end
2018-06-19 01:27:32 -07:00
lua:append("}")
if i <= #tree then
lua = LuaCode(tree.source, "(function()\n local it = ", lua)
while i <= #tree do
lua:append("\n ")
if tree[i].type == 'Block' or tree[i].type == 'Comment' then
lua:append(compile(tree[i]))
elseif tree[i].type == "DictEntry" then
lua:append("it[ ", compile(tree[i][1]), "] = ", (tree[i][2] and compile(tree[i][2]) or "true"))
else
lua:append("it:add(", compile(tree[i]), ")")
end
i = i + 1
end
lua:append("\n return it\nend)()")
end
2018-06-19 01:27:32 -07:00
return lua
elseif "DictEntry" == _exp_0 then
local key, value = tree[1], tree[2]
local key_lua = compile(key)
local value_lua = value and compile(value) or LuaCode(key.source, "true")
local key_str = match(key_lua:text(), [=[^["']([a-zA-Z_][a-zA-Z0-9_]*)['"]$]=])
if key_str and key_str:is_lua_id() then
2018-06-19 01:27:32 -07:00
return LuaCode(tree.source, key_str, "=", value_lua)
elseif sub(key_lua:text(), 1, 1) == "[" then
2018-06-19 01:27:32 -07:00
return LuaCode(tree.source, "[ ", key_lua, "]=", value_lua)
else
return LuaCode(tree.source, "[", key_lua, "]=", value_lua)
end
elseif "IndexChain" == _exp_0 then
local lua = compile(tree[1])
local first_char = sub(lua:text(), 1, 1)
2018-06-19 01:27:32 -07:00
if first_char == "{" or first_char == '"' or first_char == "[" then
lua:parenthesize()
end
for i = 2, #tree do
local key = tree[i]
local key_lua = compile(key)
local key_lua_str = key_lua:text()
local lua_id = match(key_lua_str, "^['\"]([a-zA-Z_][a-zA-Z0-9_]*)['\"]$")
if lua_id and lua_id:is_lua_id() then
lua:append("." .. tostring(lua_id))
elseif sub(key_lua_str, 1, 1) == '[' then
lua:append("[ ", key_lua, " ]")
else
lua:append("[", key_lua, "]")
2018-06-19 01:27:32 -07:00
end
end
return lua
elseif "Number" == _exp_0 then
return LuaCode(tree.source, tostring(tree[1]))
2018-06-19 01:27:32 -07:00
elseif "Var" == _exp_0 then
return LuaCode(tree.source, (tree[1]):as_lua_id())
elseif "FileChunks" == _exp_0 then
return error("Can't convert FileChunks to a single block of lua, since each chunk's " .. "compilation depends on the earlier chunks")
2018-09-12 15:31:59 -07:00
elseif "Comment" == _exp_0 then
return LuaCode(tree.source, "")
elseif "Error" == _exp_0 then
return error("Can't compile errors")
2018-06-19 01:27:32 -07:00
else
return error("Unknown type: " .. tostring(tree.type))
end
end
})
return compile