aboutsummaryrefslogtreecommitdiffstats
path: root/corewords.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'corewords.cpp')
-rw-r--r--corewords.cpp408
1 files changed, 170 insertions, 238 deletions
diff --git a/corewords.cpp b/corewords.cpp
index 9c52c8d..64708e9 100644
--- a/corewords.cpp
+++ b/corewords.cpp
@@ -18,241 +18,185 @@
#include "corewords.hpp"
-Func CoreWords::get(int index)
-{
- static const Func ops[WordCount] = {
- op_drop, op_dup, op_swap, op_pick, op_sys,
- op_add, op_sub, op_mul, op_div, op_mod,
- /*10*/ op_peek, op_poke, op_pushr, op_popr, op_eq,
- op_lt, op_allot, op_and, op_or, op_xor,
- /*20*/ op_shl, op_shr, op_colon, op_semic, op_here,
- op_depth, op_key, op_exit, op_tick, op_execute,
- /*30*/ op_jmp, op_jmp0, op_lit, op_literal, op_rdepth
+void CoreWords::run(unsigned int index, State& state)
+{
+ auto getword = [&state] {
+ auto word = state.dict.input();
+ while (word.size() == 0) {
+ state.input(state);
+ word = state.dict.input();
+ }
+ return word;
+ };
+ auto newdef = [](Dictionary& dict, Word word) {
+ auto addr = dict.alignhere();
+ dict.addDefinition(word);
+ dict.write(addr,
+ (dict.read(addr) & 0x1F) |
+ ((addr - dict.latest()) << 6));
+ dict.latest(addr);
+ };
+ auto tick = [&state](Word word) {
+ if (auto j = state.dict.find(word); j > 0)
+ state.push(state.dict.getexec(j));
+ else if (auto i = CoreWords::findi(state, word); i >= 0)
+ state.push(i & ~CoreWords::Compiletime);
+ else
+ state.push(0);
};
- return index >= 0 && index < WordCount ? ops[index] : nullptr;
-}
-
-void CoreWords::op_drop(State& state)
-{
- state.pop();
-}
-
-void CoreWords::op_dup(State& state)
-{
- state.push(state.top());
-}
-
-void CoreWords::op_swap(State& state)
-{
- std::swap(state.top(), state.pick(1));
-}
-
-void CoreWords::op_pick(State& state)
-{
- state.push(state.pick(state.pop()));
-}
-
-void CoreWords::op_sys(State& state)
-{
- return user_sys(state);
-}
-
-void CoreWords::op_add(State& state)
-{
- const auto a = state.pop();
- state.top() += a;
-}
-
-void CoreWords::op_sub(State& state)
-{
- const auto a = state.pop();
- state.top() -= a;
-}
-
-void CoreWords::op_mul(State& state) {
- const auto a = state.pop();
- state.top() *= a;
-}
-
-void CoreWords::op_div(State& state) {
- const auto a = state.pop();
- state.top() /= a;
-}
-
-void CoreWords::op_mod(State& state) {
- const auto a = state.pop();
- state.top() %= a;
-}
-
-void CoreWords::op_peek(State& state) {
- if (auto w = state.pop(); w == 1)
- state.push(state.dict.readbyte(state.pop()));
- else
- state.push(state.dict.read(state.pop()));
-}
-
-void CoreWords::op_poke(State& state) {
- const auto w = state.pop();
- const auto addr = state.pop();
- if (w == 1)
- state.dict.writebyte(addr, state.pop());
- else
- state.dict.write(addr, state.pop());
-}
-
-void CoreWords::op_pushr(State& state) {
- state.pushr(state.pop());
-}
-
-void CoreWords::op_popr(State& state) {
- state.push(state.popr());
-}
-
-void CoreWords::op_eq(State& state) {
- const auto a = state.pop();
- state.top() = state.top() == a;
-}
-
-void CoreWords::op_lt(State& state) {
- const auto a = state.pop();
- state.top() = state.top() < a;
-}
-
-void CoreWords::op_allot(State& state) {
- state.dict.allot(state.pop());
-}
-
-void CoreWords::op_and(State& state) {
- const auto a = state.pop();
- state.top() &= a;
-}
-
-void CoreWords::op_or(State& state) {
- const auto a = state.pop();
- state.top() |= a;
-}
-
-void CoreWords::op_xor(State& state) {
- const auto a = state.pop();
- state.top() ^= a;
-}
-
-void CoreWords::op_shl(State& state) {
- const auto a = state.pop();
- state.top() <<= a;
-}
-
-void CoreWords::op_shr(State& state) {
- const auto a = state.pop();
- state.top() >>= a;
-}
-
-void CoreWords::op_colon(State& state) {
- Word word = state.dict.input();
- while (word.size() == 0) {
- state.input(state);
- word = state.dict.input();
- }
-
- const auto start = state.dict.alignhere();
- state.dict.addDefinition(word);
- state.dict.write(start,
- (state.dict.read(start) & 0x1F) |
- ((start - state.dict.latest()) << 6));
- state.dict.latest(start);
- state.compiling(true);
-}
-
-void CoreWords::op_tick(State& state) {
- Word word = state.dict.input();
- while (word.size() == 0) {
- state.input(state);
- word = state.dict.input();
- }
-
- Cell xt = 0;
- if (auto i = CoreWords::findi(state, word); i >= 0) {
- xt = i & ~CoreWords::Compiletime;
- } else if (auto j = state.dict.find(word); j > 0) {
- xt = state.dict.getexec(j);
- }
-
- state.push(xt);
-}
-
-void CoreWords::op_execute(State& state) {
- state.execute(state.pop());
-}
-
-void CoreWords::op_exit(State& state) {
- state.ip = state.popr();
-}
-
-void CoreWords::op_semic(State& state) {
- if (state.compiling()) {
+ Cell cell;
+
+ switch (index) {
+ default:
+ // must be calling a defined subroutine
+ state.pushr(state.ip);
+ state.ip = index - sizeof(Cell);
+ break;
+ case 0: // drop
+ state.pop();
+ break;
+ case 1: // dup
+ state.push(state.top());
+ break;
+ case 2: // swap
+ std::swap(state.top(), state.pick(1));
+ break;
+ case 3: // pick
+ state.push(state.pick(state.pop()));
+ break;
+ case 4: // sys
+ user_sys(state);
+ break;
+ case 5: // add
+ cell = state.pop();
+ state.top() += cell;
+ break;
+ case 6: // sub
+ cell = state.pop();
+ state.top() -= cell;
+ break;
+ case 7: // mul
+ cell = state.pop();
+ state.top() *= cell;
+ break;
+ case 8: // div
+ cell = state.pop();
+ state.top() /= cell;
+ break;
+ case 9: // mod
+ cell = state.pop();
+ state.top() %= cell;
+ break;
+ case 10: // peek
+ if (state.pop())
+ state.push(state.dict.read(state.pop()));
+ else
+ state.push(state.dict.readbyte(state.pop()));
+ break;
+ case 11: // poke
+ cell = state.pop();
+ if (auto addr = state.pop(); cell)
+ state.dict.write(addr, state.pop());
+ else
+ state.dict.writebyte(addr, state.pop());
+ break;
+ case 12: // pushr
+ state.pushr(state.pop());
+ break;
+ case 13: // popr
+ state.push(state.popr());
+ break;
+ case 14: // equal
+ cell = state.pop();
+ state.top() = state.top() == cell;
+ break;
+ case 15: // lt
+ cell = state.pop();
+ state.top() = state.top() < cell;
+ break;
+ case 16: // allot
+ state.dict.allot(state.pop());
+ break;
+ case 17: // and
+ cell = state.pop();
+ state.top() &= cell;
+ break;
+ case 18: // or
+ cell = state.pop();
+ state.top() |= cell;
+ break;
+ case 19: // xor
+ cell = state.pop();
+ state.top() ^= cell;
+ break;
+ case 20: // shl
+ cell = state.pop();
+ state.top() <<= cell;
+ break;
+ case 21: // shr
+ cell = state.pop();
+ state.top() >>= cell;
+ break;
+ case 22: // colon
+ newdef(state.dict, getword());
+ state.compiling(true);
+ break;
+ case 23: // tick
+ tick(getword());
+ break;
+ case 24: // execute
+ state.execute(state.pop());
+ break;
+ case 25: // exit
+ state.ip = state.popr();
+ break;
+ case 26: // semic
state.dict.add(findi("exit"));
state.compiling(false);
- }
-}
-
-void CoreWords::op_here(State& state) {
- state.push(state.dict.here);
-}
-
-void CoreWords::op_lit(State& state)
-{
- state.push(state.beyondip());
- state.ip += sizeof(Cell);
-}
-
-void CoreWords::op_literal(State& state)
-{
- if (state.compiling()) {
+ break;
+ case 27: // here
+ state.push(state.dict.here);
+ break;
+ case 28: // _lit
+ state.push(state.beyondip());
+ break;
+ case 29: // literal
state.dict.add(findi("_lit"));
state.dict.add(state.pop());
+ break;
+ case 30: // _jmp
+ state.ip = state.beyondip() - sizeof(Cell);
+ break;
+ case 31: // _jmp0
+ if (state.pop())
+ state.beyondip();
+ else
+ state.ip = state.beyondip() - sizeof(Cell);
+ break;
+ case 32: // depth
+ state.push(state.size());
+ break;
+ case 33: // _rdepth
+ state.push(state.rsize());
+ break;
+ case 34: // key
+ cell = state.dict.read(Dictionary::Input);
+ while (cell <= 0) {
+ state.input(state);
+ cell = state.dict.read(Dictionary::Input);
+ }
+
+ state.dict.write(Dictionary::Input, cell - 1);
+
+ state.push(
+ state.dict.readbyte(
+ Dictionary::Input + sizeof(Cell) +
+ Dictionary::InputCells - cell));
+ break;
}
}
-void CoreWords::op_jmp(State& state)
-{
- state.ip = state.beyondip() - sizeof(Cell);
-}
-
-void CoreWords::op_jmp0(State& state)
-{
- if (state.pop())
- state.ip += sizeof(Cell);
- else
- op_jmp(state);
-}
-
-void CoreWords::op_depth(State& state)
-{
- state.push(state.size());
-}
-
-void CoreWords::op_rdepth(State& state)
-{
- state.push(state.rsize());
-}
-
-void CoreWords::op_key(State& state)
-{
- auto len = state.dict.read(Dictionary::Input);
- while (len <= 0) {
- state.input(state);
- len = state.dict.read(Dictionary::Input);
- }
-
- state.dict.write(Dictionary::Input, len - 1);
- Addr addr = Dictionary::Input + sizeof(Cell) +
- Dictionary::InputCells - len;
- Cell val = state.dict.readbyte(addr);
-
- state.push(val);
-}
-
int CoreWords::findi(std::string_view word)
{
std::size_t i;
@@ -293,15 +237,3 @@ int CoreWords::findi(State& state, Word word)
return -1;
}
-Func CoreWords::find(State& state, Word word)
-{
- const auto i = findi(state, word);
- return i >= 0 ? get(i & ~Compiletime) : nullptr;
-}
-
-void CoreWords::run(int i, State& state)
-{
- if (i >= 0 && i < WordCount)
- get(i)(state);
-}
-