2023-02-24 21:49:38 +00:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/OptimizeConstProp.h"
|
|
|
|
|
|
|
|
#include "Luau/DenseHash.h"
|
2023-03-03 20:21:14 +00:00
|
|
|
#include "Luau/IrAnalysis.h"
|
2023-02-24 21:49:38 +00:00
|
|
|
#include "Luau/IrBuilder.h"
|
|
|
|
#include "Luau/IrUtils.h"
|
|
|
|
|
|
|
|
#include "lua.h"
|
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
LUAU_FASTINTVARIABLE(LuauCodeGenMinLinearBlockPath, 3)
|
|
|
|
|
2023-02-24 21:49:38 +00:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace CodeGen
|
|
|
|
{
|
|
|
|
|
|
|
|
// Data we know about the register value
|
|
|
|
struct RegisterInfo
|
|
|
|
{
|
|
|
|
uint8_t tag = 0xff;
|
|
|
|
IrOp value;
|
|
|
|
|
|
|
|
// Used to quickly invalidate links between SSA values and register memory
|
|
|
|
// It's a bit imprecise where value and tag both always invalidate together
|
|
|
|
uint32_t version = 0;
|
|
|
|
|
|
|
|
bool knownNotReadonly = false;
|
|
|
|
bool knownNoMetatable = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Load instructions are linked to target register to carry knowledge about the target
|
|
|
|
// We track a register version at the point of the load so it's easy to break the link when register is updated
|
|
|
|
struct RegisterLink
|
|
|
|
{
|
|
|
|
uint8_t reg = 0;
|
|
|
|
uint32_t version = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Data we know about the current VM state
|
|
|
|
struct ConstPropState
|
|
|
|
{
|
|
|
|
uint8_t tryGetTag(IrOp op)
|
|
|
|
{
|
|
|
|
if (RegisterInfo* info = tryGetRegisterInfo(op))
|
|
|
|
return info->tag;
|
|
|
|
|
|
|
|
return 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
void saveTag(IrOp op, uint8_t tag)
|
|
|
|
{
|
|
|
|
if (RegisterInfo* info = tryGetRegisterInfo(op))
|
|
|
|
info->tag = tag;
|
|
|
|
}
|
|
|
|
|
|
|
|
IrOp tryGetValue(IrOp op)
|
|
|
|
{
|
|
|
|
if (RegisterInfo* info = tryGetRegisterInfo(op))
|
|
|
|
return info->value;
|
|
|
|
|
|
|
|
return IrOp{IrOpKind::None, 0u};
|
|
|
|
}
|
|
|
|
|
|
|
|
void saveValue(IrOp op, IrOp value)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(value.kind == IrOpKind::Constant);
|
|
|
|
|
|
|
|
if (RegisterInfo* info = tryGetRegisterInfo(op))
|
|
|
|
info->value = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidate(RegisterInfo& reg, bool invalidateTag, bool invalidateValue)
|
|
|
|
{
|
|
|
|
if (invalidateTag)
|
|
|
|
{
|
|
|
|
reg.tag = 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (invalidateValue)
|
|
|
|
{
|
|
|
|
reg.value = {};
|
|
|
|
reg.knownNotReadonly = false;
|
|
|
|
reg.knownNoMetatable = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
reg.version++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidateTag(IrOp regOp)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(regOp.kind == IrOpKind::VmReg);
|
|
|
|
invalidate(regs[regOp.index], /* invalidateTag */ true, /* invalidateValue */ false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidateValue(IrOp regOp)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(regOp.kind == IrOpKind::VmReg);
|
|
|
|
invalidate(regs[regOp.index], /* invalidateTag */ false, /* invalidateValue */ true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidate(IrOp regOp)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(regOp.kind == IrOpKind::VmReg);
|
|
|
|
invalidate(regs[regOp.index], /* invalidateTag */ true, /* invalidateValue */ true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidateRegistersFrom(uint32_t firstReg)
|
|
|
|
{
|
|
|
|
for (int i = int(firstReg); i <= maxReg; ++i)
|
|
|
|
invalidate(regs[i], /* invalidateTag */ true, /* invalidateValue */ true);
|
|
|
|
|
|
|
|
maxReg = int(firstReg) - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidateHeap()
|
|
|
|
{
|
|
|
|
for (int i = 0; i <= maxReg; ++i)
|
|
|
|
invalidateHeap(regs[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidateHeap(RegisterInfo& reg)
|
|
|
|
{
|
|
|
|
reg.knownNotReadonly = false;
|
|
|
|
reg.knownNoMetatable = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void invalidateAll()
|
|
|
|
{
|
|
|
|
// Invalidating registers also invalidates what we know about the heap (stored in RegisterInfo)
|
|
|
|
invalidateRegistersFrom(0u);
|
|
|
|
inSafeEnv = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void createRegLink(uint32_t instIdx, IrOp regOp)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(regOp.kind == IrOpKind::VmReg);
|
|
|
|
LUAU_ASSERT(!instLink.contains(instIdx));
|
|
|
|
instLink[instIdx] = RegisterLink{uint8_t(regOp.index), regs[regOp.index].version};
|
|
|
|
}
|
|
|
|
|
|
|
|
RegisterInfo* tryGetRegisterInfo(IrOp op)
|
|
|
|
{
|
|
|
|
if (op.kind == IrOpKind::VmReg)
|
|
|
|
{
|
|
|
|
maxReg = int(op.index) > maxReg ? int(op.index) : maxReg;
|
|
|
|
return ®s[op.index];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (RegisterLink* link = tryGetRegLink(op))
|
|
|
|
{
|
|
|
|
maxReg = int(link->reg) > maxReg ? int(link->reg) : maxReg;
|
|
|
|
return ®s[link->reg];
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
RegisterLink* tryGetRegLink(IrOp instOp)
|
|
|
|
{
|
|
|
|
if (instOp.kind != IrOpKind::Inst)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
if (RegisterLink* link = instLink.find(instOp.index))
|
|
|
|
{
|
|
|
|
// Check that the target register hasn't changed the value
|
|
|
|
if (link->version > regs[link->reg].version)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
RegisterInfo regs[256];
|
|
|
|
|
|
|
|
// For range/full invalidations, we only want to visit a limited number of data that we have recorded
|
|
|
|
int maxReg = 0;
|
|
|
|
|
|
|
|
bool inSafeEnv = false;
|
|
|
|
bool checkedGc = false;
|
|
|
|
|
|
|
|
DenseHashMap<uint32_t, RegisterLink> instLink{~0u};
|
|
|
|
};
|
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
static void handleBuiltinEffects(ConstPropState& state, LuauBuiltinFunction bfid, uint32_t firstReturnReg, int nresults)
|
|
|
|
{
|
|
|
|
// Switch over all values is used to force new items to be handled
|
|
|
|
switch (bfid)
|
|
|
|
{
|
|
|
|
case LBF_NONE:
|
|
|
|
case LBF_ASSERT:
|
|
|
|
case LBF_MATH_ABS:
|
|
|
|
case LBF_MATH_ACOS:
|
|
|
|
case LBF_MATH_ASIN:
|
|
|
|
case LBF_MATH_ATAN2:
|
|
|
|
case LBF_MATH_ATAN:
|
|
|
|
case LBF_MATH_CEIL:
|
|
|
|
case LBF_MATH_COSH:
|
|
|
|
case LBF_MATH_COS:
|
|
|
|
case LBF_MATH_DEG:
|
|
|
|
case LBF_MATH_EXP:
|
|
|
|
case LBF_MATH_FLOOR:
|
|
|
|
case LBF_MATH_FMOD:
|
|
|
|
case LBF_MATH_FREXP:
|
|
|
|
case LBF_MATH_LDEXP:
|
|
|
|
case LBF_MATH_LOG10:
|
|
|
|
case LBF_MATH_LOG:
|
|
|
|
case LBF_MATH_MAX:
|
|
|
|
case LBF_MATH_MIN:
|
|
|
|
case LBF_MATH_MODF:
|
|
|
|
case LBF_MATH_POW:
|
|
|
|
case LBF_MATH_RAD:
|
|
|
|
case LBF_MATH_SINH:
|
|
|
|
case LBF_MATH_SIN:
|
|
|
|
case LBF_MATH_SQRT:
|
|
|
|
case LBF_MATH_TANH:
|
|
|
|
case LBF_MATH_TAN:
|
|
|
|
case LBF_BIT32_ARSHIFT:
|
|
|
|
case LBF_BIT32_BAND:
|
|
|
|
case LBF_BIT32_BNOT:
|
|
|
|
case LBF_BIT32_BOR:
|
|
|
|
case LBF_BIT32_BXOR:
|
|
|
|
case LBF_BIT32_BTEST:
|
|
|
|
case LBF_BIT32_EXTRACT:
|
|
|
|
case LBF_BIT32_LROTATE:
|
|
|
|
case LBF_BIT32_LSHIFT:
|
|
|
|
case LBF_BIT32_REPLACE:
|
|
|
|
case LBF_BIT32_RROTATE:
|
|
|
|
case LBF_BIT32_RSHIFT:
|
|
|
|
case LBF_TYPE:
|
|
|
|
case LBF_STRING_BYTE:
|
|
|
|
case LBF_STRING_CHAR:
|
|
|
|
case LBF_STRING_LEN:
|
|
|
|
case LBF_TYPEOF:
|
|
|
|
case LBF_STRING_SUB:
|
|
|
|
case LBF_MATH_CLAMP:
|
|
|
|
case LBF_MATH_SIGN:
|
|
|
|
case LBF_MATH_ROUND:
|
|
|
|
case LBF_RAWSET:
|
|
|
|
case LBF_RAWGET:
|
|
|
|
case LBF_RAWEQUAL:
|
|
|
|
case LBF_TABLE_INSERT:
|
|
|
|
case LBF_TABLE_UNPACK:
|
|
|
|
case LBF_VECTOR:
|
|
|
|
case LBF_BIT32_COUNTLZ:
|
|
|
|
case LBF_BIT32_COUNTRZ:
|
|
|
|
case LBF_SELECT_VARARG:
|
|
|
|
case LBF_RAWLEN:
|
|
|
|
case LBF_BIT32_EXTRACTK:
|
|
|
|
case LBF_GETMETATABLE:
|
|
|
|
break;
|
|
|
|
case LBF_SETMETATABLE:
|
|
|
|
state.invalidateHeap(); // TODO: only knownNoMetatable is affected and we might know which one
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: classify further using switch above, some fastcalls only modify the value, not the tag
|
|
|
|
state.invalidateRegistersFrom(firstReturnReg);
|
|
|
|
}
|
|
|
|
|
2023-02-24 21:49:38 +00:00
|
|
|
static void constPropInInst(ConstPropState& state, IrBuilder& build, IrFunction& function, IrBlock& block, IrInst& inst, uint32_t index)
|
|
|
|
{
|
|
|
|
switch (inst.cmd)
|
|
|
|
{
|
|
|
|
case IrCmd::LOAD_TAG:
|
|
|
|
if (uint8_t tag = state.tryGetTag(inst.a); tag != 0xff)
|
|
|
|
substitute(function, inst, build.constTag(tag));
|
|
|
|
else if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
state.createRegLink(index, inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::LOAD_POINTER:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
state.createRegLink(index, inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::LOAD_DOUBLE:
|
|
|
|
if (IrOp value = state.tryGetValue(inst.a); value.kind == IrOpKind::Constant)
|
|
|
|
substitute(function, inst, value);
|
|
|
|
else if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
state.createRegLink(index, inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::LOAD_INT:
|
|
|
|
if (IrOp value = state.tryGetValue(inst.a); value.kind == IrOpKind::Constant)
|
|
|
|
substitute(function, inst, value);
|
|
|
|
else if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
state.createRegLink(index, inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::LOAD_TVALUE:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
state.createRegLink(index, inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::STORE_TAG:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
{
|
|
|
|
if (inst.b.kind == IrOpKind::Constant)
|
|
|
|
{
|
|
|
|
uint8_t value = function.tagOp(inst.b);
|
|
|
|
|
|
|
|
if (state.tryGetTag(inst.a) == value)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
state.saveTag(inst.a, value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.invalidateTag(inst.a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::STORE_POINTER:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
state.invalidateValue(inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::STORE_DOUBLE:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
{
|
|
|
|
if (inst.b.kind == IrOpKind::Constant)
|
|
|
|
{
|
|
|
|
std::optional<double> oldValue = function.asDoubleOp(state.tryGetValue(inst.a));
|
|
|
|
double newValue = function.doubleOp(inst.b);
|
|
|
|
|
|
|
|
if (oldValue && *oldValue == newValue)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
state.saveValue(inst.a, inst.b);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.invalidateValue(inst.a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::STORE_INT:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
{
|
|
|
|
if (inst.b.kind == IrOpKind::Constant)
|
|
|
|
{
|
|
|
|
std::optional<int> oldValue = function.asIntOp(state.tryGetValue(inst.a));
|
|
|
|
int newValue = function.intOp(inst.b);
|
|
|
|
|
|
|
|
if (oldValue && *oldValue == newValue)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
state.saveValue(inst.a, inst.b);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.invalidateValue(inst.a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::STORE_TVALUE:
|
|
|
|
if (inst.a.kind == IrOpKind::VmReg)
|
|
|
|
{
|
|
|
|
state.invalidate(inst.a);
|
|
|
|
|
|
|
|
if (uint8_t tag = state.tryGetTag(inst.b); tag != 0xff)
|
|
|
|
state.saveTag(inst.a, tag);
|
|
|
|
|
|
|
|
if (IrOp value = state.tryGetValue(inst.b); value.kind != IrOpKind::None)
|
|
|
|
state.saveValue(inst.a, value);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::JUMP_IF_TRUTHY:
|
|
|
|
if (uint8_t tag = state.tryGetTag(inst.a); tag != 0xff)
|
|
|
|
{
|
|
|
|
if (tag == LUA_TNIL)
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.c});
|
|
|
|
else if (tag != LUA_TBOOLEAN)
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.b});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::JUMP_IF_FALSY:
|
|
|
|
if (uint8_t tag = state.tryGetTag(inst.a); tag != 0xff)
|
|
|
|
{
|
|
|
|
if (tag == LUA_TNIL)
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.b});
|
|
|
|
else if (tag != LUA_TBOOLEAN)
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.c});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::JUMP_EQ_TAG:
|
|
|
|
{
|
|
|
|
uint8_t tagA = inst.a.kind == IrOpKind::Constant ? function.tagOp(inst.a) : state.tryGetTag(inst.a);
|
|
|
|
uint8_t tagB = inst.b.kind == IrOpKind::Constant ? function.tagOp(inst.b) : state.tryGetTag(inst.b);
|
|
|
|
|
|
|
|
if (tagA != 0xff && tagB != 0xff)
|
|
|
|
{
|
|
|
|
if (tagA == tagB)
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.c});
|
|
|
|
else
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.d});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case IrCmd::JUMP_EQ_INT:
|
|
|
|
{
|
|
|
|
std::optional<int> valueA = function.asIntOp(inst.a.kind == IrOpKind::Constant ? inst.a : state.tryGetValue(inst.a));
|
|
|
|
std::optional<int> valueB = function.asIntOp(inst.b.kind == IrOpKind::Constant ? inst.b : state.tryGetValue(inst.b));
|
|
|
|
|
|
|
|
if (valueA && valueB)
|
|
|
|
{
|
|
|
|
if (*valueA == *valueB)
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.c});
|
|
|
|
else
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.d});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case IrCmd::JUMP_CMP_NUM:
|
|
|
|
{
|
|
|
|
std::optional<double> valueA = function.asDoubleOp(inst.a.kind == IrOpKind::Constant ? inst.a : state.tryGetValue(inst.a));
|
|
|
|
std::optional<double> valueB = function.asDoubleOp(inst.b.kind == IrOpKind::Constant ? inst.b : state.tryGetValue(inst.b));
|
|
|
|
|
|
|
|
if (valueA && valueB)
|
|
|
|
{
|
|
|
|
if (compare(*valueA, *valueB, function.conditionOp(inst.c)))
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.d});
|
|
|
|
else
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.e});
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case IrCmd::GET_UPVALUE:
|
|
|
|
state.invalidate(inst.a);
|
|
|
|
break;
|
|
|
|
case IrCmd::CHECK_TAG:
|
|
|
|
{
|
|
|
|
uint8_t b = function.tagOp(inst.b);
|
|
|
|
|
|
|
|
if (uint8_t tag = state.tryGetTag(inst.a); tag != 0xff)
|
|
|
|
{
|
|
|
|
if (tag == b)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
replace(function, block, index, {IrCmd::JUMP, inst.c}); // Shows a conflict in assumptions on this path
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.saveTag(inst.a, b); // We can assume the tag value going forward
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case IrCmd::CHECK_READONLY:
|
|
|
|
if (RegisterInfo* info = state.tryGetRegisterInfo(inst.a))
|
|
|
|
{
|
|
|
|
if (info->knownNotReadonly)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
info->knownNotReadonly = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::CHECK_NO_METATABLE:
|
|
|
|
if (RegisterInfo* info = state.tryGetRegisterInfo(inst.a))
|
|
|
|
{
|
|
|
|
if (info->knownNoMetatable)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
info->knownNoMetatable = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::CHECK_SAFE_ENV:
|
|
|
|
if (state.inSafeEnv)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
state.inSafeEnv = true;
|
|
|
|
break;
|
|
|
|
case IrCmd::CHECK_GC:
|
|
|
|
// It is enough to perform a GC check once in a block
|
|
|
|
if (state.checkedGc)
|
|
|
|
kill(function, inst);
|
|
|
|
else
|
|
|
|
state.checkedGc = true;
|
|
|
|
break;
|
|
|
|
case IrCmd::BARRIER_OBJ:
|
|
|
|
case IrCmd::BARRIER_TABLE_FORWARD:
|
|
|
|
if (inst.b.kind == IrOpKind::VmReg)
|
|
|
|
{
|
|
|
|
if (uint8_t tag = state.tryGetTag(inst.b); tag != 0xff)
|
|
|
|
{
|
|
|
|
// If the written object is not collectable, barrier is not required
|
|
|
|
if (!isGCO(tag))
|
|
|
|
kill(function, inst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case IrCmd::LOP_AND:
|
|
|
|
case IrCmd::LOP_ANDK:
|
|
|
|
case IrCmd::LOP_OR:
|
|
|
|
case IrCmd::LOP_ORK:
|
|
|
|
state.invalidate(inst.b);
|
|
|
|
break;
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCmd::FASTCALL:
|
|
|
|
case IrCmd::INVOKE_FASTCALL:
|
|
|
|
handleBuiltinEffects(state, LuauBuiltinFunction(function.uintOp(inst.a)), inst.b.index, function.intOp(inst.f));
|
|
|
|
break;
|
2023-02-24 21:49:38 +00:00
|
|
|
|
|
|
|
// These instructions don't have an effect on register/memory state we are tracking
|
|
|
|
case IrCmd::NOP:
|
|
|
|
case IrCmd::LOAD_NODE_VALUE_TV:
|
|
|
|
case IrCmd::LOAD_ENV:
|
|
|
|
case IrCmd::GET_ARR_ADDR:
|
|
|
|
case IrCmd::GET_SLOT_NODE_ADDR:
|
|
|
|
case IrCmd::STORE_NODE_VALUE_TV:
|
|
|
|
case IrCmd::ADD_INT:
|
|
|
|
case IrCmd::SUB_INT:
|
|
|
|
case IrCmd::ADD_NUM:
|
|
|
|
case IrCmd::SUB_NUM:
|
|
|
|
case IrCmd::MUL_NUM:
|
|
|
|
case IrCmd::DIV_NUM:
|
|
|
|
case IrCmd::MOD_NUM:
|
|
|
|
case IrCmd::POW_NUM:
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCmd::MIN_NUM:
|
|
|
|
case IrCmd::MAX_NUM:
|
2023-02-24 21:49:38 +00:00
|
|
|
case IrCmd::UNM_NUM:
|
|
|
|
case IrCmd::NOT_ANY:
|
|
|
|
case IrCmd::JUMP:
|
|
|
|
case IrCmd::JUMP_EQ_POINTER:
|
|
|
|
case IrCmd::TABLE_LEN:
|
|
|
|
case IrCmd::NEW_TABLE:
|
|
|
|
case IrCmd::DUP_TABLE:
|
|
|
|
case IrCmd::NUM_TO_INDEX:
|
|
|
|
case IrCmd::INT_TO_NUM:
|
|
|
|
case IrCmd::CHECK_ARRAY_SIZE:
|
|
|
|
case IrCmd::CHECK_SLOT_MATCH:
|
|
|
|
case IrCmd::BARRIER_TABLE_BACK:
|
|
|
|
case IrCmd::LOP_RETURN:
|
|
|
|
case IrCmd::LOP_COVERAGE:
|
|
|
|
case IrCmd::SET_UPVALUE:
|
|
|
|
case IrCmd::LOP_SETLIST: // We don't track table state that this can invalidate
|
|
|
|
case IrCmd::SET_SAVEDPC: // TODO: we may be able to remove some updates to PC
|
|
|
|
case IrCmd::CLOSE_UPVALS: // Doesn't change memory that we track
|
|
|
|
case IrCmd::CAPTURE:
|
|
|
|
case IrCmd::SUBSTITUTE:
|
|
|
|
case IrCmd::ADJUST_STACK_TO_REG: // Changes stack top, but not the values
|
|
|
|
case IrCmd::ADJUST_STACK_TO_TOP: // Changes stack top, but not the values
|
2023-03-03 20:21:14 +00:00
|
|
|
case IrCmd::CHECK_FASTCALL_RES: // Changes stack top, but not the values
|
2023-02-24 21:49:38 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
// We don't model the following instructions, so we just clear all the knowledge we have built up
|
|
|
|
// Many of these call user functions that can change memory and captured registers
|
|
|
|
// Some of these might yield with similar effects
|
|
|
|
case IrCmd::JUMP_CMP_ANY:
|
|
|
|
case IrCmd::DO_ARITH:
|
|
|
|
case IrCmd::DO_LEN:
|
|
|
|
case IrCmd::GET_TABLE:
|
|
|
|
case IrCmd::SET_TABLE:
|
|
|
|
case IrCmd::GET_IMPORT:
|
|
|
|
case IrCmd::CONCAT:
|
|
|
|
case IrCmd::PREPARE_FORN:
|
|
|
|
case IrCmd::INTERRUPT: // TODO: it will be important to keep tag/value state, but we have to track register capture
|
|
|
|
case IrCmd::LOP_NAMECALL:
|
|
|
|
case IrCmd::LOP_CALL:
|
|
|
|
case IrCmd::LOP_FORGLOOP:
|
|
|
|
case IrCmd::LOP_FORGLOOP_FALLBACK:
|
|
|
|
case IrCmd::LOP_FORGPREP_XNEXT_FALLBACK:
|
|
|
|
case IrCmd::FALLBACK_GETGLOBAL:
|
|
|
|
case IrCmd::FALLBACK_SETGLOBAL:
|
|
|
|
case IrCmd::FALLBACK_GETTABLEKS:
|
|
|
|
case IrCmd::FALLBACK_SETTABLEKS:
|
|
|
|
case IrCmd::FALLBACK_NAMECALL:
|
|
|
|
case IrCmd::FALLBACK_PREPVARARGS:
|
|
|
|
case IrCmd::FALLBACK_GETVARARGS:
|
|
|
|
case IrCmd::FALLBACK_NEWCLOSURE:
|
|
|
|
case IrCmd::FALLBACK_DUPCLOSURE:
|
|
|
|
case IrCmd::FALLBACK_FORGPREP:
|
|
|
|
// TODO: this is very conservative, some of there instructions can be tracked better
|
|
|
|
// TODO: non-captured register tags and values should not be cleared here
|
|
|
|
state.invalidateAll();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void constPropInBlock(IrBuilder& build, IrBlock& block, ConstPropState& state)
|
|
|
|
{
|
|
|
|
IrFunction& function = build.function;
|
|
|
|
|
|
|
|
for (uint32_t index = block.start; index <= block.finish; index++)
|
|
|
|
{
|
|
|
|
LUAU_ASSERT(index < function.instructions.size());
|
|
|
|
IrInst& inst = function.instructions[index];
|
|
|
|
|
|
|
|
applySubstitutions(function, inst);
|
|
|
|
|
|
|
|
foldConstants(build, function, block, index);
|
|
|
|
|
|
|
|
constPropInInst(state, build, function, block, inst, index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void constPropInBlockChain(IrBuilder& build, std::vector<uint8_t>& visited, IrBlock* block)
|
|
|
|
{
|
|
|
|
IrFunction& function = build.function;
|
|
|
|
|
|
|
|
ConstPropState state;
|
|
|
|
|
|
|
|
while (block)
|
|
|
|
{
|
|
|
|
uint32_t blockIdx = function.getBlockIndex(*block);
|
|
|
|
LUAU_ASSERT(!visited[blockIdx]);
|
|
|
|
visited[blockIdx] = true;
|
|
|
|
|
|
|
|
constPropInBlock(build, *block, state);
|
|
|
|
|
|
|
|
IrInst& termInst = function.instructions[block->finish];
|
|
|
|
|
|
|
|
IrBlock* nextBlock = nullptr;
|
|
|
|
|
|
|
|
// Unconditional jump into a block with a single user (current block) allows us to continue optimization
|
|
|
|
// with the information we have gathered so far (unless we have already visited that block earlier)
|
|
|
|
if (termInst.cmd == IrCmd::JUMP)
|
|
|
|
{
|
|
|
|
IrBlock& target = function.blockOp(termInst.a);
|
2023-03-03 20:21:14 +00:00
|
|
|
uint32_t targetIdx = function.getBlockIndex(target);
|
2023-02-24 21:49:38 +00:00
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
if (target.useCount == 1 && !visited[targetIdx] && target.kind != IrBlockKind::Fallback)
|
2023-02-24 21:49:38 +00:00
|
|
|
nextBlock = ⌖
|
|
|
|
}
|
|
|
|
|
|
|
|
block = nextBlock;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
// Note that blocks in the collected path are marked as visited
|
|
|
|
static std::vector<uint32_t> collectDirectBlockJumpPath(IrFunction& function, std::vector<uint8_t>& visited, IrBlock* block)
|
|
|
|
{
|
|
|
|
// Path shouldn't be built starting with a block that has 'live out' values.
|
|
|
|
// One theoretical way to get it is if we had 'block' jumping unconditionally into a successor that uses values from 'block'
|
|
|
|
// * if the successor has only one use, the starting conditions of 'tryCreateLinearBlock' would prevent this
|
|
|
|
// * if the successor has multiple uses, it can't have such 'live in' values without phi nodes that we don't have yet
|
|
|
|
// Another possibility is to have two paths from 'block' into the target through two intermediate blocks
|
|
|
|
// Usually that would mean that we would have a conditional jump at the end of 'block'
|
|
|
|
// But using check guards and fallback clocks it becomes a possible setup
|
|
|
|
// We avoid this by making sure fallbacks rejoin the other immediate successor of 'block'
|
|
|
|
LUAU_ASSERT(getLiveOutValueCount(function, *block) == 0);
|
|
|
|
|
|
|
|
std::vector<uint32_t> path;
|
|
|
|
|
|
|
|
while (block)
|
|
|
|
{
|
|
|
|
IrInst& termInst = function.instructions[block->finish];
|
|
|
|
IrBlock* nextBlock = nullptr;
|
|
|
|
|
|
|
|
// A chain is made from internal blocks that were not a part of bytecode CFG
|
|
|
|
if (termInst.cmd == IrCmd::JUMP)
|
|
|
|
{
|
|
|
|
IrBlock& target = function.blockOp(termInst.a);
|
|
|
|
uint32_t targetIdx = function.getBlockIndex(target);
|
|
|
|
|
|
|
|
if (!visited[targetIdx] && target.kind == IrBlockKind::Internal)
|
|
|
|
{
|
|
|
|
// Additional restriction is that to join a block, it cannot produce values that are used in other blocks
|
|
|
|
// And it also can't use values produced in other blocks
|
|
|
|
auto [liveIns, liveOuts] = getLiveInOutValueCount(function, target);
|
|
|
|
|
|
|
|
if (liveIns == 0 && liveOuts == 0)
|
|
|
|
{
|
|
|
|
visited[targetIdx] = true;
|
|
|
|
path.push_back(targetIdx);
|
|
|
|
|
|
|
|
nextBlock = ⌖
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
block = nextBlock;
|
|
|
|
}
|
|
|
|
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tryCreateLinearBlock(IrBuilder& build, std::vector<uint8_t>& visited, IrBlock& startingBlock)
|
|
|
|
{
|
|
|
|
IrFunction& function = build.function;
|
|
|
|
|
|
|
|
uint32_t blockIdx = function.getBlockIndex(startingBlock);
|
|
|
|
LUAU_ASSERT(!visited[blockIdx]);
|
|
|
|
visited[blockIdx] = true;
|
|
|
|
|
|
|
|
IrInst& termInst = function.instructions[startingBlock.finish];
|
|
|
|
|
|
|
|
// Block has to end with an unconditional jump
|
|
|
|
if (termInst.cmd != IrCmd::JUMP)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// And it has to jump to a block with more than one user
|
|
|
|
// If there's only one use, it should already be optimized by constPropInBlockChain
|
|
|
|
if (function.blockOp(termInst.a).useCount == 1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
uint32_t targetBlockIdx = termInst.a.index;
|
|
|
|
|
|
|
|
// Check if this path is worth it (and it will also mark path blocks as visited)
|
|
|
|
std::vector<uint32_t> path = collectDirectBlockJumpPath(function, visited, &startingBlock);
|
|
|
|
|
|
|
|
// If path is too small, we are not going to linearize it
|
|
|
|
if (int(path.size()) < FInt::LuauCodeGenMinLinearBlockPath)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Initialize state with the knowledge of our current block
|
|
|
|
ConstPropState state;
|
|
|
|
constPropInBlock(build, startingBlock, state);
|
|
|
|
|
|
|
|
// Veryfy that target hasn't changed
|
|
|
|
LUAU_ASSERT(function.instructions[startingBlock.finish].a.index == targetBlockIdx);
|
|
|
|
|
|
|
|
// Create new linearized block into which we are going to redirect starting block jump
|
|
|
|
IrOp newBlock = build.block(IrBlockKind::Linearized);
|
|
|
|
visited.push_back(false);
|
|
|
|
|
|
|
|
// TODO: placement of linear blocks in final lowering is sub-optimal, it should follow our predecessor
|
|
|
|
build.beginBlock(newBlock);
|
|
|
|
|
|
|
|
replace(function, termInst.a, newBlock);
|
|
|
|
|
|
|
|
// Clone the collected path int our fresh block
|
|
|
|
for (uint32_t pathBlockIdx : path)
|
|
|
|
build.clone(function.blocks[pathBlockIdx], /* removeCurrentTerminator */ true);
|
|
|
|
|
|
|
|
// Optimize our linear block
|
|
|
|
IrBlock& linearBlock = function.blockOp(newBlock);
|
|
|
|
constPropInBlock(build, linearBlock, state);
|
|
|
|
}
|
|
|
|
|
2023-02-24 21:49:38 +00:00
|
|
|
void constPropInBlockChains(IrBuilder& build)
|
|
|
|
{
|
|
|
|
IrFunction& function = build.function;
|
|
|
|
|
|
|
|
std::vector<uint8_t> visited(function.blocks.size(), false);
|
|
|
|
|
2023-03-03 20:21:14 +00:00
|
|
|
// First pass: go over existing blocks once and propagate constants
|
2023-02-24 21:49:38 +00:00
|
|
|
for (IrBlock& block : function.blocks)
|
|
|
|
{
|
|
|
|
if (block.kind == IrBlockKind::Fallback || block.kind == IrBlockKind::Dead)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (visited[function.getBlockIndex(block)])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
constPropInBlockChain(build, visited, &block);
|
|
|
|
}
|
2023-03-03 20:21:14 +00:00
|
|
|
|
|
|
|
// Second pass: go through internal block chains and outline them into a single new block
|
|
|
|
// Outlining will be able to linearize the execution, even if there was a jump to a block with multiple users,
|
|
|
|
// new 'block' will only be reachable from a single one and all gathered information can be preserved.
|
|
|
|
std::fill(visited.begin(), visited.end(), false);
|
|
|
|
|
|
|
|
// This next loop can create new 'linear' blocks, so index-based loop has to be used (and it intentionally won't reach those new blocks)
|
|
|
|
size_t originalBlockCount = function.blocks.size();
|
|
|
|
for (size_t i = 0; i < originalBlockCount; i++)
|
|
|
|
{
|
|
|
|
IrBlock& block = function.blocks[i];
|
|
|
|
|
|
|
|
if (block.kind == IrBlockKind::Fallback || block.kind == IrBlockKind::Dead)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (visited[function.getBlockIndex(block)])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
tryCreateLinearBlock(build, visited, block);
|
|
|
|
}
|
2023-02-24 21:49:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace CodeGen
|
|
|
|
} // namespace Luau
|