2022-10-14 20:48:41 +01:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "EmitCommonX64.h"
|
|
|
|
|
|
|
|
#include "Luau/AssemblyBuilderX64.h"
|
2023-03-31 19:42:49 +01:00
|
|
|
#include "Luau/IrCallWrapperX64.h"
|
2023-03-03 20:21:14 +00:00
|
|
|
#include "Luau/IrData.h"
|
2023-03-31 19:42:49 +01:00
|
|
|
#include "Luau/IrRegAllocX64.h"
|
2023-06-24 07:19:39 +01:00
|
|
|
#include "Luau/IrUtils.h"
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
#include "NativeState.h"
|
|
|
|
|
|
|
|
#include "lgc.h"
|
|
|
|
#include "lstate.h"
|
|
|
|
|
2023-09-08 01:13:49 +01:00
|
|
|
#include <utility>
|
|
|
|
|
2022-10-14 20:48:41 +01:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace CodeGen
|
|
|
|
{
|
2023-03-03 20:21:14 +00:00
|
|
|
namespace X64
|
|
|
|
{
|
2022-10-14 20:48:41 +01:00
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
void jumpOnNumberCmp(AssemblyBuilderX64& build, RegisterX64 tmp, OperandX64 lhs, OperandX64 rhs, IrCondition cond, Label& label)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
|
|
|
// Refresher on comi/ucomi EFLAGS:
|
2023-09-08 01:13:49 +01:00
|
|
|
// all zero: greater
|
2022-10-14 20:48:41 +01:00
|
|
|
// CF only: less
|
|
|
|
// ZF only: equal
|
|
|
|
// PF+CF+ZF: unordered (NaN)
|
|
|
|
|
2023-09-08 01:13:49 +01:00
|
|
|
// To avoid the lack of conditional jumps that check for "greater" conditions in IEEE 754 compliant way, we use "less" forms to emulate these
|
|
|
|
if (cond == IrCondition::Greater || cond == IrCondition::GreaterEqual || cond == IrCondition::NotGreater || cond == IrCondition::NotGreaterEqual)
|
|
|
|
std::swap(lhs, rhs);
|
|
|
|
|
2022-10-14 20:48:41 +01:00
|
|
|
if (rhs.cat == CategoryX64::reg)
|
|
|
|
{
|
|
|
|
build.vucomisd(rhs, lhs);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
build.vmovsd(tmp, rhs);
|
|
|
|
build.vucomisd(tmp, lhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep in mind that 'Not' conditions want 'true' for comparisons with NaN
|
|
|
|
// And because of NaN, integer check interchangeability like 'not less or equal' <-> 'greater' does not hold
|
|
|
|
switch (cond)
|
|
|
|
{
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCondition::NotLessEqual:
|
2023-09-08 01:13:49 +01:00
|
|
|
case IrCondition::NotGreaterEqual:
|
2022-10-14 20:48:41 +01:00
|
|
|
// (b < a) is the same as !(a <= b). jnae checks CF=1 which means < or NaN
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::NotAboveEqual, label);
|
2022-10-14 20:48:41 +01:00
|
|
|
break;
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCondition::LessEqual:
|
2023-09-08 01:13:49 +01:00
|
|
|
case IrCondition::GreaterEqual:
|
2022-10-14 20:48:41 +01:00
|
|
|
// (b >= a) is the same as (a <= b). jae checks CF=0 which means >= and not NaN
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::AboveEqual, label);
|
2022-10-14 20:48:41 +01:00
|
|
|
break;
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCondition::NotLess:
|
2023-09-08 01:13:49 +01:00
|
|
|
case IrCondition::NotGreater:
|
2022-10-14 20:48:41 +01:00
|
|
|
// (b <= a) is the same as !(a < b). jna checks CF=1 or ZF=1 which means <= or NaN
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::NotAbove, label);
|
2022-10-14 20:48:41 +01:00
|
|
|
break;
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCondition::Less:
|
2023-09-08 01:13:49 +01:00
|
|
|
case IrCondition::Greater:
|
2022-10-14 20:48:41 +01:00
|
|
|
// (b > a) is the same as (a < b). ja checks CF=0 and ZF=0 which means > and not NaN
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::Above, label);
|
2022-10-14 20:48:41 +01:00
|
|
|
break;
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCondition::NotEqual:
|
2022-10-14 20:48:41 +01:00
|
|
|
// ZF=0 or PF=1 means != or NaN
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::NotZero, label);
|
|
|
|
build.jcc(ConditionX64::Parity, label);
|
2022-10-14 20:48:41 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LUAU_ASSERT(!"Unsupported condition");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-08 01:13:49 +01:00
|
|
|
ConditionX64 getConditionInt(IrCondition cond)
|
|
|
|
{
|
|
|
|
switch (cond)
|
|
|
|
{
|
|
|
|
case IrCondition::Equal:
|
|
|
|
return ConditionX64::Equal;
|
|
|
|
case IrCondition::NotEqual:
|
|
|
|
return ConditionX64::NotEqual;
|
|
|
|
case IrCondition::Less:
|
|
|
|
return ConditionX64::Less;
|
|
|
|
case IrCondition::NotLess:
|
|
|
|
return ConditionX64::NotLess;
|
|
|
|
case IrCondition::LessEqual:
|
|
|
|
return ConditionX64::LessEqual;
|
|
|
|
case IrCondition::NotLessEqual:
|
|
|
|
return ConditionX64::NotLessEqual;
|
|
|
|
case IrCondition::Greater:
|
|
|
|
return ConditionX64::Greater;
|
|
|
|
case IrCondition::NotGreater:
|
|
|
|
return ConditionX64::NotGreater;
|
|
|
|
case IrCondition::GreaterEqual:
|
|
|
|
return ConditionX64::GreaterEqual;
|
|
|
|
case IrCondition::NotGreaterEqual:
|
|
|
|
return ConditionX64::NotGreaterEqual;
|
|
|
|
case IrCondition::UnsignedLess:
|
|
|
|
return ConditionX64::Below;
|
|
|
|
case IrCondition::UnsignedLessEqual:
|
|
|
|
return ConditionX64::BelowEqual;
|
|
|
|
case IrCondition::UnsignedGreater:
|
|
|
|
return ConditionX64::Above;
|
|
|
|
case IrCondition::UnsignedGreaterEqual:
|
|
|
|
return ConditionX64::AboveEqual;
|
|
|
|
default:
|
|
|
|
LUAU_ASSERT(!"Unsupported condition");
|
|
|
|
return ConditionX64::Zero;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-09 19:57:01 +00:00
|
|
|
void getTableNodeAtCachedSlot(AssemblyBuilderX64& build, RegisterX64 tmp, RegisterX64 node, RegisterX64 table, int pcpos)
|
2022-10-21 18:54:01 +01:00
|
|
|
{
|
|
|
|
LUAU_ASSERT(tmp != node);
|
|
|
|
LUAU_ASSERT(table != node);
|
|
|
|
|
|
|
|
build.mov(node, qword[table + offsetof(Table, node)]);
|
|
|
|
|
|
|
|
// compute cached slot
|
|
|
|
build.mov(tmp, sCode);
|
|
|
|
build.movzx(dwordReg(tmp), byte[tmp + pcpos * sizeof(Instruction) + kOffsetOfInstructionC]);
|
|
|
|
build.and_(byteReg(tmp), byte[table + offsetof(Table, nodemask8)]);
|
|
|
|
|
|
|
|
// LuaNode* n = &h->node[slot];
|
|
|
|
build.shl(dwordReg(tmp), kLuaNodeSizeLog2);
|
|
|
|
build.add(node, tmp);
|
|
|
|
}
|
|
|
|
|
2022-12-09 19:57:01 +00:00
|
|
|
void convertNumberToIndexOrJump(AssemblyBuilderX64& build, RegisterX64 tmp, RegisterX64 numd, RegisterX64 numi, Label& label)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
|
|
|
LUAU_ASSERT(numi.size == SizeX64::dword);
|
|
|
|
|
|
|
|
// Convert to integer, NaN is converted into 0x80000000
|
|
|
|
build.vcvttsd2si(numi, numd);
|
|
|
|
|
|
|
|
// Convert that integer back to double
|
|
|
|
build.vcvtsi2sd(tmp, numd, numi);
|
|
|
|
|
|
|
|
build.vucomisd(tmp, numd); // Sets ZF=1 if equal or NaN
|
|
|
|
// We don't need non-integer values
|
|
|
|
// But to skip the PF=1 check, we proceed with NaN because 0x80000000 index is out of bounds
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::NotZero, label);
|
2022-10-14 20:48:41 +01:00
|
|
|
}
|
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
void callArithHelper(IrRegAllocX64& regs, AssemblyBuilderX64& build, int ra, int rb, OperandX64 c, TMS tm)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
2023-03-31 19:42:49 +01:00
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(ra));
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(rb));
|
|
|
|
callWrap.addArgument(SizeX64::qword, c);
|
|
|
|
callWrap.addArgument(SizeX64::dword, tm);
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaV_doarith)]);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
emitUpdateBase(build);
|
|
|
|
}
|
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
void callLengthHelper(IrRegAllocX64& regs, AssemblyBuilderX64& build, int ra, int rb)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
2023-03-31 19:42:49 +01:00
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(ra));
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(rb));
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaV_dolen)]);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
emitUpdateBase(build);
|
|
|
|
}
|
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
void callGetTable(IrRegAllocX64& regs, AssemblyBuilderX64& build, int rb, OperandX64 c, int ra)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
2023-03-31 19:42:49 +01:00
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(rb));
|
|
|
|
callWrap.addArgument(SizeX64::qword, c);
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(ra));
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaV_gettable)]);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
emitUpdateBase(build);
|
|
|
|
}
|
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
void callSetTable(IrRegAllocX64& regs, AssemblyBuilderX64& build, int rb, OperandX64 c, int ra)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
2023-03-31 19:42:49 +01:00
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(rb));
|
|
|
|
callWrap.addArgument(SizeX64::qword, c);
|
|
|
|
callWrap.addArgument(SizeX64::qword, luauRegAddress(ra));
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaV_settable)]);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
emitUpdateBase(build);
|
|
|
|
}
|
|
|
|
|
2023-09-01 18:58:27 +01:00
|
|
|
void checkObjectBarrierConditions(AssemblyBuilderX64& build, RegisterX64 tmp, RegisterX64 object, IrOp ra, int ratag, Label& skip)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
2023-06-24 07:19:39 +01:00
|
|
|
// Barrier should've been optimized away if we know that it's not collectable, checking for correctness
|
|
|
|
if (ratag == -1 || !isGCO(ratag))
|
|
|
|
{
|
|
|
|
// iscollectable(ra)
|
2023-09-01 18:58:27 +01:00
|
|
|
OperandX64 tag = (ra.kind == IrOpKind::VmReg) ? luauRegTag(vmRegOp(ra)) : luauConstantTag(vmConstOp(ra));
|
|
|
|
build.cmp(tag, LUA_TSTRING);
|
2023-06-24 07:19:39 +01:00
|
|
|
build.jcc(ConditionX64::Less, skip);
|
|
|
|
}
|
2022-10-14 20:48:41 +01:00
|
|
|
|
2022-10-21 18:54:01 +01:00
|
|
|
// isblack(obj2gco(o))
|
|
|
|
build.test(byte[object + offsetof(GCheader, marked)], bitmask(BLACKBIT));
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::Zero, skip);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
// iswhite(gcvalue(ra))
|
2023-09-01 18:58:27 +01:00
|
|
|
OperandX64 value = (ra.kind == IrOpKind::VmReg) ? luauRegValue(vmRegOp(ra)) : luauConstantValue(vmConstOp(ra));
|
|
|
|
build.mov(tmp, value);
|
2022-10-14 20:48:41 +01:00
|
|
|
build.test(byte[tmp + offsetof(GCheader, marked)], bit2mask(WHITE0BIT, WHITE1BIT));
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::Zero, skip);
|
2022-10-21 18:54:01 +01:00
|
|
|
}
|
|
|
|
|
2023-09-01 18:58:27 +01:00
|
|
|
|
|
|
|
void callBarrierObject(IrRegAllocX64& regs, AssemblyBuilderX64& build, RegisterX64 object, IrOp objectOp, IrOp ra, int ratag)
|
2022-10-21 18:54:01 +01:00
|
|
|
{
|
2023-04-07 22:01:29 +01:00
|
|
|
Label skip;
|
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
ScopedRegX64 tmp{regs, SizeX64::qword};
|
2023-06-24 07:19:39 +01:00
|
|
|
checkObjectBarrierConditions(build, tmp.reg, object, ra, ratag, skip);
|
2023-03-31 19:42:49 +01:00
|
|
|
|
2023-04-07 22:01:29 +01:00
|
|
|
{
|
|
|
|
ScopedSpills spillGuard(regs);
|
|
|
|
|
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::qword, object, objectOp);
|
|
|
|
callWrap.addArgument(SizeX64::qword, tmp);
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaC_barrierf)]);
|
|
|
|
}
|
|
|
|
|
|
|
|
build.setLabel(skip);
|
2022-10-21 18:54:01 +01:00
|
|
|
}
|
|
|
|
|
2023-04-07 22:01:29 +01:00
|
|
|
void callBarrierTableFast(IrRegAllocX64& regs, AssemblyBuilderX64& build, RegisterX64 table, IrOp tableOp)
|
2022-10-21 18:54:01 +01:00
|
|
|
{
|
2023-04-07 22:01:29 +01:00
|
|
|
Label skip;
|
|
|
|
|
2022-10-21 18:54:01 +01:00
|
|
|
// isblack(obj2gco(t))
|
|
|
|
build.test(byte[table + offsetof(GCheader, marked)], bitmask(BLACKBIT));
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::Zero, skip);
|
2022-10-21 18:54:01 +01:00
|
|
|
|
2023-04-07 22:01:29 +01:00
|
|
|
{
|
|
|
|
ScopedSpills spillGuard(regs);
|
|
|
|
|
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::qword, table, tableOp);
|
|
|
|
callWrap.addArgument(SizeX64::qword, addr[table + offsetof(Table, gclist)]);
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaC_barrierback)]);
|
|
|
|
}
|
|
|
|
|
|
|
|
build.setLabel(skip);
|
2022-10-21 18:54:01 +01:00
|
|
|
}
|
|
|
|
|
2023-04-07 22:01:29 +01:00
|
|
|
void callStepGc(IrRegAllocX64& regs, AssemblyBuilderX64& build)
|
2022-10-21 18:54:01 +01:00
|
|
|
{
|
2023-04-07 22:01:29 +01:00
|
|
|
Label skip;
|
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
{
|
|
|
|
ScopedRegX64 tmp1{regs, SizeX64::qword};
|
|
|
|
ScopedRegX64 tmp2{regs, SizeX64::qword};
|
2022-10-21 18:54:01 +01:00
|
|
|
|
2023-03-31 19:42:49 +01:00
|
|
|
build.mov(tmp1.reg, qword[rState + offsetof(lua_State, global)]);
|
|
|
|
build.mov(tmp2.reg, qword[tmp1.reg + offsetof(global_State, totalbytes)]);
|
|
|
|
build.cmp(tmp2.reg, qword[tmp1.reg + offsetof(global_State, GCthreshold)]);
|
|
|
|
build.jcc(ConditionX64::Below, skip);
|
|
|
|
}
|
2022-10-21 18:54:01 +01:00
|
|
|
|
2023-04-07 22:01:29 +01:00
|
|
|
{
|
|
|
|
ScopedSpills spillGuard(regs);
|
|
|
|
|
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
|
|
|
callWrap.addArgument(SizeX64::dword, 1);
|
|
|
|
callWrap.call(qword[rNativeContext + offsetof(NativeContext, luaC_step)]);
|
|
|
|
emitUpdateBase(build);
|
|
|
|
}
|
|
|
|
|
|
|
|
build.setLabel(skip);
|
2022-10-14 20:48:41 +01:00
|
|
|
}
|
|
|
|
|
2023-07-07 21:10:48 +01:00
|
|
|
void emitClearNativeFlag(AssemblyBuilderX64& build)
|
|
|
|
{
|
|
|
|
build.mov(rax, qword[rState + offsetof(lua_State, ci)]);
|
|
|
|
build.and_(dword[rax + offsetof(CallInfo, flags)], ~LUA_CALLINFO_NATIVE);
|
|
|
|
}
|
|
|
|
|
2022-10-14 20:48:41 +01:00
|
|
|
void emitExit(AssemblyBuilderX64& build, bool continueInVm)
|
|
|
|
{
|
|
|
|
if (continueInVm)
|
2023-05-12 18:50:47 +01:00
|
|
|
build.mov(eax, 1);
|
2022-10-14 20:48:41 +01:00
|
|
|
else
|
|
|
|
build.xor_(eax, eax);
|
|
|
|
|
|
|
|
build.jmp(qword[rNativeContext + offsetof(NativeContext, gateExit)]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void emitUpdateBase(AssemblyBuilderX64& build)
|
|
|
|
{
|
|
|
|
build.mov(rBase, qword[rState + offsetof(lua_State, base)]);
|
|
|
|
}
|
|
|
|
|
2023-06-16 18:35:18 +01:00
|
|
|
void emitInterrupt(AssemblyBuilderX64& build)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
2023-06-16 18:35:18 +01:00
|
|
|
// rax = pcpos + 1
|
|
|
|
// rbx = return address in native code
|
2023-04-14 19:06:22 +01:00
|
|
|
|
2023-06-16 18:35:18 +01:00
|
|
|
// note: rbx is non-volatile so it will be saved across interrupt call automatically
|
|
|
|
|
|
|
|
RegisterX64 rArg1 = (build.abi == ABIX64::Windows) ? rcx : rdi;
|
|
|
|
RegisterX64 rArg2 = (build.abi == ABIX64::Windows) ? rdx : rsi;
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
Label skip;
|
|
|
|
|
2023-06-16 18:35:18 +01:00
|
|
|
// Update L->ci->savedpc; required in case interrupt errors
|
|
|
|
build.mov(rcx, sCode);
|
|
|
|
build.lea(rcx, addr[rcx + rax * sizeof(Instruction)]);
|
|
|
|
build.mov(rax, qword[rState + offsetof(lua_State, ci)]);
|
|
|
|
build.mov(qword[rax + offsetof(CallInfo, savedpc)], rcx);
|
2023-04-14 19:06:22 +01:00
|
|
|
|
2023-06-16 18:35:18 +01:00
|
|
|
// Load interrupt handler; it may be nullptr in case the update raced with the check before we got here
|
|
|
|
build.mov(rax, qword[rState + offsetof(lua_State, global)]);
|
|
|
|
build.mov(rax, qword[rax + offsetof(global_State, cb.interrupt)]);
|
|
|
|
build.test(rax, rax);
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::Zero, skip);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
// Call interrupt
|
2023-06-16 18:35:18 +01:00
|
|
|
build.mov(rArg1, rState);
|
|
|
|
build.mov(dwordReg(rArg2), -1);
|
|
|
|
build.call(rax);
|
2023-03-31 19:42:49 +01:00
|
|
|
|
2022-10-14 20:48:41 +01:00
|
|
|
// Check if we need to exit
|
|
|
|
build.mov(al, byte[rState + offsetof(lua_State, status)]);
|
|
|
|
build.test(al, al);
|
2022-11-04 17:33:22 +00:00
|
|
|
build.jcc(ConditionX64::Zero, skip);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
|
|
|
build.mov(rax, qword[rState + offsetof(lua_State, ci)]);
|
|
|
|
build.sub(qword[rax + offsetof(CallInfo, savedpc)], sizeof(Instruction));
|
|
|
|
emitExit(build, /* continueInVm */ false);
|
|
|
|
|
|
|
|
build.setLabel(skip);
|
2023-06-16 18:35:18 +01:00
|
|
|
|
|
|
|
emitUpdateBase(build); // interrupt may have reallocated stack
|
|
|
|
|
|
|
|
build.jmp(rbx);
|
2022-10-14 20:48:41 +01:00
|
|
|
}
|
|
|
|
|
2023-05-19 20:37:30 +01:00
|
|
|
void emitFallback(IrRegAllocX64& regs, AssemblyBuilderX64& build, int offset, int pcpos)
|
2022-10-14 20:48:41 +01:00
|
|
|
{
|
|
|
|
// fallback(L, instruction, base, k)
|
2023-04-14 19:06:22 +01:00
|
|
|
IrCallWrapperX64 callWrap(regs, build);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rState);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
2023-04-14 19:06:22 +01:00
|
|
|
RegisterX64 reg = callWrap.suggestNextArgumentRegister(SizeX64::qword);
|
|
|
|
build.mov(reg, sCode);
|
|
|
|
callWrap.addArgument(SizeX64::qword, addr[reg + pcpos * sizeof(Instruction)]);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
2023-04-14 19:06:22 +01:00
|
|
|
callWrap.addArgument(SizeX64::qword, rBase);
|
|
|
|
callWrap.addArgument(SizeX64::qword, rConstants);
|
2023-05-19 20:37:30 +01:00
|
|
|
callWrap.call(qword[rNativeContext + offset]);
|
2022-10-14 20:48:41 +01:00
|
|
|
|
2023-04-14 19:06:22 +01:00
|
|
|
emitUpdateBase(build);
|
2022-11-04 17:33:22 +00:00
|
|
|
}
|
2022-10-14 20:48:41 +01:00
|
|
|
|
2023-08-11 15:42:37 +01:00
|
|
|
void emitUpdatePcForExit(AssemblyBuilderX64& build)
|
2023-07-07 21:10:48 +01:00
|
|
|
{
|
|
|
|
// edx = pcpos * sizeof(Instruction)
|
|
|
|
build.add(rdx, sCode);
|
|
|
|
build.mov(rax, qword[rState + offsetof(lua_State, ci)]);
|
|
|
|
build.mov(qword[rax + offsetof(CallInfo, savedpc)], rdx);
|
|
|
|
}
|
|
|
|
|
2023-06-09 18:08:00 +01:00
|
|
|
void emitReturn(AssemblyBuilderX64& build, ModuleHelpers& helpers)
|
|
|
|
{
|
2023-06-16 18:35:18 +01:00
|
|
|
// input: res in rdi, number of written values in ecx
|
2023-06-09 18:08:00 +01:00
|
|
|
RegisterX64 res = rdi;
|
|
|
|
RegisterX64 written = ecx;
|
|
|
|
|
2023-06-16 18:35:18 +01:00
|
|
|
RegisterX64 ci = r8;
|
2023-06-09 18:08:00 +01:00
|
|
|
RegisterX64 cip = r9;
|
|
|
|
RegisterX64 nresults = esi;
|
|
|
|
|
2023-06-16 18:35:18 +01:00
|
|
|
build.mov(ci, qword[rState + offsetof(lua_State, ci)]);
|
2023-06-09 18:08:00 +01:00
|
|
|
build.lea(cip, addr[ci - sizeof(CallInfo)]);
|
|
|
|
|
|
|
|
// nresults = ci->nresults
|
|
|
|
build.mov(nresults, dword[ci + offsetof(CallInfo, nresults)]);
|
|
|
|
|
|
|
|
Label skipResultCopy;
|
|
|
|
|
|
|
|
// Fill the rest of the expected results (nresults - written) with 'nil'
|
|
|
|
RegisterX64 counter = written;
|
|
|
|
build.sub(counter, nresults); // counter = -(nresults - written)
|
|
|
|
build.jcc(ConditionX64::GreaterEqual, skipResultCopy);
|
|
|
|
|
|
|
|
Label repeatNilLoop = build.setLabel();
|
|
|
|
build.mov(dword[res + offsetof(TValue, tt)], LUA_TNIL);
|
|
|
|
build.add(res, sizeof(TValue));
|
|
|
|
build.inc(counter);
|
|
|
|
build.jcc(ConditionX64::NotZero, repeatNilLoop);
|
|
|
|
|
|
|
|
build.setLabel(skipResultCopy);
|
|
|
|
|
|
|
|
build.mov(qword[rState + offsetof(lua_State, ci)], cip); // L->ci = cip
|
|
|
|
build.mov(rBase, qword[cip + offsetof(CallInfo, base)]); // sync base = L->base while we have a chance
|
|
|
|
build.mov(qword[rState + offsetof(lua_State, base)], rBase); // L->base = cip->base
|
|
|
|
|
|
|
|
Label skipFixedRetTop;
|
|
|
|
build.test(nresults, nresults); // test here will set SF=1 for a negative number and it always sets OF to 0
|
|
|
|
build.jcc(ConditionX64::Less, skipFixedRetTop); // jl jumps if SF != OF
|
|
|
|
build.mov(res, qword[cip + offsetof(CallInfo, top)]); // res = cip->top
|
|
|
|
build.setLabel(skipFixedRetTop);
|
|
|
|
|
|
|
|
build.mov(qword[rState + offsetof(lua_State, top)], res); // L->top = res
|
|
|
|
|
|
|
|
// Unlikely, but this might be the last return from VM
|
|
|
|
build.test(byte[ci + offsetof(CallInfo, flags)], LUA_CALLINFO_RETURN);
|
|
|
|
build.jcc(ConditionX64::NotZero, helpers.exitNoContinueVm);
|
|
|
|
|
|
|
|
// Returning back to the previous function is a bit tricky
|
|
|
|
// Registers alive: r9 (cip)
|
|
|
|
RegisterX64 proto = rcx;
|
|
|
|
RegisterX64 execdata = rbx;
|
|
|
|
|
|
|
|
// Change closure
|
|
|
|
build.mov(rax, qword[cip + offsetof(CallInfo, func)]);
|
|
|
|
build.mov(rax, qword[rax + offsetof(TValue, value.gc)]);
|
|
|
|
build.mov(sClosure, rax);
|
|
|
|
|
|
|
|
build.mov(proto, qword[rax + offsetof(Closure, l.p)]);
|
|
|
|
|
|
|
|
build.mov(execdata, qword[proto + offsetof(Proto, execdata)]);
|
|
|
|
|
|
|
|
build.test(byte[cip + offsetof(CallInfo, flags)], LUA_CALLINFO_NATIVE);
|
|
|
|
build.jcc(ConditionX64::Zero, helpers.exitContinueVm); // Continue in interpreter if function has no native data
|
|
|
|
|
|
|
|
// Change constants
|
|
|
|
build.mov(rConstants, qword[proto + offsetof(Proto, k)]);
|
|
|
|
|
|
|
|
// Change code
|
|
|
|
build.mov(rdx, qword[proto + offsetof(Proto, code)]);
|
|
|
|
build.mov(sCode, rdx);
|
|
|
|
|
|
|
|
build.mov(rax, qword[cip + offsetof(CallInfo, savedpc)]);
|
|
|
|
|
|
|
|
// To get instruction index from instruction pointer, we need to divide byte offset by 4
|
|
|
|
// But we will actually need to scale instruction index by 4 back to byte offset later so it cancels out
|
|
|
|
build.sub(rax, rdx);
|
|
|
|
|
|
|
|
// Get new instruction location and jump to it
|
|
|
|
build.mov(edx, dword[execdata + rax]);
|
|
|
|
build.add(rdx, qword[proto + offsetof(Proto, exectarget)]);
|
|
|
|
build.jmp(rdx);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
} // namespace X64
|
2022-10-14 20:48:41 +01:00
|
|
|
} // namespace CodeGen
|
|
|
|
} // namespace Luau
|