Add .busted config, seperate interpreter functions from initial setup code for forth machine
This commit is contained in:
parent
e8bcb186e3
commit
444a5590cd
|
@ -1,6 +1,6 @@
|
|||
local ds = require("DataStructures")
|
||||
local Dictionary, Stack, WordInfo, Environment = ds.Dictionary, ds.Stack, ds.WordInfo, ds.Environment
|
||||
|
||||
local Pointer = ds.Pointer
|
||||
local CoreHelpers = {}
|
||||
|
||||
function CoreHelpers.defineWord(dict: Dictionary, str: string, func: function(Environment, ...: any): any..., imm: boolean)
|
||||
|
@ -8,7 +8,10 @@ function CoreHelpers.defineWord(dict: Dictionary, str: string, func: function(En
|
|||
dict:define(str, info)
|
||||
end
|
||||
|
||||
|
||||
function CoreHelpers.standardInputRefill(): string
|
||||
local input = io.read().."\n"
|
||||
return input
|
||||
end
|
||||
|
||||
function CoreHelpers.areNumbers(a: any, b: any): boolean
|
||||
return a is number and b is number
|
||||
|
@ -63,5 +66,21 @@ function CoreHelpers.reset(env: Environment)
|
|||
env.instructionPointer.index = 0 -- will get incremented after this instruction
|
||||
end
|
||||
|
||||
function CoreHelpers.ret(env: Environment)
|
||||
local retAddr = env.returnStack:pop() as Pointer
|
||||
env.instructionPointer = retAddr
|
||||
end
|
||||
|
||||
function CoreHelpers.makeCall(body: {function(Environment)}): function(Environment)
|
||||
return function(e: Environment)
|
||||
e.returnStack:push(e.instructionPointer)
|
||||
e.instructionPointer = Pointer:new(body, 0)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
return CoreHelpers
|
||||
|
||||
|
|
17
CoreWords.tl
17
CoreWords.tl
|
@ -5,6 +5,8 @@ local areNumbers, getActiveDataStack, isNumber, isWhitespace= helpers.areNumbers
|
|||
local skipWhitespace, parseToken = helpers.skipWhitespace, helpers.parseToken
|
||||
local popTwoOperands = helpers.popTwoOperands
|
||||
local defineWord = helpers.defineWord
|
||||
local ret = helpers.ret
|
||||
local makeCall = helpers.makeCall
|
||||
-- Mathematical operations
|
||||
local function add(state: Environment)
|
||||
local a, b = popTwoOperands(state)
|
||||
|
@ -189,6 +191,21 @@ CoreWords:define("2DUP", twoDupInfo)
|
|||
CoreWords:define("2SWAP", twoSwapInfo)
|
||||
CoreWords:define("2OVER", twoOverInfo)
|
||||
CoreWords:define("NIP", nipInfo)
|
||||
|
||||
|
||||
|
||||
local sqr = {dup, mul, ret}
|
||||
local sqrf = makeCall(sqr)
|
||||
|
||||
defineWord(CoreWords, "SQR", sqrf, false)
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
return CoreWords
|
||||
|
||||
|
||||
|
|
|
@ -27,7 +27,7 @@ local type DataStructures = record
|
|||
referant: {number: any}
|
||||
deref: function(self: Pointer): any
|
||||
inc: function(self: Pointer)
|
||||
new: function(self: Pointer): Pointer
|
||||
new: function(self: Pointer, body: {function(Environment)}, idx: number): Pointer
|
||||
end
|
||||
|
||||
record Environment
|
||||
|
@ -127,7 +127,11 @@ local pointer_mt = {__index = Pointer}
|
|||
function Pointer:new(): Pointer
|
||||
return setmetatable({referant = {}, index = 1} as Pointer, pointer_mt)
|
||||
end
|
||||
|
||||
function Pointer:new(body: {function(Environment)}, idx: number): Pointer
|
||||
local tbl = body or {}
|
||||
local jdx = idx or 1
|
||||
return setmetatable({referant = tbl, index = jdx} as Pointer, pointer_mt)
|
||||
end
|
||||
return DataStructures
|
||||
|
||||
|
||||
|
|
|
@ -3,54 +3,32 @@ local ds = require("DataStructures")
|
|||
local istream = require("InputStream")
|
||||
local Stack, Dictionary, WordInfo, Environment = ds.Stack, ds.Dictionary, ds.WordInfo, ds.Environment
|
||||
local helpers = require("CoreHelpers")
|
||||
local interpreters = require("Interpreters")
|
||||
local isNumber, skipWhitespace, parseToken = helpers.isNumber, helpers.skipWhitespace, helpers.parseToken
|
||||
local reset = helpers.reset
|
||||
local machineEnvironment = Environment:new()
|
||||
local Pointer = ds.Pointer
|
||||
function standardInputRefill(): string
|
||||
local input = io.read().."\n"
|
||||
return input
|
||||
end
|
||||
local standardInputRefill = helpers.standardInputRefill
|
||||
local innerInterpreter, outerInterpreter = interpreters.inner, interpreters.outer
|
||||
|
||||
|
||||
|
||||
local machineEnvironment = Environment:new()
|
||||
local standardInputStream = istream:new(standardInputRefill)
|
||||
|
||||
machineEnvironment.activeInputStream = standardInputStream
|
||||
machineEnvironment.activeDataStack = Stack:new()
|
||||
machineEnvironment.returnStack = Stack:new()
|
||||
|
||||
table.insert(machineEnvironment.dictionaries, CoreWords)
|
||||
|
||||
function outerInterpreter(env: Environment)
|
||||
while(true) do
|
||||
skipWhitespace(env)
|
||||
local token: string = parseToken(env)
|
||||
if isNumber(token) then
|
||||
env.activeDataStack:push(tonumber(token))
|
||||
else
|
||||
for i, dictionary in ipairs(env.dictionaries) do
|
||||
local wordinfo = dictionary:lookup(token)
|
||||
if wordinfo then
|
||||
(wordinfo as WordInfo).func(env)
|
||||
goto continue
|
||||
end
|
||||
end
|
||||
print(string.format('%q ?', token))
|
||||
::continue::
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
function innerInterpreter(env: Environment)
|
||||
while(true) do
|
||||
(env.instructionPointer:deref() as function(Environment))(env)
|
||||
env.instructionPointer:inc()
|
||||
end
|
||||
end
|
||||
|
||||
local interpreterInstructions = {outerInterpreter, reset}
|
||||
local interpreterInstructions = {}
|
||||
table.insert(interpreterInstructions, outerInterpreter)
|
||||
table.insert(interpreterInstructions, reset)
|
||||
local startPointer = Pointer:new()
|
||||
startPointer.referant = interpreterInstructions
|
||||
startPointer.index = 0
|
||||
|
||||
machineEnvironment.instructionPointer = startPointer
|
||||
|
||||
innerInterpreter(machineEnvironment)
|
||||
|
||||
outerInterpreter(machineEnvironment)
|
|
@ -0,0 +1,38 @@
|
|||
local ds = require("DataStructures")
|
||||
local Stack, Dictionary, WordInfo, Environment = ds.Stack, ds.Dictionary, ds.WordInfo, ds.Environment
|
||||
local helpers = require("CoreHelpers")
|
||||
local isNumber, skipWhitespace, parseToken = helpers.isNumber, helpers.skipWhitespace, helpers.parseToken
|
||||
|
||||
local type Interpreters = record
|
||||
inner: function(Environment)
|
||||
outer: function(Environment)
|
||||
end
|
||||
|
||||
|
||||
|
||||
function Interpreters.outer(env: Environment)
|
||||
skipWhitespace(env)
|
||||
local token: string = parseToken(env)
|
||||
if isNumber(token) then
|
||||
env.activeDataStack:push(tonumber(token))
|
||||
else
|
||||
for _, dictionary in ipairs(env.dictionaries) do
|
||||
local wordinfo = dictionary:lookup(token)
|
||||
if wordinfo then
|
||||
(wordinfo as WordInfo).func(env)
|
||||
goto continue
|
||||
end
|
||||
end
|
||||
print(string.format('%q ?', token))
|
||||
::continue::
|
||||
end
|
||||
end
|
||||
|
||||
function Interpreters.inner(env: Environment)
|
||||
while(true) do
|
||||
env.instructionPointer:deref() as function(Environment)(env)
|
||||
env.instructionPointer:inc()
|
||||
end
|
||||
end
|
||||
|
||||
return Interpreters
|
Loading…
Reference in New Issue