mirror of
https://github.com/minetest-mods/mesecons.git
synced 2025-01-12 09:50:23 +01:00
Compile FPGA configurations to Lua
This commit is contained in:
parent
c10ce2dbc5
commit
776c7e0b23
@ -312,6 +312,37 @@ plg.red_box_around = function(i)
|
|||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
|
local compile_cache = {}
|
||||||
|
|
||||||
|
plg.compile_with_cache = function(instr)
|
||||||
|
local entry = compile_cache[instr]
|
||||||
|
if entry then
|
||||||
|
-- Prevent removal from the cache
|
||||||
|
entry.age = 0
|
||||||
|
else
|
||||||
|
entry = {
|
||||||
|
func = lcore.compile(lcore.deserialize(instr)),
|
||||||
|
age = 0,
|
||||||
|
}
|
||||||
|
compile_cache[instr] = entry
|
||||||
|
end
|
||||||
|
return entry.func
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Entries are removed from the cache if they are unused for at least a minute.
|
||||||
|
local function clean_compile_cache()
|
||||||
|
for instr, entry in pairs(compile_cache) do
|
||||||
|
if entry.age < 1 then
|
||||||
|
entry.age = entry.age + 1
|
||||||
|
else
|
||||||
|
compile_cache[instr] = nil
|
||||||
|
end
|
||||||
|
end
|
||||||
|
minetest.after(60, clean_compile_cache)
|
||||||
|
end
|
||||||
|
minetest.after(60, clean_compile_cache)
|
||||||
|
|
||||||
|
|
||||||
plg.update = function(pos)
|
plg.update = function(pos)
|
||||||
local meta = minetest.get_meta(pos)
|
local meta = minetest.get_meta(pos)
|
||||||
if meta:get_int("valid") ~= 1 then
|
if meta:get_int("valid") ~= 1 then
|
||||||
@ -323,9 +354,9 @@ plg.update = function(pos)
|
|||||||
return
|
return
|
||||||
end
|
end
|
||||||
|
|
||||||
local is = lcore.deserialize(meta:get_string("instr"))
|
local func = plg.compile_with_cache(meta:get_string("instr"))
|
||||||
local A, B, C, D = plg.getports(pos)
|
local A, B, C, D = plg.getports(pos)
|
||||||
A, B, C, D = lcore.interpret(is, A, B, C, D)
|
A, B, C, D = func(A, B, C, D)
|
||||||
plg.setports(pos, A, B, C, D)
|
plg.setports(pos, A, B, C, D)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -6,16 +6,16 @@ local operations = {
|
|||||||
-- gate: Internal name
|
-- gate: Internal name
|
||||||
-- short: Serialized form, single character
|
-- short: Serialized form, single character
|
||||||
-- fs_name: Display name, padded to 4 characters
|
-- fs_name: Display name, padded to 4 characters
|
||||||
-- func: Function that applies the operation
|
-- func: Function that returns tokens representing the operation
|
||||||
-- unary: Whether this gate only has one input
|
-- unary: Whether this gate only has one input
|
||||||
{ gate = "and", short = "&", fs_name = " AND", func = function(a, b) return a and b end },
|
{ gate = "and", short = "&", fs_name = " AND", func = function(a, b) return a, "and", b end },
|
||||||
{ gate = "or", short = "|", fs_name = " OR", func = function(a, b) return a or b end },
|
{ gate = "or", short = "|", fs_name = " OR", func = function(a, b) return a, "or", b end },
|
||||||
{ gate = "not", short = "~", fs_name = " NOT", func = function(_, b) return not b end, unary = true },
|
{ gate = "not", short = "~", fs_name = " NOT", func = function(_, b) return "not", b end, unary = true },
|
||||||
{ gate = "xor", short = "^", fs_name = " XOR", func = function(a, b) return a ~= b end },
|
{ gate = "xor", short = "^", fs_name = " XOR", func = function(a, b) return a, "~=", b end },
|
||||||
{ gate = "nand", short = "?", fs_name = "NAND", func = function(a, b) return not (a and b) end },
|
{ gate = "nand", short = "?", fs_name = "NAND", func = function(a, b) return "not (", a, "and", b, ")" end },
|
||||||
{ gate = "buf", short = "_", fs_name = " =", func = function(_, b) return b end, unary = true },
|
{ gate = "buf", short = "_", fs_name = " =", func = function(_, b) return b end, unary = true },
|
||||||
{ gate = "xnor", short = "=", fs_name = "XNOR", func = function(a, b) return a == b end },
|
{ gate = "xnor", short = "=", fs_name = "XNOR", func = function(a, b) return a, "==", b end },
|
||||||
{ gate = "nor", short = "!", fs_name = " NOR", func = function(a, b) return not (a or b) end },
|
{ gate = "nor", short = "!", fs_name = " NOR", func = function(a, b) return "not (", a, "or", b, ")" end },
|
||||||
}
|
}
|
||||||
|
|
||||||
lg.get_operations = function()
|
lg.get_operations = function()
|
||||||
@ -177,46 +177,63 @@ lg.validate = function(t)
|
|||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
-- interpreter
|
local fpga_env = setmetatable({}, {
|
||||||
lg.interpret = function(t, a, b, c, d)
|
__index = function(_, var)
|
||||||
local function _action(s, v1, v2)
|
error("FPGA code tried to read undeclared variable " .. var)
|
||||||
|
end,
|
||||||
|
__newindex = function(_, var)
|
||||||
|
error("FPGA code tried to set undeclared variable " .. var)
|
||||||
|
end,
|
||||||
|
})
|
||||||
|
|
||||||
|
-- compiler
|
||||||
|
lg.compile = function(t)
|
||||||
|
-- Get token generation function from action gate string
|
||||||
|
local function _action(s)
|
||||||
for i, data in ipairs(operations) do
|
for i, data in ipairs(operations) do
|
||||||
if data.gate == s then
|
if data.gate == s then
|
||||||
return data.func(v1, v2)
|
return data.func
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
return false -- unknown gate
|
return nil -- unknown gate
|
||||||
end
|
end
|
||||||
local function _op(t, regs, io_in)
|
-- Serialize input operand
|
||||||
if t.type == "reg" then
|
local function _op(t)
|
||||||
return regs[t.n]
|
if t == nil then
|
||||||
|
return nil
|
||||||
|
elseif t.type == "reg" then
|
||||||
|
return "r" .. t.n
|
||||||
else -- t.type == "io"
|
else -- t.type == "io"
|
||||||
return io_in[t.port]
|
return "i" .. t.port
|
||||||
|
end
|
||||||
|
end
|
||||||
|
-- Serialize destination
|
||||||
|
local function _dst(t)
|
||||||
|
if t.type == "reg" then
|
||||||
|
return "local r" .. t.n
|
||||||
|
else -- t.type == "io"
|
||||||
|
return "o" .. t.port
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
local io_in = {A=a, B=b, C=c, D=d}
|
local tokens = {
|
||||||
local regs = {}
|
-- Declare inputs and outputs:
|
||||||
local io_out = {}
|
"return function(iA, iB, iC, iD) local oA, oB, oC, oD",
|
||||||
|
}
|
||||||
for i = 1, 14 do
|
for i = 1, 14 do
|
||||||
local cur = t[i]
|
local cur = t[i]
|
||||||
if next(cur) ~= nil then
|
if next(cur) ~= nil then
|
||||||
local v1, v2
|
table.insert(tokens, _dst(cur.dst))
|
||||||
if cur.op1 ~= nil then
|
table.insert(tokens, "=")
|
||||||
v1 = _op(cur.op1, regs, io_in)
|
table.insert_all(tokens, {_action(cur.action)(_op(cur.op1), _op(cur.op2))})
|
||||||
end
|
end
|
||||||
v2 = _op(cur.op2, regs, io_in)
|
end
|
||||||
|
table.insert(tokens, "return oA, oB, oC, oD end")
|
||||||
|
|
||||||
local result = _action(cur.action, v1, v2)
|
local code = table.concat(tokens, " ")
|
||||||
|
local func = assert(loadstring(code))()
|
||||||
if cur.dst.type == "reg" then
|
setfenv(func, fpga_env)
|
||||||
regs[cur.dst.n] = result
|
return func
|
||||||
else -- cur.dst.type == "io"
|
|
||||||
io_out[cur.dst.port] = result
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
return io_out.A, io_out.B, io_out.C, io_out.D
|
|
||||||
end
|
end
|
||||||
|
|
||||||
return lg
|
return lg
|
||||||
|
Loading…
Reference in New Issue
Block a user