2010-06-29 16:11:19 +02:00
|
|
|
#ifndef INTERPRETER_MATHOPCODES_H_INCLUDED
|
|
|
|
#define INTERPRETER_MATHOPCODES_H_INCLUDED
|
|
|
|
|
|
|
|
#include <stdexcept>
|
2010-06-29 17:51:02 +02:00
|
|
|
#include <cmath>
|
2010-06-29 16:11:19 +02:00
|
|
|
|
|
|
|
#include "opcodes.hpp"
|
|
|
|
#include "runtime.hpp"
|
|
|
|
|
|
|
|
namespace Interpreter
|
|
|
|
{
|
|
|
|
template<typename T>
|
|
|
|
class OpAddInt : public Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Runtime& runtime)
|
|
|
|
{
|
|
|
|
T result =
|
|
|
|
*reinterpret_cast<T *> (&runtime[1])
|
|
|
|
+
|
|
|
|
*reinterpret_cast<T *> (&runtime[0]);
|
|
|
|
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
runtime[0] = *reinterpret_cast<Type_Data *> (&result);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class OpSubInt : public Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Runtime& runtime)
|
|
|
|
{
|
|
|
|
T result =
|
|
|
|
*reinterpret_cast<T *> (&runtime[1])
|
|
|
|
-
|
|
|
|
*reinterpret_cast<T *> (&runtime[0]);
|
|
|
|
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
runtime[0] = *reinterpret_cast<Type_Data *> (&result);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class OpMulInt : public Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Runtime& runtime)
|
|
|
|
{
|
|
|
|
T result =
|
|
|
|
*reinterpret_cast<T *> (&runtime[1])
|
|
|
|
*
|
|
|
|
*reinterpret_cast<T *> (&runtime[0]);
|
|
|
|
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
runtime[0] = *reinterpret_cast<Type_Data *> (&result);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
class OpDivInt : public Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Runtime& runtime)
|
|
|
|
{
|
|
|
|
T left = *reinterpret_cast<T *> (&runtime[0]);
|
|
|
|
|
|
|
|
if (left==0)
|
|
|
|
throw std::runtime_error ("division by zero");
|
|
|
|
|
|
|
|
T result =
|
|
|
|
*reinterpret_cast<T *> (&runtime[1])
|
|
|
|
/
|
|
|
|
left;
|
|
|
|
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
runtime[0] = *reinterpret_cast<Type_Data *> (&result);
|
|
|
|
}
|
|
|
|
};
|
2010-06-29 17:51:02 +02:00
|
|
|
|
|
|
|
class OpSquareRoot : public Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Runtime& runtime)
|
|
|
|
{
|
|
|
|
Type_Float value = *reinterpret_cast<Type_Float *> (&runtime[0]);
|
|
|
|
|
|
|
|
if (value<0)
|
|
|
|
throw std::runtime_error (
|
|
|
|
"square root of negative number (we aren't that imaginary)");
|
|
|
|
|
|
|
|
value = std::sqrt (value);
|
|
|
|
|
|
|
|
runtime[0] = *reinterpret_cast<Type_Data *> (&value);
|
|
|
|
}
|
|
|
|
};
|
2010-07-01 12:19:52 +02:00
|
|
|
|
|
|
|
template<typename T, typename C>
|
|
|
|
class OpCompare : public Opcode0
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
|
|
|
virtual void execute (Runtime& runtime)
|
|
|
|
{
|
|
|
|
int result = C() (
|
|
|
|
*reinterpret_cast<T *> (&runtime[1]),
|
|
|
|
*reinterpret_cast<T *> (&runtime[0]));
|
|
|
|
|
|
|
|
runtime.pop();
|
|
|
|
|
|
|
|
runtime[0] = *reinterpret_cast<Type_Data *> (&result);
|
|
|
|
}
|
|
|
|
};
|
2010-06-29 16:11:19 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|