f54064df07
var.key pushes the value indexed by key in var onto the stack quotes also work for assignment numerical keys do not work yet
742 lines
17 KiB
Lua
742 lines
17 KiB
Lua
-- CC0/Unlicense Emilia 2020
|
|
|
|
local tlang = {}
|
|
|
|
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)
|
|
return list[value] ~= nil
|
|
end
|
|
|
|
-- state
|
|
--[[
|
|
{
|
|
locals = {},
|
|
stack = {},
|
|
builtins = {},
|
|
code_stack = {},
|
|
wait_target = float,
|
|
paused = f/t,
|
|
nextpop = f/t
|
|
}
|
|
--]]
|
|
|
|
-- program counter
|
|
--[[
|
|
sg = 0/1,
|
|
pos = int/string,
|
|
elem = int
|
|
--]]
|
|
|
|
function tlang.boolean_to_number(b)
|
|
if b then
|
|
return 1
|
|
else
|
|
return 0
|
|
end
|
|
end
|
|
|
|
function tlang.number_to_boolean(n)
|
|
if n ~= 0 then
|
|
return true
|
|
else
|
|
return false
|
|
end
|
|
end
|
|
|
|
-- convert a lua value into a tlang literal
|
|
function tlang.value_to_tlang(value)
|
|
local t = type(value)
|
|
if t == "string" then
|
|
return {type = "string", value = value}
|
|
elseif t == "number" then
|
|
return {type = "number", value = value}
|
|
elseif t == "boolean" then
|
|
return {type = "number", value = tlang.boolean_to_number(value)}
|
|
elseif t == "table" then
|
|
local map = {}
|
|
|
|
for k, v in pairs(value) do
|
|
map[k] = tlang.value_to_tlang(v)
|
|
end
|
|
|
|
return {type = "map", value = map}
|
|
end
|
|
end
|
|
|
|
-- convert a tlang literal to a lua value
|
|
function tlang.tlang_to_value(tl)
|
|
if type(tl) ~= "table" then
|
|
return
|
|
end
|
|
|
|
if tl.type == "map" then
|
|
local o = {}
|
|
|
|
for k, v in pairs(tl.value) do
|
|
o[k] = tlang.tlang_to_value(v)
|
|
end
|
|
|
|
return o
|
|
else
|
|
return tl.value
|
|
end
|
|
end
|
|
|
|
local literals = {
|
|
"quote",
|
|
"code",
|
|
"map",
|
|
"string",
|
|
"number"
|
|
}
|
|
|
|
|
|
function tlang.call(state, target)
|
|
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
|
|
|
|
state.locals[#state.locals + 1] = {vars = {}, pc = target}
|
|
end
|
|
|
|
function tlang.call_tos(state)
|
|
tlang.call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
end
|
|
|
|
function tlang.call_var(state, name)
|
|
tlang.call(state, {sg = 1, pos = name, elem = 1})
|
|
end
|
|
|
|
function tlang.call_builtin(state, name)
|
|
local f = state.builtins[name]
|
|
f(state)
|
|
end
|
|
|
|
function tlang.call_var_or_builtin(state, name)
|
|
if in_keys(name, state.builtins) then
|
|
tlang.call_builtin(state, name)
|
|
else
|
|
tlang.call_var(state, name)
|
|
end
|
|
end
|
|
|
|
function tlang.push_values(state, vals)
|
|
for i, v in ipairs(vals) do
|
|
tlang.push(state, v)
|
|
end
|
|
end
|
|
|
|
function tlang.lua_call_tos(state, ...)
|
|
tlang.push_values(state, {...})
|
|
tlang.call_tos(state)
|
|
end
|
|
|
|
function tlang.lua_call_var(state, name, ...)
|
|
tlang.push_values(state, {...})
|
|
tlang.call_var(state, name)
|
|
end
|
|
|
|
local function find_var_pos(state, name)
|
|
local slen = #state.locals
|
|
|
|
for i = 1, slen do
|
|
local v = state.locals[slen + 1 - i]
|
|
if in_keys(name, v.vars) then
|
|
return slen + 1 - i
|
|
end
|
|
end
|
|
end
|
|
|
|
function tlang.near_access(state, name)
|
|
local n = find_var_pos(state, name)
|
|
if n then
|
|
return state.locals[n].vars[name]
|
|
end
|
|
end
|
|
|
|
function tlang.map_access_assign(state, index, start, assign)
|
|
local container
|
|
local curtab
|
|
|
|
if start then
|
|
container = start
|
|
elseif index[1] == "" and #index > 1 then
|
|
curtab = state.stack[#state.stack].value
|
|
else
|
|
local pos = find_var_pos(state, index[1])
|
|
-- assignments can go at the current scope
|
|
if assign then
|
|
pos = pos or #state.locals
|
|
end
|
|
|
|
container = state.locals[pos].vars
|
|
end
|
|
|
|
if not container and not curtab then
|
|
return
|
|
end
|
|
|
|
if #index == 1 then
|
|
if assign then
|
|
container[index[1]] = assign
|
|
return
|
|
else
|
|
return container[index[1]]
|
|
end
|
|
end
|
|
|
|
curtab = curtab or container[index[1]].value
|
|
|
|
for idx = 2, #index - 1 do
|
|
curtab = curtab[index[idx]]
|
|
|
|
if not curtab then
|
|
return nil
|
|
end
|
|
|
|
curtab = curtab.value
|
|
end
|
|
|
|
if assign then
|
|
curtab[index[#index]] = assign
|
|
else
|
|
return curtab[index[#index]]
|
|
end
|
|
end
|
|
|
|
function tlang.near_access_indexed(state, index)
|
|
return tlang.map_access_assign(state, index)
|
|
end
|
|
|
|
function tlang.near_assign_indexed(state, index, value)
|
|
tlang.map_access_assign(state, index, nil, value)
|
|
end
|
|
|
|
function tlang.global_assign(state, name, value)
|
|
state.locals[0].vars[name] = value
|
|
end
|
|
|
|
function tlang.local_assign(state, name, value)
|
|
state.locals[#state.locals].vars[name] = value
|
|
end
|
|
|
|
function tlang.near_assign(state, name, value)
|
|
local n = find_var_pos(state, name)
|
|
if n then
|
|
state.locals[n].vars[name] = value
|
|
else
|
|
state.locals[#state.locals].vars[name] = value
|
|
end
|
|
end
|
|
|
|
function tlang.get_pc(state)
|
|
return state.locals[#state.locals].pc
|
|
end
|
|
|
|
local function accesspc(state, pc)
|
|
local code
|
|
if pc.sg == 0 then -- stack
|
|
code = state.code_stack[pc.pos]
|
|
elseif pc.sg == 1 then -- global
|
|
code = tlang.near_access_indexed(state, pc.pos)
|
|
end
|
|
|
|
if code then
|
|
return code.value[pc.elem]
|
|
end
|
|
end
|
|
|
|
function tlang.increment_pc(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)
|
|
if state.locals[#state.locals].nextpop then
|
|
local pc = tlang.get_pc(state)
|
|
|
|
-- allows for finished states to be used in calls
|
|
if #state.locals == 1 then
|
|
return nil
|
|
end
|
|
|
|
state.locals[#state.locals] = nil
|
|
|
|
-- pop code stack
|
|
if pc.sg == 0 then
|
|
state.code_stack[pc.pos] = nil
|
|
end
|
|
|
|
return getnext(state)
|
|
end
|
|
|
|
local current
|
|
if not state.locals[#state.locals].nextpop then
|
|
state.current_pc = tlang.get_pc(state)
|
|
current = accesspc(state, state.current_pc)
|
|
end
|
|
|
|
local incd = tlang.increment_pc(state, tlang.get_pc(state))
|
|
if not incd then
|
|
state.locals[#state.locals].nextpop = true
|
|
else
|
|
state.locals[#state.locals].pc = incd
|
|
end
|
|
|
|
return current
|
|
end
|
|
|
|
-- doesn't support jumping out of scope yet
|
|
function tlang.set_next_pc(state, pc)
|
|
-- this probably causes issues when jumping outside scope
|
|
state.locals[#state.locals].nextpop = nil
|
|
|
|
state.locals[#state.locals].pc = pc
|
|
end
|
|
|
|
function tlang.peek_raw(state)
|
|
return state.stack[#state.stack]
|
|
end
|
|
|
|
function tlang.pop_raw(state)
|
|
local tos = tlang.peek_raw(state)
|
|
state.stack[#state.stack] = nil
|
|
return tos
|
|
end
|
|
|
|
function tlang.push_raw(state, value)
|
|
state.stack[#state.stack + 1] = value
|
|
end
|
|
|
|
function tlang.peek(state)
|
|
return tlang.tlang_to_value(tlang.peek_raw(state))
|
|
end
|
|
|
|
function tlang.pop(state)
|
|
return tlang.tlang_to_value(tlang.pop_raw(state))
|
|
end
|
|
|
|
function tlang.push(state, value)
|
|
tlang.push_raw(state, tlang.value_to_tlang(value))
|
|
end
|
|
|
|
local function statepeek_type(state, t)
|
|
local tos = tlang.peek_raw(state)
|
|
|
|
if tos.type == t then
|
|
return tos
|
|
else
|
|
return nil -- ERROR
|
|
end
|
|
end
|
|
|
|
local function statepop_type(state, t)
|
|
local tos = tlang.peek_raw(state)
|
|
|
|
if tos.type == t then
|
|
return tlang.pop_raw(state)
|
|
else
|
|
return nil -- ERROR
|
|
end
|
|
end
|
|
|
|
local function statepop_num(state)
|
|
return statepop_type(state, "number")
|
|
end
|
|
|
|
local function statepush_num(state, number)
|
|
tlang.push_raw(state, {type = "number", value = number})
|
|
end
|
|
|
|
|
|
|
|
tlang.builtins = {}
|
|
|
|
function tlang.builtins.run(state)
|
|
tlang.call_tos(state)
|
|
end
|
|
|
|
tlang.builtins["="] = function(state)
|
|
local name = statepop_type(state, "quote")
|
|
local value = tlang.pop_raw(state)
|
|
|
|
tlang.near_assign_indexed(state, name.value, value)
|
|
end
|
|
|
|
function tlang.unary(func)
|
|
return function(state)
|
|
local tos = tlang.pop_raw(state)
|
|
if tos.type == "number" then
|
|
statepush_num(state, func(tos.value))
|
|
elseif tos.type == "quote" then
|
|
local n = tlang.near_access(state, tos.value)
|
|
tlang.near_assign_indexed(state, tos.value, {type = "number", value = func(n.value)})
|
|
end
|
|
end
|
|
end
|
|
|
|
function tlang.binary(func)
|
|
return function(state)
|
|
local tos = statepop_num(state)
|
|
local tos1 = statepop_num(state)
|
|
|
|
statepush_num(state, func(tos1.value, tos.value))
|
|
end
|
|
end
|
|
|
|
tlang.builtins["--"] = tlang.unary(function(v)
|
|
return v - 1
|
|
end)
|
|
|
|
tlang.builtins["++"] = tlang.unary(function(v)
|
|
return v + 1
|
|
end)
|
|
|
|
tlang.builtins["!"] = tlang.unary(function(v)
|
|
return tlang.boolean_to_number(not tlang.number_to_boolean(v))
|
|
end)
|
|
|
|
tlang.builtins["+"] = tlang.binary(function(v1, v2)
|
|
return v1 + v2
|
|
end)
|
|
|
|
tlang.builtins["-"] = tlang.binary(function(v1, v2)
|
|
return v1 - v2
|
|
end)
|
|
|
|
tlang.builtins["*"] = tlang.binary(function(v1, v2)
|
|
return v1 * v2
|
|
end)
|
|
|
|
tlang.builtins["/"] = tlang.binary(function(v1, v2)
|
|
return v1 / v2
|
|
end)
|
|
|
|
tlang.builtins["%"] = tlang.binary(function(v1, v2)
|
|
return v1 % v2
|
|
end)
|
|
|
|
tlang.builtins["=="] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(v1 == v2)
|
|
end)
|
|
|
|
tlang.builtins["!="] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(v1 ~= v2)
|
|
end)
|
|
|
|
tlang.builtins[">="] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(v1 >= v2)
|
|
end)
|
|
|
|
tlang.builtins["<="] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(v1 <= v2)
|
|
end)
|
|
|
|
tlang.builtins[">"] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(v1 > v2)
|
|
end)
|
|
|
|
tlang.builtins["<"] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(v1 < v2)
|
|
end)
|
|
|
|
tlang.builtins["&&"] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(
|
|
tlang.number_to_boolean(v1) and tlang.number_to_boolean(v2))
|
|
end)
|
|
|
|
tlang.builtins["||"] = tlang.binary(function(v1, v2)
|
|
return tlang.boolean_to_number(
|
|
tlang.number_to_boolean(v1) or tlang.number_to_boolean(v2))
|
|
end)
|
|
|
|
tlang.builtins["if"] = function(state)
|
|
local tos = statepop_type(state, "code")
|
|
local tos1 = tlang.pop_raw(state)
|
|
|
|
if tos1.type == "number" then
|
|
if tos1.value ~= 0 then
|
|
tlang.push_raw(state, tos)
|
|
tlang.call_tos(state)
|
|
end
|
|
elseif tos1.type == "code" then
|
|
local tos2 = statepop_num(state)
|
|
if tos2.value ~= 0 then
|
|
tlang.push_raw(state, tos1)
|
|
tlang.call_tos(state)
|
|
else
|
|
tlang.push_raw(state, tos)
|
|
tlang.call_tos(state)
|
|
end
|
|
end
|
|
end
|
|
|
|
function tlang.builtins.print(state)
|
|
local value = tlang.pop_raw(state)
|
|
|
|
if minetest then
|
|
local message = "[tlang] " .. tostring(value.value)
|
|
minetest.display_chat_message(message)
|
|
minetest.log("info", message)
|
|
else
|
|
print(value.value)
|
|
end
|
|
end
|
|
|
|
function tlang.builtins.dup(state)
|
|
tlang.push_raw(state, tlang.peek_raw(state))
|
|
end
|
|
|
|
function tlang.builtins.popoff(state)
|
|
state.stack[#state.stack] = nil
|
|
end
|
|
|
|
function tlang.builtins.wait(state)
|
|
local tos = statepop_type(state, "number")
|
|
state.wait_target = os.clock() + tos.value
|
|
end
|
|
|
|
tlang.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 = tlang.pop_raw(state)
|
|
|
|
if tos.type == "code" then
|
|
state.locals[slen].loop_code = tos
|
|
elseif tos.type == "quote" then
|
|
state.locals[slen].loop_code = statepop_type(state, "code")
|
|
state.locals[slen].repeat_n = 0
|
|
state.locals[slen].loop_var = tos.value
|
|
end
|
|
end
|
|
|
|
if state.locals[slen].loop_var then
|
|
tlang.local_assign(state,
|
|
state.locals[slen].loop_var,
|
|
{type = "number", value = state.locals[slen].repeat_n})
|
|
state.locals[slen].repeat_n = state.locals[slen].repeat_n + 1
|
|
end
|
|
|
|
tlang.push_raw(state, state.locals[slen].loop_code)
|
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
|
|
|
tlang.call_tos(state)
|
|
end
|
|
|
|
tlang.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
|
|
tlang.push_raw(state, state.locals[slen].test_code)
|
|
tlang.set_next_pc(state, state.current_pc)
|
|
tlang.call_tos(state)
|
|
|
|
state.locals[slen].loop_stage = 1
|
|
-- stage 1, run while
|
|
elseif state.locals[slen].loop_stage == 1 then
|
|
local tos = tlang.pop_raw(state)
|
|
if tos and tos.value ~= 0 then
|
|
tlang.push_raw(state, state.locals[slen].loop_code)
|
|
tlang.set_next_pc(state, state.current_pc)
|
|
tlang.call_tos(state)
|
|
else
|
|
tlang.set_next_pc(state, state.current_pc)
|
|
state.locals[slen].broke = true
|
|
end
|
|
|
|
state.locals[slen].loop_stage = 0
|
|
end
|
|
end
|
|
|
|
tlang.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 = tlang.pop_raw(state)
|
|
local count
|
|
local block
|
|
|
|
if num_var.type == "quote" then
|
|
count = statepop_num(state)
|
|
state.locals[slen].loop_var = num_var.value
|
|
else
|
|
count = num_var
|
|
end
|
|
|
|
block = statepop_type(state, "code")
|
|
|
|
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
|
|
tlang.local_assign(state,
|
|
state.locals[slen].loop_var,
|
|
{type = "number", value = state.locals[slen].repeat_n})
|
|
end
|
|
|
|
tlang.push_raw(state, state.locals[slen].loop_code)
|
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
|
|
|
tlang.call_tos(state)
|
|
|
|
state.locals[slen].repeat_n = state.locals[slen].repeat_n + 1
|
|
else
|
|
tlang.set_next_pc(state, state.current_pc)
|
|
state.locals[slen].broke = true
|
|
end
|
|
end
|
|
|
|
tlang.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
|
|
|
|
tlang.builtins["return"] = function(state)
|
|
state.locals[#state.locals] = nil
|
|
end
|
|
|
|
tlang.builtins["args"] = function(state)
|
|
local vars = {}
|
|
local vari = 1
|
|
|
|
while true do
|
|
local n = tlang.pop_raw(state)
|
|
if n.type == "quote" then
|
|
vars[vari] = n.value
|
|
vari = vari + 1
|
|
elseif n.type == "number" and n.value == 0 then
|
|
break
|
|
else
|
|
return false
|
|
end
|
|
end
|
|
|
|
for i, v in ipairs(vars) do
|
|
tlang.local_assign(state, v, tlang.pop_raw(state))
|
|
end
|
|
end
|
|
|
|
|
|
-- returns:
|
|
-- true - more to do
|
|
-- nil - more to do but waiting
|
|
-- false - finished
|
|
-- string - error
|
|
function tlang.step(state)
|
|
if state.paused or (state.wait_target and os.clock() < state.wait_target) then
|
|
return nil
|
|
end
|
|
|
|
local cur = getnext(state)
|
|
|
|
if cur == nil then
|
|
state.finished = true
|
|
return false
|
|
else
|
|
state.finished = false
|
|
end
|
|
|
|
if in_list(cur.type, literals) then
|
|
state.stack[#state.stack + 1] = cur
|
|
elseif cur.type == "identifier" or cur.type == "symbol" then
|
|
local strname = cur.value
|
|
if type(cur.value) == "table" then
|
|
strname = cur.value[1]
|
|
end
|
|
|
|
if in_keys(strname, state.builtins) then
|
|
local f = state.builtins[strname]
|
|
f(state)
|
|
else
|
|
local var = tlang.near_access_indexed(state, cur.value)
|
|
if var == nil then
|
|
return "Undefined identifier: " .. table.concat(cur.value, ".")
|
|
elseif var.type == "code" then
|
|
tlang.call_var(state, cur.value)
|
|
else
|
|
state.stack[#state.stack + 1] = var
|
|
end
|
|
end
|
|
end
|
|
|
|
return true
|
|
end
|
|
|
|
return tlang
|