2020-09-06 12:57:28 +02:00
|
|
|
//
|
|
|
|
// interpreter.cpp
|
|
|
|
// curly
|
|
|
|
//
|
|
|
|
// Created by Bob Polis at 2020-09-05
|
|
|
|
// Copyright (c) 2020 SwiftCoder. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "interpreter.hpp"
|
2020-09-07 11:40:58 +02:00
|
|
|
#include <cctype>
|
|
|
|
#include <algorithm>
|
2020-09-22 13:38:17 +02:00
|
|
|
#include <sstream>
|
2020-09-07 11:40:58 +02:00
|
|
|
|
|
|
|
int to_int(const std::string& val) {
|
|
|
|
std::istringstream iss {val};
|
|
|
|
int result;
|
|
|
|
iss >> result;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::reset() {
|
|
|
|
_prog.clear();
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.clear();
|
2020-09-07 11:40:58 +02:00
|
|
|
_labels.clear();
|
|
|
|
_vars.clear();
|
|
|
|
_calls.clear();
|
2020-09-22 10:51:20 +02:00
|
|
|
_pc_offsets.clear();
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string interpreter::eval(std::istream& in, bool& done) {
|
|
|
|
std::string result;
|
|
|
|
reset();
|
|
|
|
done = false;
|
|
|
|
|
2020-09-22 11:02:41 +02:00
|
|
|
// first pass: read program & resolve labels, ignore comments
|
2020-09-22 10:51:20 +02:00
|
|
|
for (std::string line; std::getline(in, line);) {
|
2020-09-21 16:37:40 +02:00
|
|
|
if (line[0] == ':') { // check label definition
|
2020-09-22 10:51:20 +02:00
|
|
|
_labels.emplace(line.substr(1), _prog.size());
|
|
|
|
_pc_offsets.push_back(_prog.size());
|
2020-09-21 12:41:29 +02:00
|
|
|
} else if (line[0] == '#') { // check comment
|
2020-09-22 10:51:20 +02:00
|
|
|
_pc_offsets.push_back(_prog.size());
|
2020-09-07 11:40:58 +02:00
|
|
|
} else {
|
|
|
|
_prog.push_back(line);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// second pass: run program
|
|
|
|
while (_pc < _prog.size() && !done) {
|
|
|
|
// fetch next instruction
|
|
|
|
std::string code {_prog[_pc]};
|
2020-09-21 10:45:49 +02:00
|
|
|
|
2020-09-07 11:40:58 +02:00
|
|
|
// check literal int
|
|
|
|
if (std::isdigit(code[0])) {
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(code);
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check literal string
|
|
|
|
if (code[0] == '\\') {
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(code.substr(1));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check label ref
|
|
|
|
if (code[0] == '>') {
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(code.substr(1));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check var assignment
|
|
|
|
if (code[0] == '=') {
|
2020-09-22 13:13:50 +02:00
|
|
|
_vars[code.substr(1)] = _values.back();
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// check var ref
|
|
|
|
if (code[0] == '$') {
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(_vars[code.substr(1)]);
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// exec instruction
|
|
|
|
exec_instruction(code, done);
|
|
|
|
}
|
|
|
|
|
2020-09-22 13:13:50 +02:00
|
|
|
if (_values.size()) {
|
|
|
|
result = _values.back();
|
2020-09-07 11:40:58 +02:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::exec_instruction(const std::string& code, bool& done) {
|
|
|
|
if (code == "end") done = true;
|
|
|
|
else if (code == "add") add();
|
|
|
|
else if (code == "sub") sub();
|
|
|
|
else if (code == "mul") mul();
|
|
|
|
else if (code == "div") div();
|
|
|
|
else if (code == "mod") mod();
|
|
|
|
else if (code == "abs") abs();
|
|
|
|
else if (code == "neg") neg();
|
|
|
|
else if (code == "dup") dup();
|
|
|
|
else if (code == "inc") inc();
|
|
|
|
else if (code == "dec") dec();
|
|
|
|
else if (code == "rev") rev();
|
|
|
|
else if (code == "slc") slc();
|
|
|
|
else if (code == "idx") idx();
|
|
|
|
else if (code == "cat") cat();
|
|
|
|
else if (code == "len") len();
|
|
|
|
else if (code == "rot") rot();
|
|
|
|
else if (code == "gto") gto();
|
|
|
|
else if (code == "geq") geq();
|
|
|
|
else if (code == "gne") gne();
|
|
|
|
else if (code == "glt") glt();
|
|
|
|
else if (code == "gle") gle();
|
|
|
|
else if (code == "ggt") ggt();
|
|
|
|
else if (code == "gge") gge();
|
|
|
|
else if (code == "fun") fun();
|
|
|
|
else if (code == "ret") ret();
|
2020-09-07 18:50:09 +02:00
|
|
|
else if (code == "enl") enl();
|
2020-09-21 10:17:16 +02:00
|
|
|
else if (code == "inp") inp();
|
|
|
|
else if (code == "out") out();
|
|
|
|
else if (code == "err") err();
|
2020-09-21 16:37:40 +02:00
|
|
|
else {
|
|
|
|
auto it = std::upper_bound(_pc_offsets.begin(), _pc_offsets.end(), _pc);
|
|
|
|
size_t lineno {it - _pc_offsets.begin() + _pc + 1};
|
|
|
|
throw syntax_error {code, lineno};
|
|
|
|
}
|
2020-09-07 11:40:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// integer operations -----------------------------------------------------
|
|
|
|
|
|
|
|
void interpreter::add() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val1 + val2));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::sub() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val1 - val2));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::mul() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val1 * val2));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::div() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val1 / val2));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::mod() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val1 % val2));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::abs() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val < 0 ? -val : val));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::neg() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(-val));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::inc() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val {to_int(_values.back()) + 1};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::dec() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int val {to_int(_values.back()) - 1};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// string operations ------------------------------------------------------
|
|
|
|
|
|
|
|
void interpreter::dup() {
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(_values.back());
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::rev() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string val {_values.back()};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
std::string result;
|
|
|
|
for (auto it = val.rbegin(); it != val.rend(); ++it) {
|
|
|
|
result += *it;
|
|
|
|
}
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(result);
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::slc() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int to {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int from {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(val.substr(from, to - from));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::idx() {
|
2020-09-22 13:13:50 +02:00
|
|
|
int idx {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val {_values.back()};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
val = val[idx];
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.emplace_back(val);
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::cat() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string val2 {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val1 {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.emplace_back(val1 + val2);
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::len() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string val {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
_values.push_back(std::to_string(val.size()));
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::rot() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string val {_values.back()};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
std::transform(val.begin(), val.end(), val.begin(), [](char ch) -> char {
|
|
|
|
if (ch >= 'a' && ch <= 'm') {
|
|
|
|
return ch + 13;
|
|
|
|
} else if (ch >= 'n' && ch <= 'z') {
|
|
|
|
return ch - 13;
|
|
|
|
} else if (ch >= 'A' && ch <= 'M') {
|
|
|
|
return ch + 13;
|
|
|
|
} else if (ch >= 'N' && ch <= 'Z') {
|
|
|
|
return ch - 13;
|
|
|
|
}
|
|
|
|
return ch;
|
|
|
|
});
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(val);
|
2020-09-07 11:40:58 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
2020-09-07 18:50:09 +02:00
|
|
|
void interpreter::enl() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string val {_values.back()};
|
|
|
|
_values.pop_back();
|
2020-09-07 18:50:09 +02:00
|
|
|
val += '\n';
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(val);
|
2020-09-07 18:50:09 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
2020-09-07 11:40:58 +02:00
|
|
|
// tests & jumps ----------------------------------------------------------
|
|
|
|
|
|
|
|
void interpreter::gto() {
|
2020-09-22 13:13:50 +02:00
|
|
|
_pc = _labels[_values.back()];
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::geq() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string label {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val2 {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val1 {_values.back()};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
if (val1 == val2) {
|
|
|
|
_pc = _labels[label];
|
|
|
|
} else {
|
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::gne() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string label {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val2 {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
std::string val1 {_values.back()};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
if (val1 != val2) {
|
|
|
|
_pc = _labels[label];
|
|
|
|
} else {
|
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::glt() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string label {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
if (val1 < val2) {
|
|
|
|
_pc = _labels[label];
|
|
|
|
} else {
|
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::gle() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string label {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
if (val1 <= val2) {
|
|
|
|
_pc = _labels[label];
|
|
|
|
} else {
|
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::ggt() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string label {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
if (val1 > val2) {
|
|
|
|
_pc = _labels[label];
|
|
|
|
} else {
|
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::gge() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::string label {_values.back()};
|
|
|
|
_values.pop_back();
|
|
|
|
int val2 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
|
|
|
int val1 {to_int(_values.back())};
|
|
|
|
_values.pop_back();
|
2020-09-07 11:40:58 +02:00
|
|
|
if (val1 >= val2) {
|
|
|
|
_pc = _labels[label];
|
|
|
|
} else {
|
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// subroutines ------------------------------------------------------------
|
|
|
|
|
|
|
|
void interpreter::fun() {
|
|
|
|
_calls.push_back(_pc + 1); // push return address
|
|
|
|
gto();
|
|
|
|
}
|
|
|
|
|
|
|
|
void interpreter::ret() {
|
|
|
|
_pc = _calls.back();
|
|
|
|
_calls.pop_back();
|
|
|
|
}
|
2020-09-13 16:55:02 +02:00
|
|
|
|
|
|
|
// debugging --------------------------------------------------------------
|
|
|
|
|
2020-09-21 10:17:16 +02:00
|
|
|
void interpreter::inp() {
|
|
|
|
std::string val;
|
|
|
|
std::cin >> val;
|
2020-09-22 13:13:50 +02:00
|
|
|
_values.push_back(val);
|
2020-09-21 10:17:16 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
|
|
|
|
2020-09-13 16:55:02 +02:00
|
|
|
void interpreter::out() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::cout << _values.back() << '\n';
|
2020-09-13 16:55:02 +02:00
|
|
|
_pc++;
|
|
|
|
}
|
2020-09-21 10:17:16 +02:00
|
|
|
|
|
|
|
void interpreter::err() {
|
2020-09-22 13:13:50 +02:00
|
|
|
std::cerr << _values.back() << '\n';
|
2020-09-21 10:17:16 +02:00
|
|
|
_pc++;
|
|
|
|
}
|