mirror of
https://github.com/ZDoom/gzdoom-gles.git
synced 2025-01-22 00:11:38 +00:00
Merge pull request #538 from Gutawer/asmjit
Added all integer comparison opcodes
This commit is contained in:
commit
f862b96763
1 changed files with 177 additions and 13 deletions
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue