2020-11-05 23:20:28 +00:00
|
|
|
-- CC0/Unlicense Emilia 2020
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
local function in_list(value, list)
|
|
|
|
for k, v in ipairs(list) do
|
|
|
|
if v == value then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
local function in_keys(value, list)
|
|
|
|
for k, v in pairs(list) do
|
|
|
|
if k == value then
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
-- state
|
|
|
|
--[[
|
|
|
|
{
|
|
|
|
locals = {},
|
|
|
|
stack = {},
|
2020-11-03 07:06:59 +00:00
|
|
|
builtins = {},
|
|
|
|
code_stack = {},
|
|
|
|
wait_target = int,
|
|
|
|
nextpop = f/t
|
2020-11-01 02:55:54 +00:00
|
|
|
}
|
|
|
|
--]]
|
|
|
|
|
|
|
|
-- program counter
|
|
|
|
--[[
|
|
|
|
sg = 0/1,
|
|
|
|
pos = int/string,
|
|
|
|
elem = int,
|
|
|
|
wait_target = float
|
|
|
|
--]]
|
|
|
|
|
|
|
|
local literals = {
|
|
|
|
"quote",
|
|
|
|
"code",
|
|
|
|
"map",
|
|
|
|
"string",
|
|
|
|
"number"
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
local function call(state, target)
|
2020-11-03 07:06:59 +00:00
|
|
|
if target.sg == 0 then
|
|
|
|
state.code_stack[#state.code_stack + 1] = state.stack[target.pos]
|
|
|
|
table.remove(state.stack, target.pos)
|
|
|
|
target.pos = #state.code_stack
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
state.locals[#state.locals + 1] = {pc = target}
|
|
|
|
end
|
|
|
|
|
2020-11-05 02:52:48 +00:00
|
|
|
local function find_var_pos(state, name)
|
2020-11-04 06:46:43 +00:00
|
|
|
local slen = #state.locals
|
|
|
|
|
|
|
|
for i = 1, slen do
|
|
|
|
local v = state.locals[slen + 1 - i]
|
2020-11-05 02:52:48 +00:00
|
|
|
if in_keys("v" .. name, v) then
|
|
|
|
return slen + 1 - i
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-05 02:52:48 +00:00
|
|
|
local function access(state, name)
|
|
|
|
local n = find_var_pos(state, name)
|
|
|
|
if n then
|
|
|
|
return state.locals[n]["v" .. name]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
local function gassign(state, name, value)
|
|
|
|
state.locals[0]["v" .. name] = value
|
|
|
|
end
|
|
|
|
|
2020-11-05 02:52:48 +00:00
|
|
|
local function lassign(state, name, value)
|
2020-11-01 02:55:54 +00:00
|
|
|
state.locals[#state.locals]["v" .. name] = value
|
|
|
|
end
|
|
|
|
|
2020-11-05 02:52:48 +00:00
|
|
|
local function assign(state, name, value)
|
|
|
|
local n = find_var_pos(state, name)
|
|
|
|
if n then
|
|
|
|
state.locals[n]["v" .. name] = value
|
|
|
|
else
|
|
|
|
state.locals[#state.locals]["v" .. name] = value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
local function getpc(state)
|
|
|
|
return state.locals[#state.locals].pc
|
|
|
|
end
|
|
|
|
|
|
|
|
local function accesspc(state, pc)
|
|
|
|
local code
|
|
|
|
if pc.sg == 0 then -- stack
|
2020-11-03 07:06:59 +00:00
|
|
|
code = state.code_stack[pc.pos]
|
2020-11-01 02:55:54 +00:00
|
|
|
elseif pc.sg == 1 then -- global
|
|
|
|
code = access(state, pc.pos)
|
|
|
|
end
|
|
|
|
|
|
|
|
if code then
|
|
|
|
return code.value[pc.elem]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function incpc(state, pc)
|
|
|
|
local next_pc = {sg = pc.sg, pos = pc.pos, elem = pc.elem + 1}
|
|
|
|
|
|
|
|
if accesspc(state, next_pc) then
|
|
|
|
return next_pc
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function getnext(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
if state.locals[#state.locals].nextpop then
|
2020-11-01 02:55:54 +00:00
|
|
|
state.locals[#state.locals] = nil
|
|
|
|
if #state.locals == 0 then
|
|
|
|
return nil
|
|
|
|
end
|
2020-11-03 07:06:59 +00:00
|
|
|
|
|
|
|
-- pop code stack
|
|
|
|
local pc = getpc(state)
|
|
|
|
if pc.sg == 0 then
|
|
|
|
state.code_stack[pc.pos] = nil
|
|
|
|
end
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
state.current_pc = getpc(state)
|
|
|
|
local current = accesspc(state, state.current_pc)
|
2020-11-01 02:55:54 +00:00
|
|
|
|
|
|
|
local incd = incpc(state, getpc(state))
|
|
|
|
state.locals[#state.locals].pc = incd
|
|
|
|
if not incd then
|
2020-11-04 02:57:48 +00:00
|
|
|
state.locals[#state.locals].nextpop = true
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return current
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
local function statepeek(state)
|
|
|
|
return state.stack[#state.stack]
|
|
|
|
end
|
|
|
|
|
|
|
|
local function statepop(state)
|
|
|
|
local tos = statepeek(state)
|
|
|
|
state.stack[#state.stack] = nil
|
|
|
|
return tos
|
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
local function statepeek_type(state, t)
|
|
|
|
local tos = statepeek(state)
|
|
|
|
|
|
|
|
if tos.type == t then
|
|
|
|
return tos
|
|
|
|
else
|
|
|
|
return nil -- ERROR
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
local function statepop_type(state, t)
|
|
|
|
local tos = statepeek(state)
|
|
|
|
|
|
|
|
if tos.type == t then
|
|
|
|
return statepop(state)
|
|
|
|
else
|
|
|
|
return nil -- ERROR
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
local function statepop_num(state)
|
|
|
|
return statepop_type(state, "number")
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
local function statepush(state, value)
|
|
|
|
state.stack[#state.stack + 1] = value
|
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
local function statepush_num(state, number)
|
|
|
|
statepush(state, {type = "number", value = number})
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
local builtins = {}
|
|
|
|
|
2020-11-03 07:06:59 +00:00
|
|
|
function builtins.run(state)
|
|
|
|
call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
builtins["="] = function(state)
|
2020-11-04 03:03:53 +00:00
|
|
|
local name = statepop_type(state, "quote")
|
2020-11-01 02:55:54 +00:00
|
|
|
local value = statepop(state)
|
|
|
|
|
|
|
|
assign(state, name.value, value)
|
|
|
|
end
|
|
|
|
|
2020-11-05 03:12:18 +00:00
|
|
|
local function unary(func)
|
|
|
|
return function(state)
|
|
|
|
local tos = statepop(state)
|
|
|
|
if tos.type == "number" then
|
|
|
|
statepush_num(state, func(tos.value))
|
|
|
|
elseif tos.type == "quote" then
|
|
|
|
local n = access(state, tos.value)
|
|
|
|
assign(state, tos.value, {type = "number", value = func(n.value)})
|
|
|
|
end
|
|
|
|
end
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-05 03:12:18 +00:00
|
|
|
local function binary(func)
|
|
|
|
return function(state)
|
|
|
|
local tos = statepop_num(state)
|
|
|
|
local tos1 = statepop_num(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-05 03:12:18 +00:00
|
|
|
statepush_num(state, func(tos.value, tos1.value))
|
|
|
|
end
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
local function boolnum(b)
|
|
|
|
if b then
|
|
|
|
return 1
|
|
|
|
else
|
|
|
|
return 0
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-05 03:12:18 +00:00
|
|
|
local function numbool(n)
|
|
|
|
if n ~= 0 then
|
|
|
|
return true
|
|
|
|
else
|
|
|
|
return false
|
|
|
|
end
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-05 03:12:18 +00:00
|
|
|
builtins["--"] = unary(function(v)
|
|
|
|
return v - 1
|
|
|
|
end)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-05 03:12:18 +00:00
|
|
|
builtins["++"] = unary(function(v)
|
|
|
|
return v + 1
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["!"] = unary(function(v)
|
|
|
|
return boolnum(not numbool(v))
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["+"] = binary(function(v1, v2)
|
|
|
|
return v1 + v2
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["-"] = binary(function(v1, v2)
|
|
|
|
return v1 - v2
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["*"] = binary(function(v1, v2)
|
|
|
|
return v1 * v2
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["/"] = binary(function(v1, v2)
|
|
|
|
return v1 / v2
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["%"] = binary(function(v1, v2)
|
|
|
|
return v1 % v2
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["=="] = binary(function(v1, v2)
|
|
|
|
return boolnum(v1 == v2)
|
|
|
|
end)
|
|
|
|
|
|
|
|
builtins["!="] = binary(function(v1, v2)
|
|
|
|
return boolnum(v1 ~= v2)
|
|
|
|
end)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
|
|
|
builtins["if"] = function(state)
|
|
|
|
local tos = statepop_type(state, "code")
|
|
|
|
local tos1 = statepop(state)
|
|
|
|
|
|
|
|
if tos1.type == "number" then
|
|
|
|
if tos1.value ~= 0 then
|
|
|
|
statepush(state, tos)
|
|
|
|
call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
end
|
|
|
|
end
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function builtins.print(state)
|
|
|
|
local value = statepop(state)
|
|
|
|
|
2020-11-05 19:53:35 +00:00
|
|
|
if minetest then
|
2020-11-05 19:55:55 +00:00
|
|
|
local message = "[tlang] " .. tostring(value.value)
|
|
|
|
minetest.display_chat_message(message)
|
2020-11-05 23:17:41 +00:00
|
|
|
minetest.log("info", message)
|
2020-11-05 19:53:35 +00:00
|
|
|
else
|
|
|
|
print(value.value)
|
|
|
|
end
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
function builtins.dup(state)
|
|
|
|
statepush(state, statepeek(state))
|
|
|
|
end
|
|
|
|
|
|
|
|
function builtins.popoff(state)
|
|
|
|
state.stack[#state.stack] = nil
|
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
function builtins.wait(state)
|
|
|
|
local tos = statepop_type(state, "number")
|
|
|
|
state.wait_target = os.clock() + tos.value
|
|
|
|
end
|
|
|
|
|
|
|
|
builtins["forever"] = function(state)
|
|
|
|
local slen = #state.locals
|
|
|
|
|
|
|
|
if state.locals[slen].broke == true then
|
|
|
|
state.locals[slen].broke = nil
|
|
|
|
state.locals[slen].loop_code = nil
|
|
|
|
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if state.locals[slen].loop_code == nil then
|
|
|
|
local tos = statepop_type(state, "code")
|
|
|
|
|
|
|
|
state.locals[slen].loop_code = tos
|
|
|
|
end
|
|
|
|
|
|
|
|
statepush(state, state.locals[slen].loop_code)
|
|
|
|
|
|
|
|
state.locals[slen].pc = state.current_pc
|
|
|
|
|
|
|
|
call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
end
|
|
|
|
|
2020-11-04 19:19:36 +00:00
|
|
|
builtins["while"] = function(state)
|
|
|
|
local slen = #state.locals
|
|
|
|
|
|
|
|
if state.locals[slen].broke == true then
|
|
|
|
state.locals[slen].broke = nil
|
|
|
|
state.locals[slen].loop_code = nil
|
|
|
|
state.locals[slen].test_code = nil
|
|
|
|
state.locals[slen].loop_stage = nil
|
|
|
|
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if state.locals[slen].loop_code == nil then
|
|
|
|
local while_block = statepop_type(state, "code")
|
|
|
|
local test_block = statepop_type(state, "code")
|
|
|
|
|
|
|
|
state.locals[slen].test_code = test_block
|
|
|
|
state.locals[slen].loop_code = while_block
|
|
|
|
state.locals[slen].loop_stage = 0
|
|
|
|
end
|
|
|
|
|
|
|
|
-- stage 0, run test
|
|
|
|
if state.locals[slen].loop_stage == 0 then
|
|
|
|
statepush(state, state.locals[slen].test_code)
|
|
|
|
state.locals[slen].pc = state.current_pc
|
|
|
|
call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
|
|
|
|
state.locals[slen].loop_stage = 1
|
|
|
|
-- stage 1, run while
|
|
|
|
elseif state.locals[slen].loop_stage == 1 then
|
|
|
|
local tos = statepop(state)
|
|
|
|
if tos and tos.value ~= 0 then
|
|
|
|
statepush(state, state.locals[slen].loop_code)
|
|
|
|
state.locals[slen].pc = state.current_pc
|
|
|
|
call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
else
|
|
|
|
state.locals[slen].pc = state.current_pc
|
|
|
|
state.locals[slen].broke = true
|
|
|
|
end
|
|
|
|
|
|
|
|
state.locals[slen].loop_stage = 0
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-05 06:49:55 +00:00
|
|
|
builtins["repeat"] = function(state)
|
|
|
|
local slen = #state.locals
|
|
|
|
|
|
|
|
if state.locals[slen].broke == true then
|
|
|
|
state.locals[slen].broke = nil
|
|
|
|
state.locals[slen].loop_code = nil
|
|
|
|
state.locals[slen].repeat_count = nil
|
|
|
|
state.locals[slen].repeat_n = nil
|
|
|
|
state.locals[slen].loop_var = nil
|
|
|
|
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if state.locals[slen].loop_code == nil then
|
|
|
|
local num_var = statepop(state)
|
|
|
|
local count
|
|
|
|
local block
|
|
|
|
|
|
|
|
if num_var.type == "quote" then
|
|
|
|
count = statepop_num(state)
|
|
|
|
block = statepop_type(state, "code")
|
|
|
|
state.locals[slen].loop_var = num_var.value
|
|
|
|
else
|
|
|
|
count = num_var
|
|
|
|
block = statepop_type(state, "code")
|
|
|
|
end
|
|
|
|
|
|
|
|
state.locals[slen].loop_code = block
|
|
|
|
state.locals[slen].repeat_count = count.value
|
|
|
|
state.locals[slen].repeat_n = 0
|
|
|
|
end
|
|
|
|
|
|
|
|
if state.locals[slen].repeat_n ~= state.locals[slen].repeat_count then
|
|
|
|
if state.locals[slen].loop_var then
|
|
|
|
lassign(state,
|
|
|
|
state.locals[slen].loop_var,
|
|
|
|
{type = "number", value = state.locals[slen].repeat_n})
|
|
|
|
end
|
|
|
|
|
|
|
|
statepush(state, state.locals[slen].loop_code)
|
|
|
|
|
|
|
|
state.locals[slen].pc = state.current_pc
|
|
|
|
|
|
|
|
call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
|
|
|
|
state.locals[slen].repeat_n = state.locals[slen].repeat_n + 1
|
|
|
|
else
|
|
|
|
state.locals[slen].pc = state.current_pc
|
|
|
|
state.locals[slen].broke = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
builtins["break"] = function(state)
|
|
|
|
local slen = #state.locals
|
|
|
|
local pos = 0
|
|
|
|
local found = false
|
|
|
|
|
|
|
|
-- find highest loop_code
|
|
|
|
-- slen - i to perform basically bitwise inverse
|
|
|
|
-- it allows it to count down the list effectively
|
|
|
|
for i = 1, slen do
|
|
|
|
if state.locals[slen + 1 - i].loop_code then
|
|
|
|
pos = slen + 1 - i
|
|
|
|
found = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if found then
|
|
|
|
-- pop the top layers
|
|
|
|
for i = pos + 1, #state.locals do
|
|
|
|
state.locals[i] = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
-- break in the lower layer
|
|
|
|
state.locals[#state.locals].broke = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
builtins["return"] = function(state)
|
|
|
|
state.locals[#state.locals] = nil
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
|
|
|
|
-- returns:
|
|
|
|
-- true - more to do
|
|
|
|
-- nil - more to do but waiting
|
|
|
|
-- false - finished
|
|
|
|
-- string - error
|
|
|
|
local step = function(state)
|
|
|
|
if state.wait_target and os.clock() < state.wait_target then
|
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
|
|
|
local cur = getnext(state)
|
|
|
|
|
|
|
|
if cur == nil then
|
|
|
|
return false
|
|
|
|
elseif in_list(cur.type, literals) then
|
|
|
|
state.stack[#state.stack + 1] = cur
|
|
|
|
elseif cur.type == "identifier" or cur.type == "symbol" then
|
|
|
|
if in_keys(cur.value, state.builtins) then
|
|
|
|
local f = state.builtins[cur.value]
|
|
|
|
f(state)
|
|
|
|
else
|
|
|
|
local var = access(state, cur.value)
|
|
|
|
if var == nil then
|
|
|
|
return "Undefined identifier: " .. cur.value
|
|
|
|
elseif var.type == "code" then
|
|
|
|
call(state, {sg = 1, pos = cur.value, elem = 1})
|
|
|
|
else
|
|
|
|
state.stack[#state.stack + 1] = var
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return true
|
|
|
|
end
|
|
|
|
|
|
|
|
return builtins, gassign, step
|