luau/CodeGen/src/EmitCommonX64.h

225 lines
8.1 KiB
C
Raw Normal View History

2022-10-13 23:59:53 +01:00
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#pragma once
#include "Luau/AssemblyBuilderX64.h"
2023-03-03 13:45:38 +00:00
#include "EmitCommon.h"
2022-10-13 23:59:53 +01:00
#include "lobject.h"
#include "ltm.h"
// MS x64 ABI reminder:
// Arguments: rcx, rdx, r8, r9 ('overlapped' with xmm0-xmm3)
// Return: rax, xmm0
// Nonvolatile: r12-r15, rdi, rsi, rbx, rbp
// SIMD: only xmm6-xmm15 are non-volatile, all ymm upper parts are volatile
// AMD64 ABI reminder:
// Arguments: rdi, rsi, rdx, rcx, r8, r9 (xmm0-xmm7)
// Return: rax, rdx, xmm0, xmm1
// Nonvolatile: r12-r15, rbx, rbp
// SIMD: all volatile
namespace Luau
{
namespace CodeGen
{
2023-03-03 13:45:38 +00:00
enum class IrCondition : uint8_t;
2022-10-13 23:59:53 +01:00
struct NativeState;
2023-03-31 13:21:14 +01:00
struct IrOp;
2022-10-13 23:59:53 +01:00
2023-03-03 13:45:38 +00:00
namespace X64
{
2023-03-31 13:21:14 +01:00
struct IrRegAllocX64;
2023-04-14 13:05:27 +01:00
constexpr uint32_t kFunctionAlignment = 32;
2022-10-13 23:59:53 +01:00
// Data that is very common to access is placed in non-volatile registers
constexpr RegisterX64 rState = r15; // lua_State* L
constexpr RegisterX64 rBase = r14; // StkId base
constexpr RegisterX64 rNativeContext = r13; // NativeContext* context
constexpr RegisterX64 rConstants = r12; // TValue* k
2023-08-25 16:25:09 +01:00
constexpr unsigned kExtraLocals = 3; // Number of 8 byte slots available for specialized local variables specified below
2023-10-06 18:31:16 +01:00
constexpr unsigned kSpillSlots = 13; // Number of 8 byte slots available for register allocator to spill data into
2023-08-25 16:25:09 +01:00
static_assert((kExtraLocals + kSpillSlots) * 8 % 16 == 0, "locals have to preserve 16 byte alignment");
constexpr uint8_t kWindowsFirstNonVolXmmReg = 6;
constexpr uint8_t kWindowsUsableXmmRegs = 10; // Some xmm regs are non-volatile, we have to balance how many we want to use/preserve
constexpr uint8_t kSystemVUsableXmmRegs = 16; // All xmm regs are volatile
inline uint8_t getXmmRegisterCount(ABIX64 abi)
{
return abi == ABIX64::SystemV ? kSystemVUsableXmmRegs : kWindowsUsableXmmRegs;
}
2022-10-13 23:59:53 +01:00
// Native code is as stackless as the interpreter, so we can place some data on the stack once and have it accessible at any point
2023-08-25 16:25:09 +01:00
// Stack is separated into sections for different data. See CodeGenX64.cpp for layout overview
constexpr unsigned kStackAlign = 8; // Bytes we need to align the stack for non-vol xmm register storage
2023-09-01 17:38:53 +01:00
constexpr unsigned kStackLocalStorage = 8 * kExtraLocals;
constexpr unsigned kStackSpillStorage = 8 * kSpillSlots;
2023-08-25 16:25:09 +01:00
constexpr unsigned kStackExtraArgumentStorage = 2 * 8; // Bytes for 5th and 6th function call arguments used under Windows ABI
constexpr unsigned kStackRegHomeStorage = 4 * 8; // Register 'home' locations that can be used by callees under Windows ABI
inline unsigned getNonVolXmmStorageSize(ABIX64 abi, uint8_t xmmRegCount)
{
if (abi == ABIX64::SystemV)
return 0;
// First 6 are volatile
if (xmmRegCount <= kWindowsFirstNonVolXmmReg)
return 0;
LUAU_ASSERT(xmmRegCount <= 16);
return (xmmRegCount - kWindowsFirstNonVolXmmReg) * 16;
}
// Useful offsets to specific parts
constexpr unsigned kStackOffsetToLocals = kStackExtraArgumentStorage + kStackRegHomeStorage;
constexpr unsigned kStackOffsetToSpillSlots = kStackOffsetToLocals + kStackLocalStorage;
inline unsigned getFullStackSize(ABIX64 abi, uint8_t xmmRegCount)
{
2023-09-01 17:38:53 +01:00
return kStackOffsetToSpillSlots + kStackSpillStorage + getNonVolXmmStorageSize(abi, xmmRegCount) + kStackAlign;
2023-08-25 16:25:09 +01:00
}
constexpr OperandX64 sClosure = qword[rsp + kStackOffsetToLocals + 0]; // Closure* cl
constexpr OperandX64 sCode = qword[rsp + kStackOffsetToLocals + 8]; // Instruction* code
constexpr OperandX64 sTemporarySlot = addr[rsp + kStackOffsetToLocals + 16];
2022-11-10 22:04:44 +00:00
2023-08-25 16:25:09 +01:00
constexpr OperandX64 sSpillArea = addr[rsp + kStackOffsetToSpillSlots];
2022-10-13 23:59:53 +01:00
inline OperandX64 luauReg(int ri)
{
return xmmword[rBase + ri * sizeof(TValue)];
}
2022-11-04 17:02:37 +00:00
inline OperandX64 luauRegAddress(int ri)
{
return addr[rBase + ri * sizeof(TValue)];
}
2022-10-13 23:59:53 +01:00
inline OperandX64 luauRegValue(int ri)
{
return qword[rBase + ri * sizeof(TValue) + offsetof(TValue, value)];
}
inline OperandX64 luauRegTag(int ri)
{
return dword[rBase + ri * sizeof(TValue) + offsetof(TValue, tt)];
}
2022-12-09 18:07:25 +00:00
inline OperandX64 luauRegValueInt(int ri)
2022-10-13 23:59:53 +01:00
{
return dword[rBase + ri * sizeof(TValue) + offsetof(TValue, value)];
}
2023-04-07 20:56:27 +01:00
inline OperandX64 luauRegValueVector(int ri, int index)
{
return dword[rBase + ri * sizeof(TValue) + offsetof(TValue, value) + (sizeof(float) * index)];
}
2022-10-13 23:59:53 +01:00
inline OperandX64 luauConstant(int ki)
{
return xmmword[rConstants + ki * sizeof(TValue)];
}
2022-11-04 17:02:37 +00:00
inline OperandX64 luauConstantAddress(int ki)
{
return addr[rConstants + ki * sizeof(TValue)];
}
2022-10-21 18:33:43 +01:00
inline OperandX64 luauConstantTag(int ki)
{
return dword[rConstants + ki * sizeof(TValue) + offsetof(TValue, tt)];
}
2022-10-13 23:59:53 +01:00
inline OperandX64 luauConstantValue(int ki)
{
return qword[rConstants + ki * sizeof(TValue) + offsetof(TValue, value)];
}
2022-10-21 18:33:43 +01:00
inline OperandX64 luauNodeKeyValue(RegisterX64 node)
{
return qword[node + offsetof(LuaNode, key) + offsetof(TKey, value)];
}
2022-10-27 23:22:49 +01:00
// Note: tag has dirty upper bits
inline OperandX64 luauNodeKeyTag(RegisterX64 node)
{
2023-05-19 19:59:59 +01:00
return dword[node + offsetof(LuaNode, key) + kOffsetOfTKeyTagNext];
2022-10-27 23:22:49 +01:00
}
2022-10-13 23:59:53 +01:00
inline void setLuauReg(AssemblyBuilderX64& build, RegisterX64 tmp, int ri, OperandX64 op)
{
LUAU_ASSERT(op.cat == CategoryX64::mem);
build.vmovups(tmp, op);
build.vmovups(luauReg(ri), tmp);
}
inline void jumpIfTagIs(AssemblyBuilderX64& build, int ri, lua_Type tag, Label& label)
{
build.cmp(luauRegTag(ri), tag);
2022-11-04 17:02:37 +00:00
build.jcc(ConditionX64::Equal, label);
2022-10-13 23:59:53 +01:00
}
inline void jumpIfTagIsNot(AssemblyBuilderX64& build, int ri, lua_Type tag, Label& label)
{
build.cmp(luauRegTag(ri), tag);
2022-11-04 17:02:37 +00:00
build.jcc(ConditionX64::NotEqual, label);
2022-10-13 23:59:53 +01:00
}
// Note: fallthrough label should be placed after this condition
inline void jumpIfFalsy(AssemblyBuilderX64& build, int ri, Label& target, Label& fallthrough)
{
jumpIfTagIs(build, ri, LUA_TNIL, target); // false if nil
jumpIfTagIsNot(build, ri, LUA_TBOOLEAN, fallthrough); // true if not nil or boolean
2022-12-09 18:07:25 +00:00
build.cmp(luauRegValueInt(ri), 0);
2022-11-04 17:02:37 +00:00
build.jcc(ConditionX64::Equal, target); // true if boolean value is 'true'
2022-10-13 23:59:53 +01:00
}
// Note: fallthrough label should be placed after this condition
inline void jumpIfTruthy(AssemblyBuilderX64& build, int ri, Label& target, Label& fallthrough)
{
jumpIfTagIs(build, ri, LUA_TNIL, fallthrough); // false if nil
jumpIfTagIsNot(build, ri, LUA_TBOOLEAN, target); // true if not nil or boolean
2022-12-09 18:07:25 +00:00
build.cmp(luauRegValueInt(ri), 0);
2022-11-04 17:02:37 +00:00
build.jcc(ConditionX64::NotEqual, target); // true if boolean value is 'true'
2022-10-13 23:59:53 +01:00
}
2023-03-03 13:45:38 +00:00
void jumpOnNumberCmp(AssemblyBuilderX64& build, RegisterX64 tmp, OperandX64 lhs, OperandX64 rhs, IrCondition cond, Label& label);
2022-10-13 23:59:53 +01:00
2023-09-08 00:24:03 +01:00
ConditionX64 getConditionInt(IrCondition cond);
2022-12-09 18:07:25 +00:00
void getTableNodeAtCachedSlot(AssemblyBuilderX64& build, RegisterX64 tmp, RegisterX64 node, RegisterX64 table, int pcpos);
void convertNumberToIndexOrJump(AssemblyBuilderX64& build, RegisterX64 tmp, RegisterX64 numd, RegisterX64 numi, Label& label);
2022-10-13 23:59:53 +01:00
void callArithHelper(IrRegAllocX64& regs, AssemblyBuilderX64& build, int ra, OperandX64 b, OperandX64 c, TMS tm);
2023-03-31 13:21:14 +01:00
void callLengthHelper(IrRegAllocX64& regs, AssemblyBuilderX64& build, int ra, int rb);
void callGetTable(IrRegAllocX64& regs, AssemblyBuilderX64& build, int rb, OperandX64 c, int ra);
void callSetTable(IrRegAllocX64& regs, AssemblyBuilderX64& build, int rb, OperandX64 c, int ra);
2023-09-01 17:38:53 +01:00
void checkObjectBarrierConditions(AssemblyBuilderX64& build, RegisterX64 tmp, RegisterX64 object, IrOp ra, int ratag, Label& skip);
void callBarrierObject(IrRegAllocX64& regs, AssemblyBuilderX64& build, RegisterX64 object, IrOp objectOp, IrOp ra, int ratag);
2023-04-07 20:56:27 +01:00
void callBarrierTableFast(IrRegAllocX64& regs, AssemblyBuilderX64& build, RegisterX64 table, IrOp tableOp);
void callStepGc(IrRegAllocX64& regs, AssemblyBuilderX64& build);
2022-10-13 23:59:53 +01:00
2023-07-07 18:14:35 +01:00
void emitClearNativeFlag(AssemblyBuilderX64& build);
2022-10-13 23:59:53 +01:00
void emitExit(AssemblyBuilderX64& build, bool continueInVm);
void emitUpdateBase(AssemblyBuilderX64& build);
2023-06-16 18:01:18 +01:00
void emitInterrupt(AssemblyBuilderX64& build);
2023-05-19 19:59:59 +01:00
void emitFallback(IrRegAllocX64& regs, AssemblyBuilderX64& build, int offset, int pcpos);
2022-10-13 23:59:53 +01:00
2023-08-11 13:55:30 +01:00
void emitUpdatePcForExit(AssemblyBuilderX64& build);
2022-11-04 17:02:37 +00:00
2023-06-09 13:20:36 +01:00
void emitReturn(AssemblyBuilderX64& build, ModuleHelpers& helpers);
2023-03-03 13:45:38 +00:00
} // namespace X64
2022-10-13 23:59:53 +01:00
} // namespace CodeGen
} // namespace Luau