cron f54064df07 turtle/tlang: add . map indexing
var.key pushes the value indexed by key in var onto the stack
quotes also work for assignment
numerical keys do not work yet
2020-12-10 04:34:48 +00:00

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