2020-11-05 23:20:28 +00:00
|
|
|
-- CC0/Unlicense Emilia 2020
|
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
local tlang = ...
|
2020-11-06 00:13:21 +00:00
|
|
|
|
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)
|
2020-11-22 06:52:20 +00:00
|
|
|
return list[value] ~= nil
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
-- state
|
|
|
|
--[[
|
|
|
|
{
|
|
|
|
locals = {},
|
|
|
|
stack = {},
|
2020-11-03 07:06:59 +00:00
|
|
|
builtins = {},
|
|
|
|
code_stack = {},
|
2020-11-06 16:38:52 +00:00
|
|
|
wait_target = float,
|
2020-11-08 17:35:55 +00:00
|
|
|
paused = f/t,
|
2020-11-03 07:06:59 +00:00
|
|
|
nextpop = f/t
|
2020-11-01 02:55:54 +00:00
|
|
|
}
|
|
|
|
--]]
|
|
|
|
|
|
|
|
-- program counter
|
|
|
|
--[[
|
|
|
|
sg = 0/1,
|
|
|
|
pos = int/string,
|
2020-11-06 16:38:52 +00:00
|
|
|
elem = int
|
2020-11-01 02:55:54 +00:00
|
|
|
--]]
|
|
|
|
|
2020-11-06 16:38:52 +00:00
|
|
|
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
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
-- 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}
|
2020-11-06 16:38:52 +00:00
|
|
|
elseif t == "boolean" then
|
|
|
|
return {type = "number", value = tlang.boolean_to_number(value)}
|
2020-11-06 00:13:21 +00:00
|
|
|
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)
|
2020-11-06 16:20:04 +00:00
|
|
|
if type(tl) ~= "table" then
|
2020-11-06 00:13:21 +00:00
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
if tl.type == "map" then
|
|
|
|
local o = {}
|
|
|
|
|
2020-11-06 16:20:04 +00:00
|
|
|
for k, v in pairs(tl.value) do
|
|
|
|
o[k] = tlang.tlang_to_value(v)
|
2020-11-06 00:13:21 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return o
|
|
|
|
else
|
|
|
|
return tl.value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
local literals = {
|
|
|
|
"quote",
|
|
|
|
"code",
|
|
|
|
"map",
|
|
|
|
"string",
|
|
|
|
"number"
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.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-12-10 04:34:48 +00:00
|
|
|
state.locals[#state.locals + 1] = {vars = {}, pc = target}
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.call_tos(state)
|
|
|
|
tlang.call(state, {sg = 0, pos = #state.stack, elem = 1})
|
|
|
|
end
|
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
function tlang.call_var(state, name)
|
2020-12-11 02:59:40 +00:00
|
|
|
if type(name) ~= "table" then
|
|
|
|
name = {name}
|
|
|
|
end
|
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
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
|
|
|
|
|
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-12-10 04:34:48 +00:00
|
|
|
if in_keys(name, v.vars) then
|
2020-11-05 02:52:48 +00:00
|
|
|
return slen + 1 - i
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-12-10 04:34:48 +00:00
|
|
|
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
|
2020-12-11 02:59:40 +00:00
|
|
|
elseif not pos then
|
|
|
|
return nil -- ERROR, variable undefined
|
2020-12-10 04:34:48 +00:00
|
|
|
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
|
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
function tlang.near_access(state, index)
|
2020-12-10 04:34:48 +00:00
|
|
|
return tlang.map_access_assign(state, index)
|
|
|
|
end
|
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
function tlang.near_assign(state, index, value)
|
2020-12-10 04:34:48 +00:00
|
|
|
tlang.map_access_assign(state, index, nil, value)
|
|
|
|
end
|
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
function tlang.global_access(state, index)
|
|
|
|
tlang.map_access_assign(state, index, state.locals[1].vars)
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
function tlang.global_assign(state, index, value)
|
|
|
|
tlang.map_access_assign(state, index, state.locals[1].vars, value)
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
function tlang.local_access(state, index)
|
|
|
|
tlang.map_access_assign(state, index, state.locals[#state.locals].vars)
|
|
|
|
end
|
|
|
|
|
|
|
|
function tlang.local_assign(state, index, value)
|
|
|
|
tlang.map_access_assign(state, index, state.locals[#state.locals].vars, value)
|
2020-11-05 02:52:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
function tlang.get_pc(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
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
|
2020-12-11 02:59:40 +00:00
|
|
|
code = tlang.near_access(state, pc.pos)
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
if code then
|
|
|
|
return code.value[pc.elem]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
function tlang.increment_pc(state, pc)
|
2020-11-01 02:55:54 +00:00
|
|
|
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-07 19:27:25 +00:00
|
|
|
local pc = tlang.get_pc(state)
|
2020-11-06 00:54:06 +00:00
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
-- allows for finished states to be used in calls
|
|
|
|
if #state.locals == 1 then
|
2020-11-01 02:55:54 +00:00
|
|
|
return nil
|
|
|
|
end
|
2020-11-03 07:06:59 +00:00
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
state.locals[#state.locals] = nil
|
|
|
|
|
2020-11-03 07:06:59 +00:00
|
|
|
-- pop code stack
|
|
|
|
if pc.sg == 0 then
|
|
|
|
state.code_stack[pc.pos] = nil
|
|
|
|
end
|
2020-11-06 17:06:32 +00:00
|
|
|
|
|
|
|
return getnext(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:54:06 +00:00
|
|
|
local current
|
|
|
|
if not state.locals[#state.locals].nextpop then
|
2020-11-07 19:27:25 +00:00
|
|
|
state.current_pc = tlang.get_pc(state)
|
2020-11-06 00:54:06 +00:00
|
|
|
current = accesspc(state, state.current_pc)
|
|
|
|
end
|
2020-11-01 02:55:54 +00:00
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
local incd = tlang.increment_pc(state, tlang.get_pc(state))
|
2020-11-01 02:55:54 +00:00
|
|
|
if not incd then
|
2020-11-04 02:57:48 +00:00
|
|
|
state.locals[#state.locals].nextpop = true
|
2020-11-06 00:54:06 +00:00
|
|
|
else
|
|
|
|
state.locals[#state.locals].pc = incd
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
return current
|
|
|
|
end
|
|
|
|
|
2020-11-06 16:20:04 +00:00
|
|
|
-- 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
|
2020-11-01 02:55:54 +00:00
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.peek_raw(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
return state.stack[#state.stack]
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.pop_raw(state)
|
|
|
|
local tos = tlang.peek_raw(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
state.stack[#state.stack] = nil
|
|
|
|
return tos
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
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)
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.push_raw(state, tlang.value_to_tlang(value))
|
2020-11-06 00:13:21 +00:00
|
|
|
end
|
|
|
|
|
2020-11-04 02:57:48 +00:00
|
|
|
local function statepeek_type(state, t)
|
2020-11-06 00:13:21 +00:00
|
|
|
local tos = tlang.peek_raw(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
|
|
|
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)
|
2020-11-06 00:13:21 +00:00
|
|
|
local tos = tlang.peek_raw(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
|
|
|
|
if tos.type == t then
|
2020-11-06 00:13:21 +00:00
|
|
|
return tlang.pop_raw(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
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
|
|
|
|
|
|
|
|
local function statepush_num(state, number)
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.push_raw(state, {type = "number", value = number})
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins = {}
|
2020-11-01 02:55:54 +00:00
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.builtins.run(state)
|
|
|
|
tlang.call_tos(state)
|
2020-11-03 07:06:59 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["="] = function(state)
|
2020-11-04 03:03:53 +00:00
|
|
|
local name = statepop_type(state, "quote")
|
2020-11-06 00:13:21 +00:00
|
|
|
local value = tlang.pop_raw(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
|
2020-12-11 02:59:40 +00:00
|
|
|
tlang.near_assign(state, name.value, value)
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.unary(func)
|
2020-11-05 03:12:18 +00:00
|
|
|
return function(state)
|
2020-11-06 00:13:21 +00:00
|
|
|
local tos = tlang.pop_raw(state)
|
2020-11-05 03:12:18 +00:00
|
|
|
if tos.type == "number" then
|
|
|
|
statepush_num(state, func(tos.value))
|
|
|
|
elseif tos.type == "quote" then
|
2020-11-06 00:13:21 +00:00
|
|
|
local n = tlang.near_access(state, tos.value)
|
2020-12-11 02:59:40 +00:00
|
|
|
tlang.near_assign(state, tos.value, {type = "number", value = func(n.value)})
|
2020-11-05 03:12:18 +00:00
|
|
|
end
|
|
|
|
end
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.binary(func)
|
2020-11-05 03:12:18 +00:00
|
|
|
return function(state)
|
|
|
|
local tos = statepop_num(state)
|
|
|
|
local tos1 = statepop_num(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-06 23:39:49 +00:00
|
|
|
statepush_num(state, func(tos1.value, tos.value))
|
2020-11-05 03:12:18 +00:00
|
|
|
end
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["--"] = tlang.unary(function(v)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v - 1
|
|
|
|
end)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["++"] = tlang.unary(function(v)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v + 1
|
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["!"] = tlang.unary(function(v)
|
2020-11-06 16:38:52 +00:00
|
|
|
return tlang.boolean_to_number(not tlang.number_to_boolean(v))
|
2020-11-05 03:12:18 +00:00
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["+"] = tlang.binary(function(v1, v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v1 + v2
|
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["-"] = tlang.binary(function(v1, v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v1 - v2
|
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["*"] = tlang.binary(function(v1, v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v1 * v2
|
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["/"] = tlang.binary(function(v1, v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v1 / v2
|
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["%"] = tlang.binary(function(v1, v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
return v1 % v2
|
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["=="] = tlang.binary(function(v1, v2)
|
2020-11-06 16:38:52 +00:00
|
|
|
return tlang.boolean_to_number(v1 == v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
end)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["!="] = tlang.binary(function(v1, v2)
|
2020-11-06 16:38:52 +00:00
|
|
|
return tlang.boolean_to_number(v1 ~= v2)
|
2020-11-05 03:12:18 +00:00
|
|
|
end)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-07 19:27:25 +00:00
|
|
|
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)
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["if"] = function(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
local tos = statepop_type(state, "code")
|
2020-11-06 00:13:21 +00:00
|
|
|
local tos1 = tlang.pop_raw(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
|
|
|
if tos1.type == "number" then
|
|
|
|
if tos1.value ~= 0 then
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.push_raw(state, tos)
|
|
|
|
tlang.call_tos(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
2020-11-06 17:06:32 +00:00
|
|
|
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
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.builtins.print(state)
|
|
|
|
local value = tlang.pop_raw(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
|
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
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.builtins.dup(state)
|
|
|
|
tlang.push_raw(state, tlang.peek_raw(state))
|
2020-11-01 02:55:54 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.builtins.popoff(state)
|
2020-11-01 02:55:54 +00:00
|
|
|
state.stack[#state.stack] = nil
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.builtins.wait(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
local tos = statepop_type(state, "number")
|
|
|
|
state.wait_target = os.clock() + tos.value
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["forever"] = function(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
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
|
2020-11-08 23:17:07 +00:00
|
|
|
local tos = tlang.pop_raw(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-08 18:26:01 +00:00
|
|
|
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
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.push_raw(state, state.locals[slen].loop_code)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
2020-11-04 02:57:48 +00:00
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.call_tos(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["while"] = function(state)
|
2020-11-04 19:19:36 +00:00
|
|
|
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
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.push_raw(state, state.locals[slen].test_code)
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.call_tos(state)
|
2020-11-04 19:19:36 +00:00
|
|
|
|
|
|
|
state.locals[slen].loop_stage = 1
|
|
|
|
-- stage 1, run while
|
|
|
|
elseif state.locals[slen].loop_stage == 1 then
|
2020-11-06 00:13:21 +00:00
|
|
|
local tos = tlang.pop_raw(state)
|
2020-11-04 19:19:36 +00:00
|
|
|
if tos and tos.value ~= 0 then
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.push_raw(state, state.locals[slen].loop_code)
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.call_tos(state)
|
2020-11-04 19:19:36 +00:00
|
|
|
else
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
2020-11-04 19:19:36 +00:00
|
|
|
state.locals[slen].broke = true
|
|
|
|
end
|
|
|
|
|
|
|
|
state.locals[slen].loop_stage = 0
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["repeat"] = function(state)
|
2020-11-05 06:49:55 +00:00
|
|
|
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
|
2020-11-06 00:13:21 +00:00
|
|
|
local num_var = tlang.pop_raw(state)
|
2020-11-05 06:49:55 +00:00
|
|
|
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
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
block = statepop_type(state, "code")
|
|
|
|
|
2020-11-05 06:49:55 +00:00
|
|
|
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
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.local_assign(state,
|
2020-11-05 06:49:55 +00:00
|
|
|
state.locals[slen].loop_var,
|
|
|
|
{type = "number", value = state.locals[slen].repeat_n})
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.push_raw(state, state.locals[slen].loop_code)
|
2020-11-05 06:49:55 +00:00
|
|
|
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
2020-11-05 06:49:55 +00:00
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.call_tos(state)
|
2020-11-05 06:49:55 +00:00
|
|
|
|
|
|
|
state.locals[slen].repeat_n = state.locals[slen].repeat_n + 1
|
|
|
|
else
|
2020-11-06 16:20:04 +00:00
|
|
|
tlang.set_next_pc(state, state.current_pc)
|
2020-11-05 06:49:55 +00:00
|
|
|
state.locals[slen].broke = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["break"] = function(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
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
|
|
|
|
|
2020-11-06 00:13:21 +00:00
|
|
|
tlang.builtins["return"] = function(state)
|
2020-11-04 02:57:48 +00:00
|
|
|
state.locals[#state.locals] = nil
|
|
|
|
end
|
|
|
|
|
2020-11-06 00:54:06 +00:00
|
|
|
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
|
|
|
|
|
2020-11-01 02:55:54 +00:00
|
|
|
|
|
|
|
-- returns:
|
|
|
|
-- true - more to do
|
|
|
|
-- nil - more to do but waiting
|
|
|
|
-- false - finished
|
|
|
|
-- string - error
|
2020-11-06 00:13:21 +00:00
|
|
|
function tlang.step(state)
|
2020-11-08 23:17:07 +00:00
|
|
|
if state.paused or (state.wait_target and os.clock() < state.wait_target) then
|
2020-11-01 02:55:54 +00:00
|
|
|
return nil
|
|
|
|
end
|
|
|
|
|
|
|
|
local cur = getnext(state)
|
|
|
|
|
|
|
|
if cur == nil then
|
2020-12-11 02:59:40 +00:00
|
|
|
if state.locals[1].nextpop then
|
|
|
|
state.finished = true
|
|
|
|
return false
|
|
|
|
else
|
|
|
|
return "Error: code exited early"
|
|
|
|
end
|
2020-11-07 19:27:25 +00:00
|
|
|
else
|
|
|
|
state.finished = false
|
|
|
|
end
|
|
|
|
|
|
|
|
if in_list(cur.type, literals) then
|
2020-11-01 02:55:54 +00:00
|
|
|
state.stack[#state.stack + 1] = cur
|
|
|
|
elseif cur.type == "identifier" or cur.type == "symbol" then
|
2020-12-10 04:34:48 +00:00
|
|
|
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]
|
2020-11-01 02:55:54 +00:00
|
|
|
f(state)
|
|
|
|
else
|
2020-12-11 02:59:40 +00:00
|
|
|
local var = tlang.near_access(state, cur.value)
|
2020-11-01 02:55:54 +00:00
|
|
|
if var == nil then
|
2020-12-10 04:34:48 +00:00
|
|
|
return "Undefined identifier: " .. table.concat(cur.value, ".")
|
2020-11-01 02:55:54 +00:00
|
|
|
elseif var.type == "code" then
|
2020-11-07 19:27:25 +00:00
|
|
|
tlang.call_var(state, cur.value)
|
2020-11-01 02:55:54 +00:00
|
|
|
else
|
|
|
|
state.stack[#state.stack + 1] = var
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return true
|
|
|
|
end
|