Add colon and semicolon operators
This commit is contained in:
parent
5efd5dbb60
commit
3ddc671cb3
|
@ -156,6 +156,7 @@ function CoreHelpers.recNumber(env: Environment)
|
|||
dataStack:push(CoreHelpers.literal)
|
||||
-- return success
|
||||
dataStack:push(true)
|
||||
return
|
||||
end
|
||||
dataStack:push(token)
|
||||
dataStack:push(false)
|
||||
|
@ -172,15 +173,24 @@ function CoreHelpers.recWord(env: Environment)
|
|||
-- interpret
|
||||
dataStack:push(CoreHelpers.execute)
|
||||
-- compile
|
||||
dataStack:push(CoreHelpers.compile)
|
||||
if (wordinfo as WordInfo).immediate then
|
||||
dataStack:push(CoreHelpers.execute)
|
||||
else
|
||||
dataStack:push(CoreHelpers.compile)
|
||||
end
|
||||
-- immediate
|
||||
dataStack:push(CoreHelpers.postpone)
|
||||
-- return success
|
||||
dataStack:push(true)
|
||||
return
|
||||
end
|
||||
dataStack:push(token)
|
||||
dataStack:push(false)
|
||||
end
|
||||
dataStack:push(token)
|
||||
dataStack:push(false)
|
||||
end
|
||||
|
||||
function CoreHelpers.ack(e: Environment)
|
||||
io.write("ok\n")
|
||||
end
|
||||
|
||||
return CoreHelpers
|
||||
|
|
26
CoreWords.tl
26
CoreWords.tl
|
@ -164,6 +164,29 @@ local function exitCompileMode(e: Environment)
|
|||
e.compileState = false
|
||||
end
|
||||
|
||||
local function colon(e: Environment)
|
||||
skipWhitespace(e)
|
||||
local name: string | nil = parseToken(e)
|
||||
if not name then
|
||||
e.running = false
|
||||
return
|
||||
end
|
||||
e.currentDefinitionName = name as string
|
||||
enterCompileMode(e)
|
||||
end
|
||||
|
||||
local function semicolon(e: Environment)
|
||||
table.insert(e.currentDefinition, helpers.ret)
|
||||
local instrs = e.currentDefinition
|
||||
local call = helpers.makeCall(instrs)
|
||||
local dict = e.dictionaries[1]
|
||||
defineWord(dict, e.currentDefinitionName, call, false)
|
||||
e.currentDefinitionName = nil
|
||||
e.currentDefinition = {}
|
||||
exitCompileMode(e)
|
||||
end
|
||||
|
||||
|
||||
local CoreWords = Dictionary:new()
|
||||
local addInfo = WordInfo:new(add, false)
|
||||
local subInfo = WordInfo:new(sub, false)
|
||||
|
@ -188,7 +211,8 @@ defineWord(CoreWords, "[", exitCompileMode, true)
|
|||
defineWord(CoreWords, "]", enterCompileMode, false)
|
||||
defineWord(CoreWords, "COMPILE,", compile, true)
|
||||
defineWord(CoreWords, "LITERAL", literal, true)
|
||||
|
||||
defineWord(CoreWords, ":", colon, true)
|
||||
defineWord(CoreWords, ";", semicolon, true)
|
||||
|
||||
CoreWords:define("+", addInfo)
|
||||
CoreWords:define("-", subInfo)
|
||||
|
|
|
@ -46,7 +46,7 @@ local type DataStructures = record
|
|||
recognizers: {function(Environment)}
|
||||
instructionPointer: Pointer
|
||||
running: boolean
|
||||
|
||||
currentDefinitionName: string | nil
|
||||
currentDefinition: {function(Environment)}
|
||||
end
|
||||
|
||||
|
@ -71,7 +71,10 @@ function Environment:new(input: InputStream, dict: {Dictionary} ): Environment
|
|||
returnStack = Stack:new(),
|
||||
dictionaries = dicts,
|
||||
activeInputStream = input,
|
||||
running = true
|
||||
running = true,
|
||||
recognizers = {},
|
||||
compileState = false,
|
||||
currentDefinition = {}
|
||||
} as Environment,
|
||||
state_mt)
|
||||
end
|
||||
|
|
|
@ -15,6 +15,8 @@ function Interpreters.start(env: Environment)
|
|||
local interpreterInstructions = {}
|
||||
table.insert(interpreterInstructions, Interpreters.outer)
|
||||
table.insert(interpreterInstructions, helpers.reset)
|
||||
table.insert(env.recognizers, helpers.recNumber)
|
||||
table.insert(env.recognizers, helpers.recWord)
|
||||
local startPointer = Pointer:new()
|
||||
startPointer.referant = interpreterInstructions
|
||||
env.instructionPointer = startPointer
|
||||
|
@ -22,27 +24,44 @@ function Interpreters.start(env: Environment)
|
|||
end
|
||||
|
||||
|
||||
function selectRecXT(env: Environment): function(Environment)
|
||||
local dataStack = env.activeDataStack
|
||||
if env.compileState then
|
||||
dataStack:pop()
|
||||
local selected = dataStack:pop()
|
||||
dataStack:pop()
|
||||
return selected as function(Environment)
|
||||
else
|
||||
dataStack:pop()
|
||||
dataStack:pop()
|
||||
return dataStack:pop() as function(Environment)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
function Interpreters.outer(env: Environment)
|
||||
local dataStack = env.activeDataStack
|
||||
skipWhitespace(env)
|
||||
local token: string | nil = parseToken(env)
|
||||
if not token then
|
||||
env.running = false
|
||||
return
|
||||
end
|
||||
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
|
||||
for _, rec in ipairs(env.recognizers) do
|
||||
dataStack:push(token)
|
||||
rec(env)
|
||||
local result = dataStack:pop() as boolean
|
||||
if not result then
|
||||
token = dataStack:pop() as string
|
||||
end
|
||||
if result then
|
||||
(selectRecXT(env))(env)
|
||||
return
|
||||
end
|
||||
print(string.format('%q ?', token))
|
||||
::continue::
|
||||
end
|
||||
|
||||
-- We did not recognize the token.
|
||||
io.write("\""..token.."\"?\n")
|
||||
end
|
||||
|
||||
function Interpreters.inner(env: Environment)
|
||||
|
|
|
@ -231,6 +231,11 @@ describe("Interpreter tests", function()
|
|||
assert.are.same(expectedResult,testEnv.activeDataStack.contents[1])
|
||||
end
|
||||
end)
|
||||
it("does not push anything when a garbage input is given", function()
|
||||
local testEnv = buildEnvironment("BOGUS")
|
||||
interpreters.start(testEnv)
|
||||
assert.are.same({}, testEnv.activeDataStack.contents)
|
||||
end)
|
||||
end)
|
||||
describe("Core word tests", function()
|
||||
describe("DUP tests", function()
|
||||
|
|
Loading…
Reference in New Issue