1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 03:53:54 +00:00
openmw/components/interpreter/interpreter.cpp

258 lines
6.6 KiB
C++
Raw Normal View History

2010-06-28 17:20:45 +00:00
#include "interpreter.hpp"
2010-06-28 18:46:15 +00:00
#include <cassert>
#include <sstream>
#include <stdexcept>
#include "opcodes.hpp"
2010-06-28 17:20:45 +00:00
namespace Interpreter
{
2010-06-28 18:46:15 +00:00
void Interpreter::execute (Type_Code code)
{
unsigned int segSpec = code>>30;
2010-06-28 18:46:15 +00:00
switch (segSpec)
{
case 0:
{
int opcode = code>>24;
unsigned int arg0 = code & 0xffffff;
2010-06-28 18:46:15 +00:00
std::map<int, Opcode1 *>::iterator iter = mSegment0.find (opcode);
2010-06-28 18:46:15 +00:00
if (iter==mSegment0.end())
abortUnknownCode (0, opcode);
2010-06-28 18:46:15 +00:00
iter->second->execute (mRuntime, arg0);
2010-06-28 18:46:15 +00:00
return;
}
2010-06-28 18:46:15 +00:00
case 1:
{
int opcode = (code>>24) & 0x3f;
unsigned int arg0 = (code>>16) & 0xfff;
unsigned int arg1 = code & 0xfff;
2010-06-28 18:46:15 +00:00
std::map<int, Opcode2 *>::iterator iter = mSegment1.find (opcode);
2010-06-28 18:46:15 +00:00
if (iter==mSegment1.end())
abortUnknownCode (1, opcode);
2010-06-28 18:46:15 +00:00
iter->second->execute (mRuntime, arg0, arg1);
2010-06-28 18:46:15 +00:00
return;
}
2010-06-28 18:46:15 +00:00
case 2:
{
int opcode = (code>>20) & 0x3ff;
unsigned int arg0 = code & 0xfffff;
2010-06-28 18:46:15 +00:00
std::map<int, Opcode1 *>::iterator iter = mSegment2.find (opcode);
2010-06-28 18:46:15 +00:00
if (iter==mSegment2.end())
abortUnknownCode (2, opcode);
2010-06-28 18:46:15 +00:00
iter->second->execute (mRuntime, arg0);
2010-06-28 18:46:15 +00:00
return;
}
2010-06-28 18:46:15 +00:00
}
2010-06-28 18:46:15 +00:00
segSpec = code>>26;
2010-06-28 18:46:15 +00:00
switch (segSpec)
{
2010-06-28 18:46:15 +00:00
case 0x30:
{
int opcode = (code>>8) & 0x3ffff;
unsigned int arg0 = code & 0xff;
2010-06-28 18:46:15 +00:00
std::map<int, Opcode1 *>::iterator iter = mSegment3.find (opcode);
2010-06-28 18:46:15 +00:00
if (iter==mSegment3.end())
abortUnknownCode (3, opcode);
2010-06-28 18:46:15 +00:00
iter->second->execute (mRuntime, arg0);
2010-06-28 18:46:15 +00:00
return;
}
2010-06-28 18:46:15 +00:00
case 0x31:
{
int opcode = (code>>16) & 0x3ff;
unsigned int arg0 = (code>>8) & 0xff;
unsigned int arg1 = code & 0xff;
2010-06-28 18:46:15 +00:00
std::map<int, Opcode2 *>::iterator iter = mSegment4.find (opcode);
2010-06-28 18:46:15 +00:00
if (iter==mSegment4.end())
abortUnknownCode (4, opcode);
2010-06-28 18:46:15 +00:00
iter->second->execute (mRuntime, arg0, arg1);
2010-06-28 18:46:15 +00:00
return;
}
2010-06-28 18:46:15 +00:00
case 0x32:
{
int opcode = code & 0x3ffffff;
2010-06-28 18:46:15 +00:00
std::map<int, Opcode0 *>::iterator iter = mSegment5.find (opcode);
2010-06-28 18:46:15 +00:00
if (iter==mSegment5.end())
abortUnknownCode (5, opcode);
2010-06-28 18:46:15 +00:00
iter->second->execute (mRuntime);
2010-06-28 18:46:15 +00:00
return;
}
}
2010-06-28 18:46:15 +00:00
abortUnknownSegment (code);
}
void Interpreter::abortUnknownCode (int segment, int opcode)
{
std::ostringstream error;
2010-06-28 18:46:15 +00:00
error << "unknown opcode " << opcode << " in segment " << segment;
2010-06-28 18:46:15 +00:00
throw std::runtime_error (error.str());
}
2010-06-28 18:46:15 +00:00
void Interpreter::abortUnknownSegment (Type_Code code)
{
std::ostringstream error;
2010-06-28 18:46:15 +00:00
error << "opcode outside of the allocated segment range: " << code;
2010-06-28 18:46:15 +00:00
throw std::runtime_error (error.str());
}
void Interpreter::begin()
{
if (mRunning)
{
mCallstack.push (mRuntime);
mRuntime.clear();
}
else
{
mRunning = true;
}
}
void Interpreter::end()
{
if (mCallstack.empty())
{
mRuntime.clear();
mRunning = false;
}
else
{
mRuntime = mCallstack.top();
mCallstack.pop();
}
}
Interpreter::Interpreter() : mRunning (false)
2010-06-28 17:20:45 +00:00
{}
2010-06-28 17:20:45 +00:00
Interpreter::~Interpreter()
{
2010-06-28 18:46:15 +00:00
for (std::map<int, Opcode1 *>::iterator iter (mSegment0.begin());
iter!=mSegment0.end(); ++iter)
delete iter->second;
for (std::map<int, Opcode2 *>::iterator iter (mSegment1.begin());
iter!=mSegment1.end(); ++iter)
delete iter->second;
for (std::map<int, Opcode1 *>::iterator iter (mSegment2.begin());
iter!=mSegment2.end(); ++iter)
delete iter->second;
for (std::map<int, Opcode1 *>::iterator iter (mSegment3.begin());
iter!=mSegment3.end(); ++iter)
delete iter->second;
for (std::map<int, Opcode2 *>::iterator iter (mSegment4.begin());
iter!=mSegment4.end(); ++iter)
delete iter->second;
2010-06-28 18:46:15 +00:00
for (std::map<int, Opcode0 *>::iterator iter (mSegment5.begin());
iter!=mSegment5.end(); ++iter)
delete iter->second;
2010-06-28 17:20:45 +00:00
}
2010-06-28 19:49:48 +00:00
void Interpreter::installSegment0 (int code, Opcode1 *opcode)
{
assert(mSegment0.find(code) == mSegment0.end());
2010-06-28 19:49:48 +00:00
mSegment0.insert (std::make_pair (code, opcode));
}
void Interpreter::installSegment1 (int code, Opcode2 *opcode)
{
assert(mSegment1.find(code) == mSegment1.end());
2010-06-28 19:49:48 +00:00
mSegment1.insert (std::make_pair (code, opcode));
}
void Interpreter::installSegment2 (int code, Opcode1 *opcode)
{
assert(mSegment2.find(code) == mSegment2.end());
2010-06-28 19:49:48 +00:00
mSegment2.insert (std::make_pair (code, opcode));
}
void Interpreter::installSegment3 (int code, Opcode1 *opcode)
{
assert(mSegment3.find(code) == mSegment3.end());
2010-06-28 19:49:48 +00:00
mSegment3.insert (std::make_pair (code, opcode));
}
void Interpreter::installSegment4 (int code, Opcode2 *opcode)
{
assert(mSegment4.find(code) == mSegment4.end());
2010-06-28 19:49:48 +00:00
mSegment4.insert (std::make_pair (code, opcode));
}
void Interpreter::installSegment5 (int code, Opcode0 *opcode)
{
assert(mSegment5.find(code) == mSegment5.end());
2010-06-28 19:49:48 +00:00
mSegment5.insert (std::make_pair (code, opcode));
}
void Interpreter::run (const Type_Code *code, int codeSize, Context& context)
2010-06-28 17:20:45 +00:00
{
2010-06-28 18:46:15 +00:00
assert (codeSize>=4);
begin();
try
{
mRuntime.configure (code, codeSize, context);
int opcodes = static_cast<int> (code[0]);
const Type_Code *codeBlock = code + 4;
while (mRuntime.getPC()>=0 && mRuntime.getPC()<opcodes)
{
2016-08-29 10:20:00 +00:00
Type_Code runCode = codeBlock[mRuntime.getPC()];
mRuntime.setPC (mRuntime.getPC()+1);
2016-08-29 10:20:00 +00:00
execute (runCode);
}
}
catch (...)
2010-06-28 18:46:15 +00:00
{
end();
throw;
2010-06-28 18:46:15 +00:00
}
end();
2010-06-28 17:20:45 +00:00
}
}