aboutsummaryrefslogtreecommitdiffstats
path: root/libalee/parser.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'libalee/parser.cpp')
-rw-r--r--libalee/parser.cpp64
1 files changed, 39 insertions, 25 deletions
diff --git a/libalee/parser.cpp b/libalee/parser.cpp
index 44b5cec..d709d45 100644
--- a/libalee/parser.cpp
+++ b/libalee/parser.cpp
@@ -19,7 +19,7 @@
#include "corewords.hpp"
#include "parser.hpp"
-#include <charconv>
+#include <cctype>
#include <cstring>
int Parser::parse(State& state, const char *str)
@@ -79,34 +79,48 @@ int Parser::parseWord(State& state, Word word)
int Parser::parseNumber(State& state, Word word)
{
- char buf[MaxCellNumberChars + 1];
- unsigned i;
- for (i = 0; i < std::min(MaxCellNumberChars, word.size()); ++i)
- buf[i] = state.dict.readbyte(word.start + i);
- buf[i] = '\0';
-
- auto base = state.dict.read(0);
- DoubleCell dl;
- auto [ptr, ec] = std::from_chars(buf, buf + i, dl, base);
- Cell l = static_cast<Cell>(dl);
-
- if (ec == std::errc() && ptr == buf + i) {
- if (state.compiling()) {
- auto ins = CoreWords::findi("_lit");
-
- //if (l >= 0 && l < 0xFF) {
- // state.dict.add(ins | ((l + 1) << 8));
- //} else {
- state.dict.add(ins);
- state.dict.add(l);
- //}
+ const auto base = state.dict.read(Dictionary::Base);
+ DoubleCell result = 0;
+ auto i = word.start;
+ bool inv;
+ char c;
+
+ c = state.dict.readbyte(i);
+ if (inv = c == '-'; inv)
+ c = state.dict.readbyte(++i);
+
+ do {
+ if (isdigit(c)) {
+ result *= base;
+ result += c - '0';
+ } else if (isalpha(c) && base > 10) {
+ result *= base;
+ result += 10 + (c > 'a' ? c - 'a' : c - 'A');
} else {
- state.push(l);
+ return UnknownWord;
}
- return 0;
+ if (++i < word.end)
+ c = state.dict.readbyte(i);
+ } while (i < word.end);
+
+ if (inv)
+ result *= -1;
+
+ Cell value = static_cast<Cell>(result);
+ if (state.compiling()) {
+ auto ins = CoreWords::findi("_lit");
+
+ //if (l >= 0 && l < 0xFF) {
+ // state.dict.add(ins | ((l + 1) << 8));
+ //} else {
+ state.dict.add(ins);
+ state.dict.add(value);
+ //}
} else {
- return UnknownWord;
+ state.push(value);
}
+
+ return 0;
}