Merge pull request #538 from Gutawer/asmjit

Added all integer comparison opcodes
This commit is contained in:
Magnus Norddahl 2018-08-14 16:47:18 +02:00 committed by GitHub
commit f862b96763
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -8,6 +8,7 @@
#include <asmjit/asmjit.h> #include <asmjit/asmjit.h>
#include <asmjit/x86.h> #include <asmjit/x86.h>
#include <functional>
class AsmJitException : public std::exception class AsmJitException : public std::exception
{ {
@ -221,6 +222,29 @@ static bool CanJit(VMScriptFunction *sfunc)
return true; return true;
} }
template <typename Func>
void emitComparisonOpcode(asmjit::X86Compiler& cc, const TArray<asmjit::Label>& labels, const VMOP* pc, int i, Func compFunc) {
using namespace asmjit;
Label elseLabel = cc.newLabel();
auto tmp0 = cc.newInt32();
auto tmp1 = cc.newInt32();
compFunc(tmp0);
cc.mov(tmp1, A);
cc.and_(tmp1, CMP_CHECK);
cc.cmp(tmp0, tmp1);
cc.jne(elseLabel);
cc.jmp(labels[i + 2 + JMPOFS(pc + 1)]);
cc.bind(elseLabel);
cc.jmp(labels[i + 2]);
}
JitFuncPtr JitCompile(VMScriptFunction *sfunc) JitFuncPtr JitCompile(VMScriptFunction *sfunc)
{ {
#if 0 // For debugging #if 0 // For debugging
@ -309,6 +333,11 @@ JitFuncPtr JitCompile(VMScriptFunction *sfunc)
*/ */
int size = sfunc->CodeSize; int size = sfunc->CodeSize;
TArray<Label> labels(size, true);
for (int i = 0; i < size; i++) labels[i] = cc.newLabel();
for (int i = 0; i < size; i++) for (int i = 0; i < size; i++)
{ {
const VMOP *pc = sfunc->Code + i; const VMOP *pc = sfunc->Code + i;
@ -316,6 +345,8 @@ JitFuncPtr JitCompile(VMScriptFunction *sfunc)
int a = pc->a; int a = pc->a;
int b;// , c; int b;// , c;
cc.bind(labels[i]);
switch (op) switch (op)
{ {
default: default:
@ -905,20 +936,153 @@ JitFuncPtr JitCompile(VMScriptFunction *sfunc)
cc.not_(regD[a]); cc.not_(regD[a]);
break; break;
case OP_EQ_R: // if ((dB == dkC) != A) then pc++ case OP_EQ_R: // if ((dB == dkC) != A) then pc++
case OP_EQ_K: {
case OP_LT_RR: // if ((dkB < dkC) != A) then pc++ auto compLambda = [&] (X86Gp& result) {
case OP_LT_RK: cc.cmp(regD[B], regD[C]);
case OP_LT_KR: cc.sete(result);
case OP_LE_RR: // if ((dkB <= dkC) != A) then pc++ };
case OP_LE_RK: emitComparisonOpcode(cc, labels, pc, i, compLambda);
case OP_LE_KR:
case OP_LTU_RR: // if ((dkB < dkC) != A) then pc++ -- unsigned
case OP_LTU_RK:
case OP_LTU_KR:
case OP_LEU_RR: // if ((dkB <= dkC) != A) then pc++ -- unsigned
case OP_LEU_RK:
case OP_LEU_KR:
break; break;
}
case OP_EQ_K:
{
auto compLambda = [&] (X86Gp& result) {
cc.cmp(regD[B], konstd[C]);
cc.sete(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LT_RR: // if ((dkB < dkC) != A) then pc++
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], regD[C]);
cc.setl(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LT_RK:
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], konstd[C]);
cc.setl(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LT_KR:
{
auto compLambda = [&](X86Gp& result) {
auto tmp = cc.newIntPtr();
cc.mov(tmp, reinterpret_cast<ptrdiff_t>(&(konstd[C])));
cc.cmp(x86::ptr(tmp), regD[B]);
cc.setl(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LE_RR: // if ((dkB <= dkC) != A) then pc++
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], regD[C]);
cc.setle(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LE_RK:
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], konstd[C]);
cc.setle(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LE_KR:
{
auto compLambda = [&](X86Gp& result) {
auto tmp = cc.newIntPtr();
cc.mov(tmp, reinterpret_cast<ptrdiff_t>(&(konstd[C])));
cc.cmp(x86::ptr(tmp), regD[B]);
cc.setle(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LTU_RR: // if ((dkB < dkC) != A) then pc++ -- unsigned
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], regD[C]);
cc.setb(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LTU_RK:
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], konstd[C]);
cc.setb(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LTU_KR:
{
auto compLambda = [&](X86Gp& result) {
auto tmp = cc.newIntPtr();
cc.mov(tmp, reinterpret_cast<ptrdiff_t>(&(konstd[C])));
cc.cmp(x86::ptr(tmp), regD[B]);
cc.setb(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LEU_RR: // if ((dkB <= dkC) != A) then pc++ -- unsigned
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], regD[C]);
cc.setbe(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LEU_RK:
{
auto compLambda = [&](X86Gp& result) {
cc.cmp(regD[B], konstd[C]);
cc.setbe(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
case OP_LEU_KR:
{
auto compLambda = [&](X86Gp& result) {
auto tmp = cc.newIntPtr();
cc.mov(tmp, reinterpret_cast<ptrdiff_t>(&(konstd[C])));
cc.cmp(x86::ptr(tmp), regD[B]);
cc.setbe(result);
};
emitComparisonOpcode(cc, labels, pc, i, compLambda);
break;
}
// Double-precision floating point math. // Double-precision floating point math.
case OP_ADDF_RR: // fA = fB + fkC case OP_ADDF_RR: // fA = fB + fkC