diff options
Diffstat (limited to 'libalee')
-rw-r--r-- | libalee/corewords.cpp | 15 | ||||
-rw-r--r-- | libalee/corewords.hpp | 4 | ||||
-rw-r--r-- | libalee/parser.cpp | 22 | ||||
-rw-r--r-- | libalee/parser.hpp | 10 | ||||
-rw-r--r-- | libalee/state.cpp | 36 | ||||
-rw-r--r-- | libalee/state.hpp | 16 | ||||
-rw-r--r-- | libalee/types.hpp | 12 |
7 files changed, 76 insertions, 39 deletions
diff --git a/libalee/corewords.cpp b/libalee/corewords.cpp index ced3973..b8a415e 100644 --- a/libalee/corewords.cpp +++ b/libalee/corewords.cpp @@ -17,6 +17,7 @@ */ #include "corewords.hpp" +#include "parser.hpp" #include <cstring> #include <utility> @@ -171,10 +172,8 @@ execute: goto execute; case 25: // exit state.ip = state.popr(); - if (state.ip == 0) { - std::longjmp(state.jmpbuf, - static_cast<int>(State::Error::exit)); - } + if (state.ip == 0) + std::longjmp(state.jmpbuf, static_cast<int>(Error::exit)); break; case 26: // semic { @@ -206,6 +205,14 @@ execute: case 31: // _in state.input(state); break; + case 32: // _ex + { + const auto st = state.save(); + state.ip = 0; + Parser::parseSource(state); + state.load(st); + } + break; default: state.push(index - WordCount); break; diff --git a/libalee/corewords.hpp b/libalee/corewords.hpp index 70cbb30..33f4fc5 100644 --- a/libalee/corewords.hpp +++ b/libalee/corewords.hpp @@ -27,7 +27,7 @@ void user_sys(State&); class CoreWords { public: - constexpr static std::size_t WordCount = 32; + constexpr static std::size_t WordCount = 33; constexpr static Cell Immediate = (1 << 5); @@ -44,7 +44,7 @@ public: "<\0&\0|\0^\0" "<<\0>>\0:\0_'\0execute\0" "exit\0;\0_jmp0\0_jmp\0" - "depth\0_rdepth\0_in\0"; + "depth\0_rdepth\0_in\0_ev\0"; }; #endif // ALEEFORTH_COREWORDS_HPP diff --git a/libalee/parser.cpp b/libalee/parser.cpp index 3b14224..39695c4 100644 --- a/libalee/parser.cpp +++ b/libalee/parser.cpp @@ -22,7 +22,7 @@ #include <cstring> -int Parser::parse(State& state, const char *str) +Error Parser::parse(State& state, const char *str) { auto addr = Dictionary::Input; state.dict.write(addr, 0); @@ -38,20 +38,20 @@ int Parser::parse(State& state, const char *str) return parseSource(state); } -int Parser::parseSource(State& state) +Error Parser::parseSource(State& state) { auto word = state.dict.input(); while (word.size() > 0) { - if (auto ret = parseWord(state, word); ret) + if (auto ret = parseWord(state, word); ret != Error::none) return ret; word = state.dict.input(); } - return 0; + return Error::none; } -int Parser::parseWord(State& state, Word word) +Error Parser::parseWord(State& state, Word word) { int ins, imm; @@ -71,13 +71,13 @@ int Parser::parseWord(State& state, Word word) if (state.compiling() && !imm) state.dict.add(ins); - else if (auto stat = state.execute(ins); stat != State::Error::none) - return static_cast<int>(stat); + else if (auto stat = state.execute(ins); stat != Error::none) + return stat; - return 0; + return Error::none; } -int Parser::parseNumber(State& state, Word word) +Error Parser::parseNumber(State& state, Word word) { const auto base = state.dict.read(Dictionary::Base); DoubleCell result = 0; @@ -97,7 +97,7 @@ int Parser::parseNumber(State& state, Word word) result *= base; result += 10 + (c > 'a' ? c - 'a' : c - 'A'); } else { - return UnknownWord; + return Error::noword; } if (++i < word.wend) @@ -121,6 +121,6 @@ int Parser::parseNumber(State& state, Word word) state.push(value); } - return 0; + return Error::none; } diff --git a/libalee/parser.hpp b/libalee/parser.hpp index f211781..be6f3f7 100644 --- a/libalee/parser.hpp +++ b/libalee/parser.hpp @@ -26,14 +26,12 @@ class Parser { public: - constexpr static int UnknownWord = -1; - - static int parse(State&, const char *); - static int parseSource(State&); + static Error parse(State&, const char *); + static Error parseSource(State&); private: - static int parseWord(State&, Word); - static int parseNumber(State&, Word); + static Error parseWord(State&, Word); + static Error parseNumber(State&, Word); }; #endif // ALEEFORTH_PARSER_HPP diff --git a/libalee/state.cpp b/libalee/state.cpp index b3c98b1..bfd29d1 100644 --- a/libalee/state.cpp +++ b/libalee/state.cpp @@ -19,6 +19,7 @@ #include "corewords.hpp" #include "state.hpp" +#include <cstring> #include <iterator> bool State::compiling() const @@ -31,11 +32,25 @@ void State::compiling(bool yes) dict.write(Dictionary::Compiling, yes); } -State::Error State::execute(Addr addr) +std::pair<Addr, std::jmp_buf> State::save() { - auto stat = static_cast<State::Error>(setjmp(jmpbuf)); + std::pair<Addr, std::jmp_buf> st; + st.first = ip; + std::memcpy(st.second, jmpbuf, sizeof(std::jmp_buf)); + return st; +} + +void State::load(const std::pair<Addr, std::jmp_buf>& st) +{ + ip = st.first; + std::memcpy(jmpbuf, st.second, sizeof(std::jmp_buf)); +} + +Error State::execute(Addr addr) +{ + auto stat = static_cast<Error>(setjmp(jmpbuf)); - if (stat == State::Error::none) { + if (stat == Error::none) { CoreWords::run(addr, *this); if (ip >= Dictionary::Begin) { @@ -46,13 +61,24 @@ State::Error State::execute(Addr addr) // addr was a CoreWord, all done now. ip = 0; } - } else if (stat == State::Error::exit) { - stat = State::Error::none; + } else if (stat == Error::exit) { + stat = Error::none; } return stat; } +void State::reset() +{ + while (size()) + pop(); + while (rsize()) + popr(); + + dict.write(Dictionary::Compiling, 0); + ip = 0; +} + std::size_t State::size() const noexcept { return std::distance(dstack, static_cast<const Cell *>(dsp)); diff --git a/libalee/state.hpp b/libalee/state.hpp index 648b868..325f405 100644 --- a/libalee/state.hpp +++ b/libalee/state.hpp @@ -24,6 +24,7 @@ #include <csetjmp> #include <cstddef> +#include <tuple> constexpr unsigned DataStackSize = 16; constexpr unsigned ReturnStackSize = 16; @@ -31,17 +32,6 @@ constexpr unsigned ReturnStackSize = 16; class State { public: - enum class Error : int { - none = 0, - push, - pop, - pushr, - popr, - top, - pick, - exit - }; - Addr ip = 0; Dictionary& dict; void (*input)(State&); @@ -54,7 +44,11 @@ public: bool compiling() const; void compiling(bool); + std::pair<Addr, std::jmp_buf> save(); + void load(const std::pair<Addr, std::jmp_buf>&); + Error execute(Addr); + void reset(); std::size_t size() const noexcept; std::size_t rsize() const noexcept; diff --git a/libalee/types.hpp b/libalee/types.hpp index 5b577f2..0a7f7ee 100644 --- a/libalee/types.hpp +++ b/libalee/types.hpp @@ -32,6 +32,18 @@ using Func = void (*)(State&); constexpr unsigned int MaxCellNumberChars = 6; // -32768 +enum class Error : int { + none = 0, + push, + pop, + pushr, + popr, + top, + pick, + exit, + noword +}; + struct Word { struct iterator; |