2020-04-06 19:10:07 +00:00
|
|
|
|
|
|
|
#include "jit.h"
|
|
|
|
|
|
|
|
#include "types.h"
|
|
|
|
#include "stats.h"
|
|
|
|
|
|
|
|
// To do: get cmake to define these..
|
|
|
|
#define ASMJIT_BUILD_EMBED
|
|
|
|
#define ASMJIT_STATIC
|
|
|
|
|
|
|
|
#include <asmjit/asmjit.h>
|
|
|
|
#include <asmjit/x86.h>
|
|
|
|
#include <functional>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
extern cycle_t VMCycles[10];
|
|
|
|
extern int VMCalls[10];
|
|
|
|
|
|
|
|
#define A (pc[0].a)
|
|
|
|
#define B (pc[0].b)
|
|
|
|
#define C (pc[0].c)
|
|
|
|
#define Cs (pc[0].cs)
|
|
|
|
#define BC (pc[0].i16u)
|
|
|
|
#define BCs (pc[0].i16)
|
|
|
|
#define ABCs (pc[0].i24)
|
|
|
|
#define JMPOFS(x) ((x)->i24)
|
|
|
|
|
|
|
|
struct JitLineInfo
|
|
|
|
{
|
|
|
|
ptrdiff_t InstructionIndex = 0;
|
|
|
|
int32_t LineNumber = -1;
|
|
|
|
asmjit::Label Label;
|
|
|
|
};
|
|
|
|
|
|
|
|
class JitCompiler
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
JitCompiler(asmjit::CodeHolder *code, VMScriptFunction *sfunc) : cc(code), sfunc(sfunc) { }
|
|
|
|
|
|
|
|
asmjit::CCFunc *Codegen();
|
|
|
|
VMScriptFunction *GetScriptFunction() { return sfunc; }
|
|
|
|
|
|
|
|
TArray<JitLineInfo> LineInfo;
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Declare EmitXX functions for the opcodes:
|
|
|
|
#define xx(op, name, mode, alt, kreg, ktype) void Emit##op();
|
|
|
|
#include "vmops.h"
|
|
|
|
#undef xx
|
|
|
|
|
|
|
|
asmjit::FuncSignature CreateFuncSignature();
|
|
|
|
|
|
|
|
void Setup();
|
|
|
|
void CreateRegisters();
|
|
|
|
void IncrementVMCalls();
|
|
|
|
void SetupFrame();
|
|
|
|
void SetupSimpleFrame();
|
|
|
|
void SetupFullVMFrame();
|
|
|
|
void BindLabels();
|
|
|
|
void EmitOpcode();
|
|
|
|
void EmitPopFrame();
|
|
|
|
|
|
|
|
void EmitNativeCall(VMNativeFunction *target);
|
|
|
|
void EmitVMCall(asmjit::X86Gp ptr, VMFunction *target);
|
|
|
|
void EmitVtbl(const VMOP *op);
|
|
|
|
|
|
|
|
int StoreCallParams();
|
|
|
|
void LoadInOuts();
|
|
|
|
void LoadReturns(const VMOP *retval, int numret);
|
|
|
|
void FillReturns(const VMOP *retval, int numret);
|
|
|
|
void LoadCallResult(int type, int regnum, bool addrof);
|
|
|
|
|
|
|
|
template <typename Func>
|
|
|
|
void EmitComparisonOpcode(Func jmpFunc)
|
|
|
|
{
|
|
|
|
using namespace asmjit;
|
|
|
|
|
|
|
|
int i = (int)(ptrdiff_t)(pc - sfunc->Code);
|
|
|
|
|
|
|
|
auto successLabel = cc.newLabel();
|
|
|
|
|
|
|
|
auto failLabel = GetLabel(i + 2 + JMPOFS(pc + 1));
|
|
|
|
|
|
|
|
jmpFunc(static_cast<bool>(A & CMP_CHECK), failLabel, successLabel);
|
|
|
|
|
|
|
|
cc.bind(successLabel);
|
|
|
|
pc++; // This instruction uses two instruction slots - skip the next one
|
|
|
|
}
|
|
|
|
|
|
|
|
template<int N>
|
|
|
|
void EmitVectorComparison(bool check, asmjit::Label& fail, asmjit::Label& success)
|
|
|
|
{
|
|
|
|
bool approx = static_cast<bool>(A & CMP_APPROX);
|
|
|
|
if (!approx)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < N; i++)
|
|
|
|
{
|
|
|
|
cc.ucomisd(regF[B + i], regF[C + i]);
|
|
|
|
if (check)
|
|
|
|
{
|
|
|
|
cc.jp(success);
|
|
|
|
if (i == (N - 1))
|
|
|
|
{
|
|
|
|
cc.je(fail);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cc.jne(success);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cc.jp(fail);
|
|
|
|
cc.jne(fail);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto tmp = newTempXmmSd();
|
|
|
|
|
|
|
|
const int64_t absMaskInt = 0x7FFFFFFFFFFFFFFF;
|
|
|
|
auto absMask = cc.newDoubleConst(asmjit::kConstScopeLocal, reinterpret_cast<const double&>(absMaskInt));
|
|
|
|
auto absMaskXmm = newTempXmmPd();
|
|
|
|
|
|
|
|
auto epsilon = cc.newDoubleConst(asmjit::kConstScopeLocal, VM_EPSILON);
|
|
|
|
auto epsilonXmm = newTempXmmSd();
|
|
|
|
|
|
|
|
for (int i = 0; i < N; i++)
|
|
|
|
{
|
|
|
|
cc.movsd(tmp, regF[B + i]);
|
|
|
|
cc.subsd(tmp, regF[C + i]);
|
|
|
|
cc.movsd(absMaskXmm, absMask);
|
|
|
|
cc.andpd(tmp, absMaskXmm);
|
|
|
|
cc.movsd(epsilonXmm, epsilon);
|
|
|
|
cc.ucomisd(epsilonXmm, tmp);
|
|
|
|
|
|
|
|
if (check)
|
|
|
|
{
|
|
|
|
cc.jp(success);
|
|
|
|
if (i == (N - 1))
|
|
|
|
{
|
|
|
|
cc.ja(fail);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cc.jna(success);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
cc.jp(fail);
|
|
|
|
cc.jna(fail);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint64_t ToMemAddress(const void *d)
|
|
|
|
{
|
|
|
|
return (uint64_t)(ptrdiff_t)d;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CallSqrt(const asmjit::X86Xmm &a, const asmjit::X86Xmm &b);
|
|
|
|
|
|
|
|
static void CallAssignString(FString* to, FString* from) {
|
|
|
|
*to = *from;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename RetType, typename P1>
|
|
|
|
asmjit::CCFuncCall *CreateCall(RetType(*func)(P1 p1)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1)>(func))), asmjit::FuncSignature1<RetType, P1>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2>
|
|
|
|
asmjit::CCFuncCall *CreateCall(RetType(*func)(P1 p1, P2 p2)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2)>(func))), asmjit::FuncSignature2<RetType, P1, P2>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3>
|
|
|
|
asmjit::CCFuncCall *CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3)>(func))), asmjit::FuncSignature3<RetType, P1, P2, P3>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3, typename P4>
|
|
|
|
asmjit::CCFuncCall *CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3, P4 p4)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3, P4)>(func))), asmjit::FuncSignature4<RetType, P1, P2, P3, P4>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3, typename P4, typename P5>
|
|
|
|
asmjit::CCFuncCall *CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3, P4, P5)>(func))), asmjit::FuncSignature5<RetType, P1, P2, P3, P4, P5>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3, typename P4, typename P5, typename P6>
|
|
|
|
asmjit::CCFuncCall *CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3, P4, P5, P6)>(func))), asmjit::FuncSignature6<RetType, P1, P2, P3, P4, P5, P6>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3, typename P4, typename P5, typename P6, typename P7>
|
2022-11-24 15:56:46 +00:00
|
|
|
asmjit::CCFuncCall* CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6, P7 p7)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3, P4, P5, P6, P7)>(func))), asmjit::FuncSignature7<RetType, P1, P2, P3, P4, P5, P6, P7>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3, typename P4, typename P5, typename P6, typename P7, typename P8>
|
|
|
|
asmjit::CCFuncCall* CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6, P7 p7, P8 p8)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3, P4, P5, P6, P7, P8)>(func))), asmjit::FuncSignature8<RetType, P1, P2, P3, P4, P5, P6, P7, P8>()); }
|
|
|
|
|
|
|
|
template<typename RetType, typename P1, typename P2, typename P3, typename P4, typename P5, typename P6, typename P7, typename P8, typename P9>
|
|
|
|
asmjit::CCFuncCall* CreateCall(RetType(*func)(P1 p1, P2 p2, P3 p3, P4 p4, P5 p5, P6 p6, P7 p7, P8 p8, P9 p9)) { return cc.call(asmjit::imm_ptr(reinterpret_cast<void*>(static_cast<RetType(*)(P1, P2, P3, P4, P5, P6, P7, P8, P9)>(func))), asmjit::FuncSignature9<RetType, P1, P2, P3, P4, P5, P6, P7, P8, P9>()); }
|
2020-04-06 19:10:07 +00:00
|
|
|
|
|
|
|
FString regname;
|
|
|
|
size_t tmpPosInt32, tmpPosInt64, tmpPosIntPtr, tmpPosXmmSd, tmpPosXmmSs, tmpPosXmmPd, resultPosInt32, resultPosIntPtr, resultPosXmmSd;
|
|
|
|
std::vector<asmjit::X86Gp> regTmpInt32, regTmpInt64, regTmpIntPtr, regResultInt32, regResultIntPtr;
|
|
|
|
std::vector<asmjit::X86Xmm> regTmpXmmSd, regTmpXmmSs, regTmpXmmPd, regResultXmmSd;
|
|
|
|
|
|
|
|
void ResetTemp()
|
|
|
|
{
|
|
|
|
tmpPosInt32 = 0;
|
|
|
|
tmpPosInt64 = 0;
|
|
|
|
tmpPosIntPtr = 0;
|
|
|
|
tmpPosXmmSd = 0;
|
|
|
|
tmpPosXmmSs = 0;
|
|
|
|
tmpPosXmmPd = 0;
|
|
|
|
resultPosInt32 = 0;
|
|
|
|
resultPosIntPtr = 0;
|
|
|
|
resultPosXmmSd = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T, typename NewFunc>
|
|
|
|
T newTempRegister(std::vector<T> &tmpVector, size_t &tmpPos, const char *name, NewFunc newCallback)
|
|
|
|
{
|
|
|
|
if (tmpPos == tmpVector.size())
|
|
|
|
{
|
|
|
|
regname.Format("%s%d", name, (int)tmpVector.size());
|
|
|
|
tmpVector.push_back(newCallback(regname.GetChars()));
|
|
|
|
}
|
|
|
|
return tmpVector[tmpPos++];
|
|
|
|
}
|
|
|
|
|
|
|
|
asmjit::X86Gp newTempInt32() { return newTempRegister(regTmpInt32, tmpPosInt32, "tmpDword", [&](const char *name) { return cc.newInt32(name); }); }
|
|
|
|
asmjit::X86Gp newTempInt64() { return newTempRegister(regTmpInt64, tmpPosInt64, "tmpQword", [&](const char *name) { return cc.newInt64(name); }); }
|
|
|
|
asmjit::X86Gp newTempIntPtr() { return newTempRegister(regTmpIntPtr, tmpPosIntPtr, "tmpPtr", [&](const char *name) { return cc.newIntPtr(name); }); }
|
|
|
|
asmjit::X86Xmm newTempXmmSd() { return newTempRegister(regTmpXmmSd, tmpPosXmmSd, "tmpXmmSd", [&](const char *name) { return cc.newXmmSd(name); }); }
|
|
|
|
asmjit::X86Xmm newTempXmmSs() { return newTempRegister(regTmpXmmSs, tmpPosXmmSs, "tmpXmmSs", [&](const char *name) { return cc.newXmmSs(name); }); }
|
|
|
|
asmjit::X86Xmm newTempXmmPd() { return newTempRegister(regTmpXmmPd, tmpPosXmmPd, "tmpXmmPd", [&](const char *name) { return cc.newXmmPd(name); }); }
|
|
|
|
|
|
|
|
asmjit::X86Gp newResultInt32() { return newTempRegister(regResultInt32, resultPosInt32, "resultDword", [&](const char *name) { return cc.newInt32(name); }); }
|
|
|
|
asmjit::X86Gp newResultIntPtr() { return newTempRegister(regResultIntPtr, resultPosIntPtr, "resultPtr", [&](const char *name) { return cc.newIntPtr(name); }); }
|
|
|
|
asmjit::X86Xmm newResultXmmSd() { return newTempRegister(regResultXmmSd, resultPosXmmSd, "resultXmmSd", [&](const char *name) { return cc.newXmmSd(name); }); }
|
|
|
|
|
|
|
|
void EmitReadBarrier();
|
|
|
|
|
|
|
|
void EmitNullPointerThrow(int index, EVMAbortException reason);
|
|
|
|
void EmitThrowException(EVMAbortException reason);
|
|
|
|
asmjit::Label EmitThrowExceptionLabel(EVMAbortException reason);
|
|
|
|
|
|
|
|
static void ThrowArrayOutOfBounds(int index, int size);
|
|
|
|
static void ThrowException(int reason);
|
|
|
|
|
|
|
|
asmjit::X86Gp CheckRegD(int r0, int r1);
|
|
|
|
asmjit::X86Xmm CheckRegF(int r0, int r1);
|
|
|
|
asmjit::X86Xmm CheckRegF(int r0, int r1, int r2);
|
|
|
|
asmjit::X86Xmm CheckRegF(int r0, int r1, int r2, int r3);
|
2022-11-12 09:45:39 +00:00
|
|
|
asmjit::X86Xmm CheckRegF(int r0, int r1, int r2, int r3, int r4);
|
2020-04-06 19:10:07 +00:00
|
|
|
asmjit::X86Gp CheckRegS(int r0, int r1);
|
|
|
|
asmjit::X86Gp CheckRegA(int r0, int r1);
|
|
|
|
|
|
|
|
asmjit::X86Compiler cc;
|
|
|
|
VMScriptFunction *sfunc;
|
|
|
|
|
|
|
|
asmjit::CCFunc *func = nullptr;
|
|
|
|
asmjit::X86Gp args;
|
|
|
|
asmjit::X86Gp numargs;
|
|
|
|
asmjit::X86Gp ret;
|
|
|
|
asmjit::X86Gp numret;
|
|
|
|
asmjit::X86Gp stack;
|
|
|
|
|
|
|
|
int offsetParams;
|
|
|
|
int offsetF;
|
|
|
|
int offsetS;
|
|
|
|
int offsetA;
|
|
|
|
int offsetD;
|
|
|
|
int offsetExtra;
|
|
|
|
|
|
|
|
TArray<const VMOP *> ParamOpcodes;
|
|
|
|
|
|
|
|
void CheckVMFrame();
|
|
|
|
asmjit::X86Gp GetCallReturns();
|
|
|
|
|
|
|
|
bool vmframeAllocated = false;
|
|
|
|
asmjit::CBNode *vmframeCursor = nullptr;
|
|
|
|
asmjit::X86Gp vmframe;
|
|
|
|
|
|
|
|
bool callReturnsAllocated = false;
|
|
|
|
asmjit::CBNode *callReturnsCursor = nullptr;
|
|
|
|
asmjit::X86Gp callReturns;
|
|
|
|
|
|
|
|
const int *konstd;
|
|
|
|
const double *konstf;
|
|
|
|
const FString *konsts;
|
|
|
|
const FVoidObj *konsta;
|
|
|
|
|
|
|
|
TArray<asmjit::X86Gp> regD;
|
|
|
|
TArray<asmjit::X86Xmm> regF;
|
|
|
|
TArray<asmjit::X86Gp> regA;
|
|
|
|
TArray<asmjit::X86Gp> regS;
|
|
|
|
|
|
|
|
struct OpcodeLabel
|
|
|
|
{
|
|
|
|
asmjit::CBNode *cursor = nullptr;
|
|
|
|
asmjit::Label label;
|
|
|
|
bool inUse = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
asmjit::Label GetLabel(size_t pos)
|
|
|
|
{
|
|
|
|
auto &label = labels[pos];
|
|
|
|
if (!label.inUse)
|
|
|
|
{
|
|
|
|
label.label = cc.newLabel();
|
|
|
|
label.inUse = true;
|
|
|
|
}
|
|
|
|
return label.label;
|
|
|
|
}
|
|
|
|
|
|
|
|
TArray<OpcodeLabel> labels;
|
|
|
|
|
2022-11-24 15:56:46 +00:00
|
|
|
// Get temporary storage enough for DVector4 which is required by operation such as MULQQ and MULQV3
|
|
|
|
bool vectorStackAllocated = false;
|
|
|
|
asmjit::X86Mem vectorStack;
|
|
|
|
asmjit::X86Mem GetTemporaryVectorStackStorage()
|
|
|
|
{
|
|
|
|
if (!vectorStackAllocated)
|
|
|
|
{
|
|
|
|
vectorStack = cc.newStack(sizeof(DVector4), alignof(DVector4), "tmpDVector4");
|
|
|
|
vectorStackAllocated = true;
|
|
|
|
}
|
|
|
|
return vectorStack;
|
|
|
|
}
|
|
|
|
|
2020-04-06 19:10:07 +00:00
|
|
|
const VMOP *pc;
|
|
|
|
VM_UBYTE op;
|
|
|
|
};
|
|
|
|
|
|
|
|
class AsmJitException : public std::exception
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
AsmJitException(asmjit::Error error, const char *message) noexcept : error(error), message(message)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* what() const noexcept override
|
|
|
|
{
|
|
|
|
return message.GetChars();
|
|
|
|
}
|
|
|
|
|
|
|
|
asmjit::Error error;
|
|
|
|
FString message;
|
|
|
|
};
|
|
|
|
|
|
|
|
class ThrowingErrorHandler : public asmjit::ErrorHandler
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
bool handleError(asmjit::Error err, const char *message, asmjit::CodeEmitter *origin) override
|
|
|
|
{
|
|
|
|
throw AsmJitException(err, message);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void *AddJitFunction(asmjit::CodeHolder* code, JitCompiler *compiler);
|
|
|
|
asmjit::CodeInfo GetHostCodeInfo();
|