2010-06-28 19:20:45 +02:00
|
|
|
#include "runtime.hpp"
|
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
#include <stdexcept>
|
|
|
|
#include <cassert>
|
2010-06-30 12:04:26 +02:00
|
|
|
#include <cstring>
|
2010-06-28 21:49:48 +02:00
|
|
|
|
2010-06-28 19:20:45 +02:00
|
|
|
namespace Interpreter
|
|
|
|
{
|
2020-11-13 11:39:47 +04:00
|
|
|
Runtime::Runtime() : mContext (nullptr), mCode (nullptr), mCodeSize(0), mPC (0) {}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 20:46:15 +02:00
|
|
|
int Runtime::getPC() const
|
|
|
|
{
|
|
|
|
return mPC;
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
int Runtime::getIntegerLiteral (int index) const
|
|
|
|
{
|
2019-05-21 17:58:10 -05:00
|
|
|
if (index < 0 || index >= static_cast<int> (mCode[1]))
|
|
|
|
throw std::out_of_range("out of range");
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
const Type_Code *literalBlock = mCode + 4 + mCode[0];
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
return *reinterpret_cast<const int *> (&literalBlock[index]);
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
float Runtime::getFloatLiteral (int index) const
|
|
|
|
{
|
2019-05-21 17:58:10 -05:00
|
|
|
if (index < 0 || index >= static_cast<int> (mCode[2]))
|
|
|
|
throw std::out_of_range("out of range");
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
const Type_Code *literalBlock = mCode + 4 + mCode[0] + mCode[1];
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
return *reinterpret_cast<const float *> (&literalBlock[index]);
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2022-05-21 01:21:55 +02:00
|
|
|
std::string_view Runtime::getStringLiteral(int index) const
|
2010-06-30 12:04:26 +02:00
|
|
|
{
|
2019-05-21 17:58:10 -05:00
|
|
|
if (index < 0 || static_cast<int> (mCode[3]) <= 0)
|
|
|
|
throw std::out_of_range("out of range");
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-30 12:04:26 +02:00
|
|
|
const char *literalBlock =
|
|
|
|
reinterpret_cast<const char *> (mCode + 4 + mCode[0] + mCode[1] + mCode[2]);
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2022-05-21 01:21:55 +02:00
|
|
|
size_t offset = 0;
|
2013-04-12 14:48:53 +02:00
|
|
|
|
2010-06-30 12:04:26 +02:00
|
|
|
for (; index; --index)
|
|
|
|
{
|
2022-05-21 01:21:55 +02:00
|
|
|
offset += std::strlen(literalBlock + offset) + 1;
|
|
|
|
if (offset / 4 >= mCode[3])
|
2019-05-21 17:58:10 -05:00
|
|
|
throw std::out_of_range("out of range");
|
2010-06-30 12:04:26 +02:00
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2013-04-12 14:48:53 +02:00
|
|
|
return literalBlock+offset;
|
2010-06-30 12:04:26 +02:00
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2014-03-16 23:38:51 +01:00
|
|
|
void Runtime::configure (const Type_Code *code, int codeSize, Context& context)
|
2011-05-18 16:01:19 +02:00
|
|
|
{
|
2010-06-28 19:20:45 +02:00
|
|
|
clear();
|
2011-05-18 16:01:19 +02:00
|
|
|
|
|
|
|
mContext = &context;
|
2010-06-28 19:20:45 +02:00
|
|
|
mCode = code;
|
|
|
|
mCodeSize = codeSize;
|
2010-06-28 20:46:15 +02:00
|
|
|
mPC = 0;
|
2010-06-28 19:20:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void Runtime::clear()
|
|
|
|
{
|
2020-11-13 11:39:47 +04:00
|
|
|
mContext = nullptr;
|
|
|
|
mCode = nullptr;
|
2010-06-28 19:20:45 +02:00
|
|
|
mCodeSize = 0;
|
2010-06-28 21:49:48 +02:00
|
|
|
mStack.clear();
|
2010-06-28 19:20:45 +02:00
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 20:46:15 +02:00
|
|
|
void Runtime::setPC (int PC)
|
|
|
|
{
|
|
|
|
mPC = PC;
|
2011-05-18 16:01:19 +02:00
|
|
|
}
|
|
|
|
|
2010-07-14 15:28:55 +02:00
|
|
|
void Runtime::push (const Data& data)
|
2010-06-28 21:49:48 +02:00
|
|
|
{
|
|
|
|
mStack.push_back (data);
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-07-14 15:28:55 +02:00
|
|
|
void Runtime::push (Type_Integer value)
|
|
|
|
{
|
|
|
|
Data data;
|
|
|
|
data.mInteger = value;
|
|
|
|
push (data);
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-07-14 15:28:55 +02:00
|
|
|
void Runtime::push (Type_Float value)
|
|
|
|
{
|
|
|
|
Data data;
|
|
|
|
data.mFloat = value;
|
|
|
|
push (data);
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
void Runtime::pop()
|
|
|
|
{
|
|
|
|
if (mStack.empty())
|
|
|
|
throw std::runtime_error ("stack underflow");
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2022-01-28 16:26:43 +02:00
|
|
|
mStack.pop_back();
|
2010-06-28 21:49:48 +02:00
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-07-14 15:28:55 +02:00
|
|
|
Data& Runtime::operator[] (int Index)
|
2010-06-28 21:49:48 +02:00
|
|
|
{
|
|
|
|
if (Index<0 || Index>=static_cast<int> (mStack.size()))
|
|
|
|
throw std::runtime_error ("stack index out of range");
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
return mStack[mStack.size()-Index-1];
|
|
|
|
}
|
2011-05-18 16:01:19 +02:00
|
|
|
|
2010-06-28 21:49:48 +02:00
|
|
|
Context& Runtime::getContext()
|
|
|
|
{
|
2011-05-18 16:01:19 +02:00
|
|
|
assert (mContext);
|
|
|
|
return *mContext;
|
2010-06-28 21:49:48 +02:00
|
|
|
}
|
2010-06-28 19:20:45 +02:00
|
|
|
}
|