155 lines
3.4 KiB
Lua
155 lines
3.4 KiB
Lua
local tlang = {}
|
|
|
|
local prefix = ""
|
|
if minetest ~= nil then
|
|
prefix = minetest.get_modpath(minetest.get_current_modname()) .. "/"
|
|
end
|
|
|
|
tlang.lex = dofile(prefix .. "tlang_lex.lua")
|
|
tlang.parse = dofile(prefix .. "tlang_parse.lua")
|
|
tlang.builtins, tlang.gassign, tlang.step = dofile(prefix .. "tlang_vm.lua")
|
|
|
|
-- TODO
|
|
--[[
|
|
lexer should include line/character number in symbols
|
|
error messages
|
|
maps should be able to have out of order number indexes (like [1 2 3 10:"Out of order"])
|
|
map.key accessing syntax
|
|
parse as identifier, include . as identifier character, split on . and thats the indexing tree
|
|
--]]
|
|
|
|
function tlang.run(state)
|
|
while true do
|
|
local more = tlang.step(state)
|
|
if more == true or more == nil then
|
|
-- continue along
|
|
elseif type(more) == "string" then
|
|
print(more) -- error
|
|
elseif more == false then
|
|
return -- done
|
|
else
|
|
print("Unknown error, tlang.step returned: " .. tostring(more))
|
|
end
|
|
end
|
|
end
|
|
|
|
local function assign_many(state, source)
|
|
for k, v in pairs(source) do
|
|
tlang.gassign(state, k, v)
|
|
end
|
|
end
|
|
|
|
-- convert a lua value into a tlang literal
|
|
function tlang.valconv(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 == "table" then
|
|
local map = {}
|
|
|
|
for k, v in pairs(value) do
|
|
map[k] = tlang.valconv(v)
|
|
end
|
|
|
|
return {type = "map", value = map}
|
|
end
|
|
end
|
|
|
|
function tlang.get_state(code)
|
|
local lexed = tlang.lex(code)
|
|
local parsed = tlang.parse(lexed)
|
|
|
|
return {
|
|
locals = {{
|
|
pc = {sg = 1, pos = "__ast__", elem = 1},
|
|
v__src__ = tlang.valconv(code),
|
|
v__lex__ = tlang.valconv(lexed),
|
|
v__ast__ = {type = "code", value = parsed}}},
|
|
stack = {},
|
|
code_stack = {},
|
|
builtins = tlang.builtins,
|
|
wait_target = nil,
|
|
nextpop = false
|
|
}
|
|
end
|
|
|
|
function tlang.exec(code)
|
|
local state = tlang.get_state(code)
|
|
tlang.run(state)
|
|
end
|
|
|
|
|
|
local function test()
|
|
local complex = [[{dup *} `square =
|
|
-5.42 square
|
|
"Hello, world!" print
|
|
[1 2 3 str:"String"]
|
|
]]
|
|
|
|
local number = "-4.2123"
|
|
|
|
local simple = "{dup *}"
|
|
|
|
local map = "[this:2 that:3]"
|
|
|
|
local square = [[{dup *} `square =
|
|
5 square print]]
|
|
|
|
local square_run = "5 {dup *} run print"
|
|
|
|
local comment_test = "'asd' print # 'aft' print"
|
|
|
|
local forever_test = [[
|
|
5 # iteration count
|
|
{
|
|
dup # duplicate iter count
|
|
print # print countdown
|
|
-- # decrement
|
|
dup 0 == # check if TOS is 0
|
|
{break} if # break if TOS == 0
|
|
}
|
|
forever # run loop
|
|
]]
|
|
|
|
local local_test = [[
|
|
'outside' `var =
|
|
{
|
|
var print # should be 'outside'
|
|
'inside' `var =
|
|
var print # should be 'inside'
|
|
} run
|
|
var print # should be 'inside'
|
|
]]
|
|
|
|
local while_test = [[
|
|
5 `cur =
|
|
{
|
|
`cur --
|
|
cur
|
|
} {
|
|
"four times" print
|
|
} while
|
|
]]
|
|
|
|
local repeat_test = [[
|
|
{
|
|
"four times" print
|
|
} 4 repeat
|
|
{
|
|
i print
|
|
} 5 `i repeat
|
|
]]
|
|
|
|
local stack_test = "5 5 == print"
|
|
|
|
tlang.exec(repeat_test)
|
|
end
|
|
|
|
if minetest == nil then
|
|
test()
|
|
end
|
|
|
|
return tlang
|