2023-01-13 20:36:28 +00:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
2023-02-03 12:34:12 +00:00
|
|
|
#include "Luau/IrDump.h"
|
2023-01-13 20:36:28 +00:00
|
|
|
|
2023-02-03 12:34:12 +00:00
|
|
|
#include "Luau/IrUtils.h"
|
2023-01-13 20:36:28 +00:00
|
|
|
|
|
|
|
#include "lua.h"
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace CodeGen
|
|
|
|
{
|
|
|
|
|
|
|
|
static const char* textForCondition[] = {
|
|
|
|
"eq", "not_eq", "lt", "not_lt", "le", "not_le", "gt", "not_gt", "ge", "not_ge", "u_lt", "u_le", "u_gt", "u_ge"};
|
|
|
|
static_assert(sizeof(textForCondition) / sizeof(textForCondition[0]) == size_t(IrCondition::Count), "all conditions have to be covered");
|
|
|
|
|
|
|
|
const int kDetailsAlignColumn = 60;
|
|
|
|
|
|
|
|
LUAU_PRINTF_ATTR(2, 3)
|
|
|
|
static void append(std::string& result, const char* fmt, ...)
|
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
va_list args;
|
|
|
|
va_start(args, fmt);
|
|
|
|
vsnprintf(buf, sizeof(buf), fmt, args);
|
|
|
|
va_end(args);
|
|
|
|
result.append(buf);
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
static void padToDetailColumn(std::string& result, size_t lineStart)
|
|
|
|
{
|
|
|
|
int pad = kDetailsAlignColumn - int(result.size() - lineStart);
|
|
|
|
|
|
|
|
if (pad > 0)
|
|
|
|
result.append(pad, ' ');
|
|
|
|
}
|
|
|
|
|
2023-01-13 20:36:28 +00:00
|
|
|
static const char* getTagName(uint8_t tag)
|
|
|
|
{
|
|
|
|
switch (tag)
|
|
|
|
{
|
|
|
|
case LUA_TNIL:
|
|
|
|
return "tnil";
|
|
|
|
case LUA_TBOOLEAN:
|
|
|
|
return "tboolean";
|
|
|
|
case LUA_TLIGHTUSERDATA:
|
|
|
|
return "tlightuserdata";
|
|
|
|
case LUA_TNUMBER:
|
|
|
|
return "tnumber";
|
|
|
|
case LUA_TVECTOR:
|
|
|
|
return "tvector";
|
|
|
|
case LUA_TSTRING:
|
|
|
|
return "tstring";
|
|
|
|
case LUA_TTABLE:
|
|
|
|
return "ttable";
|
|
|
|
case LUA_TFUNCTION:
|
|
|
|
return "tfunction";
|
|
|
|
case LUA_TUSERDATA:
|
|
|
|
return "tuserdata";
|
|
|
|
case LUA_TTHREAD:
|
|
|
|
return "tthread";
|
|
|
|
default:
|
|
|
|
LUAU_UNREACHABLE();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* getCmdName(IrCmd cmd)
|
|
|
|
{
|
|
|
|
switch (cmd)
|
|
|
|
{
|
|
|
|
case IrCmd::NOP:
|
|
|
|
return "NOP";
|
|
|
|
case IrCmd::LOAD_TAG:
|
|
|
|
return "LOAD_TAG";
|
|
|
|
case IrCmd::LOAD_POINTER:
|
|
|
|
return "LOAD_POINTER";
|
|
|
|
case IrCmd::LOAD_DOUBLE:
|
|
|
|
return "LOAD_DOUBLE";
|
|
|
|
case IrCmd::LOAD_INT:
|
|
|
|
return "LOAD_INT";
|
|
|
|
case IrCmd::LOAD_TVALUE:
|
|
|
|
return "LOAD_TVALUE";
|
|
|
|
case IrCmd::LOAD_NODE_VALUE_TV:
|
|
|
|
return "LOAD_NODE_VALUE_TV";
|
|
|
|
case IrCmd::LOAD_ENV:
|
|
|
|
return "LOAD_ENV";
|
|
|
|
case IrCmd::GET_ARR_ADDR:
|
|
|
|
return "GET_ARR_ADDR";
|
|
|
|
case IrCmd::GET_SLOT_NODE_ADDR:
|
|
|
|
return "GET_SLOT_NODE_ADDR";
|
2023-03-17 14:59:30 +00:00
|
|
|
case IrCmd::GET_HASH_NODE_ADDR:
|
|
|
|
return "GET_HASH_NODE_ADDR";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::STORE_TAG:
|
|
|
|
return "STORE_TAG";
|
|
|
|
case IrCmd::STORE_POINTER:
|
|
|
|
return "STORE_POINTER";
|
|
|
|
case IrCmd::STORE_DOUBLE:
|
|
|
|
return "STORE_DOUBLE";
|
|
|
|
case IrCmd::STORE_INT:
|
|
|
|
return "STORE_INT";
|
|
|
|
case IrCmd::STORE_TVALUE:
|
|
|
|
return "STORE_TVALUE";
|
|
|
|
case IrCmd::STORE_NODE_VALUE_TV:
|
|
|
|
return "STORE_NODE_VALUE_TV";
|
|
|
|
case IrCmd::ADD_INT:
|
|
|
|
return "ADD_INT";
|
|
|
|
case IrCmd::SUB_INT:
|
|
|
|
return "SUB_INT";
|
|
|
|
case IrCmd::ADD_NUM:
|
|
|
|
return "ADD_NUM";
|
|
|
|
case IrCmd::SUB_NUM:
|
|
|
|
return "SUB_NUM";
|
|
|
|
case IrCmd::MUL_NUM:
|
|
|
|
return "MUL_NUM";
|
|
|
|
case IrCmd::DIV_NUM:
|
|
|
|
return "DIV_NUM";
|
|
|
|
case IrCmd::MOD_NUM:
|
|
|
|
return "MOD_NUM";
|
|
|
|
case IrCmd::POW_NUM:
|
|
|
|
return "POW_NUM";
|
2023-03-03 13:45:38 +00:00
|
|
|
case IrCmd::MIN_NUM:
|
|
|
|
return "MIN_NUM";
|
|
|
|
case IrCmd::MAX_NUM:
|
|
|
|
return "MAX_NUM";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::UNM_NUM:
|
|
|
|
return "UNM_NUM";
|
2023-03-31 13:21:14 +01:00
|
|
|
case IrCmd::FLOOR_NUM:
|
|
|
|
return "FLOOR_NUM";
|
|
|
|
case IrCmd::CEIL_NUM:
|
|
|
|
return "CEIL_NUM";
|
|
|
|
case IrCmd::ROUND_NUM:
|
|
|
|
return "ROUND_NUM";
|
|
|
|
case IrCmd::SQRT_NUM:
|
|
|
|
return "SQRT_NUM";
|
|
|
|
case IrCmd::ABS_NUM:
|
|
|
|
return "ABS_NUM";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::NOT_ANY:
|
|
|
|
return "NOT_ANY";
|
|
|
|
case IrCmd::JUMP:
|
|
|
|
return "JUMP";
|
|
|
|
case IrCmd::JUMP_IF_TRUTHY:
|
|
|
|
return "JUMP_IF_TRUTHY";
|
|
|
|
case IrCmd::JUMP_IF_FALSY:
|
|
|
|
return "JUMP_IF_FALSY";
|
|
|
|
case IrCmd::JUMP_EQ_TAG:
|
|
|
|
return "JUMP_EQ_TAG";
|
2023-02-10 18:50:54 +00:00
|
|
|
case IrCmd::JUMP_EQ_INT:
|
|
|
|
return "JUMP_EQ_INT";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::JUMP_EQ_POINTER:
|
|
|
|
return "JUMP_EQ_POINTER";
|
|
|
|
case IrCmd::JUMP_CMP_NUM:
|
|
|
|
return "JUMP_CMP_NUM";
|
|
|
|
case IrCmd::JUMP_CMP_ANY:
|
|
|
|
return "JUMP_CMP_ANY";
|
2023-03-17 14:59:30 +00:00
|
|
|
case IrCmd::JUMP_SLOT_MATCH:
|
|
|
|
return "JUMP_SLOT_MATCH";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::TABLE_LEN:
|
|
|
|
return "TABLE_LEN";
|
|
|
|
case IrCmd::NEW_TABLE:
|
|
|
|
return "NEW_TABLE";
|
|
|
|
case IrCmd::DUP_TABLE:
|
|
|
|
return "DUP_TABLE";
|
2023-03-17 14:59:30 +00:00
|
|
|
case IrCmd::TRY_NUM_TO_INDEX:
|
|
|
|
return "TRY_NUM_TO_INDEX";
|
|
|
|
case IrCmd::TRY_CALL_FASTGETTM:
|
|
|
|
return "TRY_CALL_FASTGETTM";
|
2023-02-10 18:50:54 +00:00
|
|
|
case IrCmd::INT_TO_NUM:
|
|
|
|
return "INT_TO_NUM";
|
2023-02-24 18:24:22 +00:00
|
|
|
case IrCmd::ADJUST_STACK_TO_REG:
|
|
|
|
return "ADJUST_STACK_TO_REG";
|
|
|
|
case IrCmd::ADJUST_STACK_TO_TOP:
|
|
|
|
return "ADJUST_STACK_TO_TOP";
|
2023-03-03 13:45:38 +00:00
|
|
|
case IrCmd::FASTCALL:
|
|
|
|
return "FASTCALL";
|
|
|
|
case IrCmd::INVOKE_FASTCALL:
|
|
|
|
return "INVOKE_FASTCALL";
|
|
|
|
case IrCmd::CHECK_FASTCALL_RES:
|
|
|
|
return "CHECK_FASTCALL_RES";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::DO_ARITH:
|
|
|
|
return "DO_ARITH";
|
|
|
|
case IrCmd::DO_LEN:
|
|
|
|
return "DO_LEN";
|
|
|
|
case IrCmd::GET_TABLE:
|
|
|
|
return "GET_TABLE";
|
|
|
|
case IrCmd::SET_TABLE:
|
|
|
|
return "SET_TABLE";
|
|
|
|
case IrCmd::GET_IMPORT:
|
|
|
|
return "GET_IMPORT";
|
|
|
|
case IrCmd::CONCAT:
|
|
|
|
return "CONCAT";
|
|
|
|
case IrCmd::GET_UPVALUE:
|
|
|
|
return "GET_UPVALUE";
|
|
|
|
case IrCmd::SET_UPVALUE:
|
|
|
|
return "SET_UPVALUE";
|
2023-02-10 18:50:54 +00:00
|
|
|
case IrCmd::PREPARE_FORN:
|
|
|
|
return "PREPARE_FORN";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::CHECK_TAG:
|
|
|
|
return "CHECK_TAG";
|
|
|
|
case IrCmd::CHECK_READONLY:
|
|
|
|
return "CHECK_READONLY";
|
|
|
|
case IrCmd::CHECK_NO_METATABLE:
|
|
|
|
return "CHECK_NO_METATABLE";
|
|
|
|
case IrCmd::CHECK_SAFE_ENV:
|
|
|
|
return "CHECK_SAFE_ENV";
|
|
|
|
case IrCmd::CHECK_ARRAY_SIZE:
|
|
|
|
return "CHECK_ARRAY_SIZE";
|
|
|
|
case IrCmd::CHECK_SLOT_MATCH:
|
|
|
|
return "CHECK_SLOT_MATCH";
|
2023-03-17 14:59:30 +00:00
|
|
|
case IrCmd::CHECK_NODE_NO_NEXT:
|
|
|
|
return "CHECK_NODE_NO_NEXT";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::INTERRUPT:
|
|
|
|
return "INTERRUPT";
|
|
|
|
case IrCmd::CHECK_GC:
|
|
|
|
return "CHECK_GC";
|
|
|
|
case IrCmd::BARRIER_OBJ:
|
|
|
|
return "BARRIER_OBJ";
|
|
|
|
case IrCmd::BARRIER_TABLE_BACK:
|
|
|
|
return "BARRIER_TABLE_BACK";
|
|
|
|
case IrCmd::BARRIER_TABLE_FORWARD:
|
|
|
|
return "BARRIER_TABLE_FORWARD";
|
|
|
|
case IrCmd::SET_SAVEDPC:
|
|
|
|
return "SET_SAVEDPC";
|
|
|
|
case IrCmd::CLOSE_UPVALS:
|
|
|
|
return "CLOSE_UPVALS";
|
|
|
|
case IrCmd::CAPTURE:
|
|
|
|
return "CAPTURE";
|
2023-03-31 13:21:14 +01:00
|
|
|
case IrCmd::SETLIST:
|
|
|
|
return "SETLIST";
|
|
|
|
case IrCmd::CALL:
|
|
|
|
return "CALL";
|
|
|
|
case IrCmd::RETURN:
|
|
|
|
return "RETURN";
|
|
|
|
case IrCmd::FORGLOOP:
|
|
|
|
return "FORGLOOP";
|
|
|
|
case IrCmd::FORGLOOP_FALLBACK:
|
|
|
|
return "FORGLOOP_FALLBACK";
|
|
|
|
case IrCmd::FORGPREP_XNEXT_FALLBACK:
|
|
|
|
return "FORGPREP_XNEXT_FALLBACK";
|
|
|
|
case IrCmd::AND:
|
|
|
|
return "AND";
|
|
|
|
case IrCmd::ANDK:
|
|
|
|
return "ANDK";
|
|
|
|
case IrCmd::OR:
|
|
|
|
return "OR";
|
|
|
|
case IrCmd::ORK:
|
|
|
|
return "ORK";
|
|
|
|
case IrCmd::COVERAGE:
|
|
|
|
return "COVERAGE";
|
2023-01-13 20:36:28 +00:00
|
|
|
case IrCmd::FALLBACK_GETGLOBAL:
|
|
|
|
return "FALLBACK_GETGLOBAL";
|
|
|
|
case IrCmd::FALLBACK_SETGLOBAL:
|
|
|
|
return "FALLBACK_SETGLOBAL";
|
|
|
|
case IrCmd::FALLBACK_GETTABLEKS:
|
|
|
|
return "FALLBACK_GETTABLEKS";
|
|
|
|
case IrCmd::FALLBACK_SETTABLEKS:
|
|
|
|
return "FALLBACK_SETTABLEKS";
|
|
|
|
case IrCmd::FALLBACK_NAMECALL:
|
|
|
|
return "FALLBACK_NAMECALL";
|
|
|
|
case IrCmd::FALLBACK_PREPVARARGS:
|
|
|
|
return "FALLBACK_PREPVARARGS";
|
|
|
|
case IrCmd::FALLBACK_GETVARARGS:
|
|
|
|
return "FALLBACK_GETVARARGS";
|
|
|
|
case IrCmd::FALLBACK_NEWCLOSURE:
|
|
|
|
return "FALLBACK_NEWCLOSURE";
|
|
|
|
case IrCmd::FALLBACK_DUPCLOSURE:
|
|
|
|
return "FALLBACK_DUPCLOSURE";
|
|
|
|
case IrCmd::FALLBACK_FORGPREP:
|
|
|
|
return "FALLBACK_FORGPREP";
|
2023-02-17 14:53:37 +00:00
|
|
|
case IrCmd::SUBSTITUTE:
|
|
|
|
return "SUBSTITUTE";
|
2023-01-13 20:36:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LUAU_UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* getBlockKindName(IrBlockKind kind)
|
|
|
|
{
|
|
|
|
switch (kind)
|
|
|
|
{
|
|
|
|
case IrBlockKind::Bytecode:
|
|
|
|
return "bb_bytecode";
|
|
|
|
case IrBlockKind::Fallback:
|
|
|
|
return "bb_fallback";
|
|
|
|
case IrBlockKind::Internal:
|
|
|
|
return "bb";
|
2023-03-03 13:45:38 +00:00
|
|
|
case IrBlockKind::Linearized:
|
|
|
|
return "bb_linear";
|
2023-02-10 18:50:54 +00:00
|
|
|
case IrBlockKind::Dead:
|
|
|
|
return "dead";
|
2023-01-13 20:36:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LUAU_UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
void toString(IrToStringContext& ctx, const IrInst& inst, uint32_t index)
|
2023-01-13 20:36:28 +00:00
|
|
|
{
|
|
|
|
append(ctx.result, " ");
|
|
|
|
|
|
|
|
// Instructions with a result display target virtual register
|
|
|
|
if (hasResult(inst.cmd))
|
|
|
|
append(ctx.result, "%%%u = ", index);
|
|
|
|
|
|
|
|
ctx.result.append(getCmdName(inst.cmd));
|
|
|
|
|
2023-02-24 18:24:22 +00:00
|
|
|
auto checkOp = [&ctx](IrOp op, const char* sep) {
|
|
|
|
if (op.kind != IrOpKind::None)
|
|
|
|
{
|
|
|
|
ctx.result.append(sep);
|
|
|
|
toString(ctx, op);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
checkOp(inst.a, " ");
|
|
|
|
checkOp(inst.b, ", ");
|
|
|
|
checkOp(inst.c, ", ");
|
|
|
|
checkOp(inst.d, ", ");
|
|
|
|
checkOp(inst.e, ", ");
|
|
|
|
checkOp(inst.f, ", ");
|
2023-01-13 20:36:28 +00:00
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
void toString(IrToStringContext& ctx, const IrBlock& block, uint32_t index)
|
|
|
|
{
|
2023-03-10 19:20:04 +00:00
|
|
|
append(ctx.result, "%s_%u", getBlockKindName(block.kind), index);
|
2023-02-10 18:50:54 +00:00
|
|
|
}
|
|
|
|
|
2023-01-13 20:36:28 +00:00
|
|
|
void toString(IrToStringContext& ctx, IrOp op)
|
|
|
|
{
|
|
|
|
switch (op.kind)
|
|
|
|
{
|
|
|
|
case IrOpKind::None:
|
|
|
|
break;
|
|
|
|
case IrOpKind::Constant:
|
|
|
|
toString(ctx.result, ctx.constants[op.index]);
|
|
|
|
break;
|
|
|
|
case IrOpKind::Condition:
|
|
|
|
LUAU_ASSERT(op.index < uint32_t(IrCondition::Count));
|
|
|
|
ctx.result.append(textForCondition[op.index]);
|
|
|
|
break;
|
|
|
|
case IrOpKind::Inst:
|
|
|
|
append(ctx.result, "%%%u", op.index);
|
|
|
|
break;
|
|
|
|
case IrOpKind::Block:
|
|
|
|
append(ctx.result, "%s_%u", getBlockKindName(ctx.blocks[op.index].kind), op.index);
|
|
|
|
break;
|
|
|
|
case IrOpKind::VmReg:
|
|
|
|
append(ctx.result, "R%u", op.index);
|
|
|
|
break;
|
|
|
|
case IrOpKind::VmConst:
|
|
|
|
append(ctx.result, "K%u", op.index);
|
|
|
|
break;
|
|
|
|
case IrOpKind::VmUpvalue:
|
|
|
|
append(ctx.result, "U%u", op.index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void toString(std::string& result, IrConst constant)
|
|
|
|
{
|
|
|
|
switch (constant.kind)
|
|
|
|
{
|
|
|
|
case IrConstKind::Bool:
|
|
|
|
append(result, constant.valueBool ? "true" : "false");
|
|
|
|
break;
|
|
|
|
case IrConstKind::Int:
|
|
|
|
append(result, "%di", constant.valueInt);
|
|
|
|
break;
|
|
|
|
case IrConstKind::Uint:
|
|
|
|
append(result, "%uu", constant.valueUint);
|
|
|
|
break;
|
|
|
|
case IrConstKind::Double:
|
|
|
|
append(result, "%.17g", constant.valueDouble);
|
|
|
|
break;
|
|
|
|
case IrConstKind::Tag:
|
|
|
|
result.append(getTagName(constant.valueTag));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 19:20:04 +00:00
|
|
|
void toStringDetailed(IrToStringContext& ctx, const IrInst& inst, uint32_t index, bool includeUseInfo)
|
2023-01-13 20:36:28 +00:00
|
|
|
{
|
|
|
|
size_t start = ctx.result.size();
|
|
|
|
|
|
|
|
toString(ctx, inst, index);
|
|
|
|
|
2023-03-10 19:20:04 +00:00
|
|
|
if (includeUseInfo)
|
|
|
|
{
|
|
|
|
padToDetailColumn(ctx.result, start);
|
|
|
|
|
|
|
|
if (inst.useCount == 0 && hasSideEffects(inst.cmd))
|
|
|
|
append(ctx.result, "; %%%u, has side-effects\n", index);
|
|
|
|
else
|
|
|
|
append(ctx.result, "; useCount: %d, lastUse: %%%u\n", inst.useCount, inst.lastUse);
|
|
|
|
}
|
2023-01-13 20:36:28 +00:00
|
|
|
else
|
2023-03-10 19:20:04 +00:00
|
|
|
{
|
|
|
|
ctx.result.append("\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void appendBlockSet(IrToStringContext& ctx, BlockIteratorWrapper blocks)
|
|
|
|
{
|
|
|
|
bool comma = false;
|
|
|
|
|
|
|
|
for (uint32_t target : blocks)
|
|
|
|
{
|
|
|
|
if (comma)
|
|
|
|
append(ctx.result, ", ");
|
|
|
|
comma = true;
|
|
|
|
|
|
|
|
toString(ctx, ctx.blocks[target], target);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:59:30 +00:00
|
|
|
static void appendRegisterSet(IrToStringContext& ctx, const RegisterSet& rs, const char* separator)
|
2023-03-10 19:20:04 +00:00
|
|
|
{
|
|
|
|
bool comma = false;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < rs.regs.size(); i++)
|
|
|
|
{
|
|
|
|
if (rs.regs.test(i))
|
|
|
|
{
|
|
|
|
if (comma)
|
2023-03-17 14:59:30 +00:00
|
|
|
ctx.result.append(separator);
|
2023-03-10 19:20:04 +00:00
|
|
|
comma = true;
|
|
|
|
|
|
|
|
append(ctx.result, "R%d", int(i));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rs.varargSeq)
|
|
|
|
{
|
|
|
|
if (comma)
|
2023-03-17 14:59:30 +00:00
|
|
|
ctx.result.append(separator);
|
2023-03-10 19:20:04 +00:00
|
|
|
|
|
|
|
append(ctx.result, "R%d...", rs.varargStart);
|
|
|
|
}
|
2023-01-13 20:36:28 +00:00
|
|
|
}
|
|
|
|
|
2023-03-10 19:20:04 +00:00
|
|
|
void toStringDetailed(IrToStringContext& ctx, const IrBlock& block, uint32_t index, bool includeUseInfo)
|
2023-02-10 18:50:54 +00:00
|
|
|
{
|
2023-03-10 19:20:04 +00:00
|
|
|
// Report captured registers for entry block
|
|
|
|
if (block.useCount == 0 && block.kind != IrBlockKind::Dead && ctx.cfg.captured.regs.any())
|
|
|
|
{
|
|
|
|
append(ctx.result, "; captured regs: ");
|
2023-03-17 14:59:30 +00:00
|
|
|
appendRegisterSet(ctx, ctx.cfg.captured, ", ");
|
2023-03-10 19:20:04 +00:00
|
|
|
append(ctx.result, "\n\n");
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
size_t start = ctx.result.size();
|
|
|
|
|
|
|
|
toString(ctx, block, index);
|
2023-03-10 19:20:04 +00:00
|
|
|
append(ctx.result, ":");
|
|
|
|
|
|
|
|
if (includeUseInfo)
|
|
|
|
{
|
|
|
|
padToDetailColumn(ctx.result, start);
|
2023-02-10 18:50:54 +00:00
|
|
|
|
2023-03-10 19:20:04 +00:00
|
|
|
append(ctx.result, "; useCount: %d\n", block.useCount);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ctx.result.append("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Predecessor list
|
2023-03-24 17:34:14 +00:00
|
|
|
if (index < ctx.cfg.predecessorsOffsets.size())
|
2023-03-10 19:20:04 +00:00
|
|
|
{
|
|
|
|
BlockIteratorWrapper pred = predecessors(ctx.cfg, index);
|
|
|
|
|
|
|
|
if (!pred.empty())
|
|
|
|
{
|
|
|
|
append(ctx.result, "; predecessors: ");
|
|
|
|
|
|
|
|
appendBlockSet(ctx, pred);
|
|
|
|
append(ctx.result, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Successor list
|
2023-03-24 17:34:14 +00:00
|
|
|
if (index < ctx.cfg.successorsOffsets.size())
|
2023-03-10 19:20:04 +00:00
|
|
|
{
|
|
|
|
BlockIteratorWrapper succ = successors(ctx.cfg, index);
|
|
|
|
|
|
|
|
if (!succ.empty())
|
|
|
|
{
|
|
|
|
append(ctx.result, "; successors: ");
|
|
|
|
|
|
|
|
appendBlockSet(ctx, succ);
|
|
|
|
append(ctx.result, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Live-in VM regs
|
|
|
|
if (index < ctx.cfg.in.size())
|
|
|
|
{
|
|
|
|
const RegisterSet& in = ctx.cfg.in[index];
|
|
|
|
|
|
|
|
if (in.regs.any() || in.varargSeq)
|
|
|
|
{
|
|
|
|
append(ctx.result, "; in regs: ");
|
2023-03-17 14:59:30 +00:00
|
|
|
appendRegisterSet(ctx, in, ", ");
|
2023-03-10 19:20:04 +00:00
|
|
|
append(ctx.result, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Live-out VM regs
|
|
|
|
if (index < ctx.cfg.out.size())
|
|
|
|
{
|
|
|
|
const RegisterSet& out = ctx.cfg.out[index];
|
|
|
|
|
|
|
|
if (out.regs.any() || out.varargSeq)
|
|
|
|
{
|
|
|
|
append(ctx.result, "; out regs: ");
|
2023-03-17 14:59:30 +00:00
|
|
|
appendRegisterSet(ctx, out, ", ");
|
2023-03-10 19:20:04 +00:00
|
|
|
append(ctx.result, "\n");
|
|
|
|
}
|
|
|
|
}
|
2023-02-10 18:50:54 +00:00
|
|
|
}
|
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
std::string toString(const IrFunction& function, bool includeUseInfo)
|
2023-01-20 12:02:39 +00:00
|
|
|
{
|
|
|
|
std::string result;
|
2023-03-10 19:20:04 +00:00
|
|
|
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
|
2023-01-20 12:02:39 +00:00
|
|
|
|
|
|
|
for (size_t i = 0; i < function.blocks.size(); i++)
|
|
|
|
{
|
2023-03-24 17:34:14 +00:00
|
|
|
const IrBlock& block = function.blocks[i];
|
2023-01-20 12:02:39 +00:00
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
if (block.kind == IrBlockKind::Dead)
|
|
|
|
continue;
|
|
|
|
|
2023-03-10 19:20:04 +00:00
|
|
|
toStringDetailed(ctx, block, uint32_t(i), includeUseInfo);
|
2023-01-20 12:02:39 +00:00
|
|
|
|
|
|
|
if (block.start == ~0u)
|
|
|
|
{
|
|
|
|
append(ctx.result, " *empty*\n\n");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
// To allow dumping blocks that are still being constructed, we can't rely on terminator and need a bounds check
|
2023-02-24 18:24:22 +00:00
|
|
|
for (uint32_t index = block.start; index <= block.finish && index < uint32_t(function.instructions.size()); index++)
|
2023-01-20 12:02:39 +00:00
|
|
|
{
|
2023-03-24 17:34:14 +00:00
|
|
|
const IrInst& inst = function.instructions[index];
|
2023-01-20 12:02:39 +00:00
|
|
|
|
2023-02-17 14:53:37 +00:00
|
|
|
// Skip pseudo instructions unless they are still referenced
|
|
|
|
if (isPseudo(inst.cmd) && inst.useCount == 0)
|
2023-01-20 12:02:39 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
append(ctx.result, " ");
|
2023-03-10 19:20:04 +00:00
|
|
|
toStringDetailed(ctx, inst, index, includeUseInfo);
|
2023-01-20 12:02:39 +00:00
|
|
|
}
|
2023-02-24 18:24:22 +00:00
|
|
|
|
|
|
|
append(ctx.result, "\n");
|
2023-01-20 12:02:39 +00:00
|
|
|
}
|
|
|
|
|
2023-02-10 18:50:54 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
std::string dump(const IrFunction& function)
|
2023-02-10 18:50:54 +00:00
|
|
|
{
|
2023-03-10 19:20:04 +00:00
|
|
|
std::string result = toString(function, /* includeUseInfo */ true);
|
2023-02-10 18:50:54 +00:00
|
|
|
|
2023-01-20 12:02:39 +00:00
|
|
|
printf("%s\n", result.c_str());
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
std::string toDot(const IrFunction& function, bool includeInst)
|
2023-03-17 14:59:30 +00:00
|
|
|
{
|
|
|
|
std::string result;
|
|
|
|
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
|
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
auto appendLabelRegset = [&ctx](const std::vector<RegisterSet>& regSets, size_t blockIdx, const char* name) {
|
2023-03-17 14:59:30 +00:00
|
|
|
if (blockIdx < regSets.size())
|
|
|
|
{
|
|
|
|
const RegisterSet& rs = regSets[blockIdx];
|
|
|
|
|
|
|
|
if (rs.regs.any() || rs.varargSeq)
|
|
|
|
{
|
|
|
|
append(ctx.result, "|{%s|", name);
|
|
|
|
appendRegisterSet(ctx, rs, "|");
|
|
|
|
append(ctx.result, "}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
append(ctx.result, "digraph CFG {\n");
|
|
|
|
append(ctx.result, "node[shape=record]\n");
|
|
|
|
|
|
|
|
for (size_t i = 0; i < function.blocks.size(); i++)
|
|
|
|
{
|
2023-03-24 17:34:14 +00:00
|
|
|
const IrBlock& block = function.blocks[i];
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
append(ctx.result, "b%u [", unsigned(i));
|
|
|
|
|
|
|
|
if (block.kind == IrBlockKind::Fallback)
|
|
|
|
append(ctx.result, "style=filled;fillcolor=salmon;");
|
|
|
|
else if (block.kind == IrBlockKind::Bytecode)
|
|
|
|
append(ctx.result, "style=filled;fillcolor=palegreen;");
|
|
|
|
|
|
|
|
append(ctx.result, "label=\"{");
|
|
|
|
toString(ctx, block, uint32_t(i));
|
|
|
|
|
|
|
|
appendLabelRegset(ctx.cfg.in, i, "in");
|
|
|
|
|
|
|
|
if (includeInst && block.start != ~0u)
|
|
|
|
{
|
|
|
|
for (uint32_t instIdx = block.start; instIdx <= block.finish; instIdx++)
|
|
|
|
{
|
2023-03-24 17:34:14 +00:00
|
|
|
const IrInst& inst = function.instructions[instIdx];
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
// Skip pseudo instructions unless they are still referenced
|
|
|
|
if (isPseudo(inst.cmd) && inst.useCount == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
append(ctx.result, "|");
|
|
|
|
toString(ctx, inst, instIdx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
appendLabelRegset(ctx.cfg.def, i, "def");
|
|
|
|
appendLabelRegset(ctx.cfg.out, i, "out");
|
|
|
|
|
|
|
|
append(ctx.result, "}\"];\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < function.blocks.size(); i++)
|
|
|
|
{
|
2023-03-24 17:34:14 +00:00
|
|
|
const IrBlock& block = function.blocks[i];
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
if (block.start == ~0u)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (uint32_t instIdx = block.start; instIdx != ~0u && instIdx <= block.finish; instIdx++)
|
|
|
|
{
|
2023-03-24 17:34:14 +00:00
|
|
|
const IrInst& inst = function.instructions[instIdx];
|
2023-03-17 14:59:30 +00:00
|
|
|
|
|
|
|
auto checkOp = [&](IrOp op) {
|
|
|
|
if (op.kind == IrOpKind::Block)
|
|
|
|
{
|
|
|
|
if (function.blocks[op.index].kind != IrBlockKind::Fallback)
|
|
|
|
append(ctx.result, "b%u -> b%u [weight=10];\n", unsigned(i), op.index);
|
|
|
|
else
|
|
|
|
append(ctx.result, "b%u -> b%u;\n", unsigned(i), op.index);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
checkOp(inst.a);
|
|
|
|
checkOp(inst.b);
|
|
|
|
checkOp(inst.c);
|
|
|
|
checkOp(inst.d);
|
|
|
|
checkOp(inst.e);
|
|
|
|
checkOp(inst.f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
append(ctx.result, "}\n");
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-03-24 17:34:14 +00:00
|
|
|
std::string dumpDot(const IrFunction& function, bool includeInst)
|
2023-03-17 14:59:30 +00:00
|
|
|
{
|
|
|
|
std::string result = toDot(function, includeInst);
|
|
|
|
|
|
|
|
printf("%s\n", result.c_str());
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-01-13 20:36:28 +00:00
|
|
|
} // namespace CodeGen
|
|
|
|
} // namespace Luau
|