luau/CodeGen/src/IrDump.cpp

990 lines
27 KiB
C++
Raw Normal View History

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";
2023-10-27 20:33:36 +01:00
case LUA_TBUFFER:
return "tbuffer";
2023-07-28 12:37:00 +01:00
case LUA_TPROTO:
return "tproto";
case LUA_TUPVAL:
return "tupval";
case LUA_TDEADKEY:
return "tdeadkey";
2023-01-13 20:36:28 +00:00
default:
2024-02-16 01:25:31 +00:00
CODEGEN_ASSERT(!"Unknown type tag");
2023-01-13 20:36:28 +00:00
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";
2024-01-27 02:30:40 +00:00
case IrCmd::LOAD_FLOAT:
return "LOAD_FLOAT";
2023-01-13 20:36:28 +00:00
case IrCmd::LOAD_TVALUE:
return "LOAD_TVALUE";
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-07-28 12:37:00 +01:00
case IrCmd::GET_CLOSURE_UPVAL_ADDR:
return "GET_CLOSURE_UPVAL_ADDR";
2023-01-13 20:36:28 +00:00
case IrCmd::STORE_TAG:
return "STORE_TAG";
2024-01-12 19:16:39 +00:00
case IrCmd::STORE_EXTRA:
return "STORE_EXTRA";
2023-01-13 20:36:28 +00:00
case IrCmd::STORE_POINTER:
return "STORE_POINTER";
case IrCmd::STORE_DOUBLE:
return "STORE_DOUBLE";
case IrCmd::STORE_INT:
return "STORE_INT";
2023-04-07 20:56:27 +01:00
case IrCmd::STORE_VECTOR:
return "STORE_VECTOR";
2023-01-13 20:36:28 +00:00
case IrCmd::STORE_TVALUE:
return "STORE_TVALUE";
2023-08-18 18:06:29 +01:00
case IrCmd::STORE_SPLIT_TVALUE:
return "STORE_SPLIT_TVALUE";
2023-01-13 20:36:28 +00:00
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";
2023-09-01 17:38:53 +01:00
case IrCmd::IDIV_NUM:
return "IDIV_NUM";
2023-01-13 20:36:28 +00:00
case IrCmd::MOD_NUM:
return "MOD_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";
2024-01-27 02:30:40 +00:00
case IrCmd::ADD_VEC:
return "ADD_VEC";
case IrCmd::SUB_VEC:
return "SUB_VEC";
case IrCmd::MUL_VEC:
return "MUL_VEC";
case IrCmd::DIV_VEC:
return "DIV_VEC";
case IrCmd::UNM_VEC:
return "UNM_VEC";
2023-01-13 20:36:28 +00:00
case IrCmd::NOT_ANY:
return "NOT_ANY";
case IrCmd::CMP_ANY:
return "CMP_ANY";
2023-01-13 20:36:28 +00:00
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-09-08 00:24:03 +01:00
case IrCmd::JUMP_CMP_INT:
return "JUMP_CMP_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";
2023-10-20 21:36:26 +01:00
case IrCmd::JUMP_FORN_LOOP_COND:
return "JUMP_FORN_LOOP_COND";
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";
2023-09-01 17:38:53 +01:00
case IrCmd::TABLE_SETNUM:
return "TABLE_SETNUM";
2023-07-07 18:14:35 +01:00
case IrCmd::STRING_LEN:
return "STRING_LEN";
2023-01-13 20:36:28 +00:00
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-04-21 22:41:03 +01:00
case IrCmd::UINT_TO_NUM:
return "UINT_TO_NUM";
case IrCmd::NUM_TO_INT:
return "NUM_TO_INT";
case IrCmd::NUM_TO_UINT:
return "NUM_TO_UINT";
2024-03-01 13:58:44 +00:00
case IrCmd::NUM_TO_VEC:
return "NUM_TO_VEC";
case IrCmd::TAG_VECTOR:
return "TAG_VECTOR";
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";
case IrCmd::CHECK_TAG:
return "CHECK_TAG";
case IrCmd::CHECK_TRUTHY:
return "CHECK_TRUTHY";
2023-01-13 20:36:28 +00:00
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-09-01 17:38:53 +01:00
case IrCmd::CHECK_NODE_VALUE:
return "CHECK_NODE_VALUE";
2023-11-10 18:05:48 +00:00
case IrCmd::CHECK_BUFFER_LEN:
return "CHECK_BUFFER_LEN";
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::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";
2023-07-28 12:37:00 +01:00
case IrCmd::NEWCLOSURE:
return "NEWCLOSURE";
2023-01-13 20:36:28 +00:00
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-04-21 22:41:03 +01:00
case IrCmd::BITAND_UINT:
return "BITAND_UINT";
case IrCmd::BITXOR_UINT:
return "BITXOR_UINT";
case IrCmd::BITOR_UINT:
return "BITOR_UINT";
case IrCmd::BITNOT_UINT:
return "BITNOT_UINT";
case IrCmd::BITLSHIFT_UINT:
return "BITLSHIFT_UINT";
case IrCmd::BITRSHIFT_UINT:
return "BITRSHIFT_UINT";
case IrCmd::BITARSHIFT_UINT:
return "BITARSHIFT_UINT";
case IrCmd::BITLROTATE_UINT:
return "BITLROTATE_UINT";
case IrCmd::BITRROTATE_UINT:
return "BITRROTATE_UINT";
case IrCmd::BITCOUNTLZ_UINT:
return "BITCOUNTLZ_UINT";
case IrCmd::BITCOUNTRZ_UINT:
return "BITCOUNTRZ_UINT";
2023-11-03 19:47:28 +00:00
case IrCmd::BYTESWAP_UINT:
return "BYTESWAP_UINT";
2023-04-21 22:41:03 +01:00
case IrCmd::INVOKE_LIBM:
return "INVOKE_LIBM";
2023-06-24 06:33:44 +01:00
case IrCmd::GET_TYPE:
return "GET_TYPE";
case IrCmd::GET_TYPEOF:
return "GET_TYPEOF";
2023-07-28 12:37:00 +01:00
case IrCmd::FINDUPVAL:
return "FINDUPVAL";
2023-11-10 18:05:48 +00:00
case IrCmd::BUFFER_READI8:
return "BUFFER_READI8";
case IrCmd::BUFFER_READU8:
return "BUFFER_READU8";
case IrCmd::BUFFER_WRITEI8:
return "BUFFER_WRITEI8";
case IrCmd::BUFFER_READI16:
return "BUFFER_READI16";
case IrCmd::BUFFER_READU16:
return "BUFFER_READU16";
case IrCmd::BUFFER_WRITEI16:
return "BUFFER_WRITEI16";
case IrCmd::BUFFER_READI32:
return "BUFFER_READI32";
case IrCmd::BUFFER_WRITEI32:
return "BUFFER_WRITEI32";
case IrCmd::BUFFER_READF32:
return "BUFFER_READF32";
case IrCmd::BUFFER_WRITEF32:
return "BUFFER_WRITEF32";
case IrCmd::BUFFER_READF64:
return "BUFFER_READF64";
case IrCmd::BUFFER_WRITEF64:
return "BUFFER_WRITEF64";
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;
2023-04-28 12:55:55 +01:00
case IrOpKind::Undef:
append(ctx.result, "undef");
break;
2023-01-13 20:36:28 +00:00
case IrOpKind::Constant:
toString(ctx.result, ctx.constants[op.index]);
break;
case IrOpKind::Condition:
2024-02-16 01:25:31 +00:00
CODEGEN_ASSERT(op.index < uint32_t(IrCondition::Count));
2023-01-13 20:36:28 +00:00
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:
2023-04-07 20:56:27 +01:00
append(ctx.result, "R%d", vmRegOp(op));
2023-01-13 20:36:28 +00:00
break;
case IrOpKind::VmConst:
2023-04-07 20:56:27 +01:00
append(ctx.result, "K%d", vmConstOp(op));
2023-01-13 20:36:28 +00:00
break;
case IrOpKind::VmUpvalue:
2023-04-07 20:56:27 +01:00
append(ctx.result, "U%d", vmUpvalueOp(op));
2023-01-13 20:36:28 +00:00
break;
2023-07-14 16:57:16 +01:00
case IrOpKind::VmExit:
if (vmExitOp(op) == kVmExitEntryGuardPc)
append(ctx.result, "exit(entry)");
else
append(ctx.result, "exit(%d)", vmExitOp(op));
2023-07-14 16:57:16 +01:00
break;
2023-01-13 20:36:28 +00:00
}
}
void toString(std::string& result, IrConst constant)
{
switch (constant.kind)
{
case IrConstKind::Int:
append(result, "%di", constant.valueInt);
break;
case IrConstKind::Uint:
append(result, "%uu", constant.valueUint);
break;
case IrConstKind::Double:
2023-04-28 12:55:55 +01:00
if (constant.valueDouble != constant.valueDouble)
append(result, "nan");
else
append(result, "%.17g", constant.valueDouble);
2023-01-13 20:36:28 +00:00
break;
case IrConstKind::Tag:
result.append(getTagName(constant.valueTag));
break;
}
}
const char* getBytecodeTypeName(uint8_t type)
{
switch (type)
{
case LBC_TYPE_NIL:
return "nil";
case LBC_TYPE_BOOLEAN:
return "boolean";
case LBC_TYPE_NUMBER:
return "number";
case LBC_TYPE_STRING:
return "string";
case LBC_TYPE_TABLE:
return "table";
case LBC_TYPE_FUNCTION:
return "function";
case LBC_TYPE_THREAD:
return "thread";
case LBC_TYPE_USERDATA:
return "userdata";
case LBC_TYPE_VECTOR:
return "vector";
case LBC_TYPE_BUFFER:
return "buffer";
case LBC_TYPE_ANY:
return "any";
}
2024-02-16 01:25:31 +00:00
CODEGEN_ASSERT(!"Unhandled type in getBytecodeTypeName");
return nullptr;
}
void toString(std::string& result, const BytecodeTypes& bcTypes)
{
if (bcTypes.c != LBC_TYPE_ANY)
append(result, "%s <- %s, %s, %s", getBytecodeTypeName(bcTypes.result), getBytecodeTypeName(bcTypes.a), getBytecodeTypeName(bcTypes.b),
getBytecodeTypeName(bcTypes.c));
else
append(result, "%s <- %s, %s", getBytecodeTypeName(bcTypes.result), getBytecodeTypeName(bcTypes.a), getBytecodeTypeName(bcTypes.b));
}
2023-03-10 19:20:04 +00:00
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-05-05 20:57:12 +01:00
static RegisterSet getJumpTargetExtraLiveIn(IrToStringContext& ctx, const IrBlock& block, uint32_t blockIdx, const IrInst& inst)
{
RegisterSet extraRs;
if (blockIdx >= ctx.cfg.in.size())
return extraRs;
const RegisterSet& defRs = ctx.cfg.in[blockIdx];
// Find first block argument, for guard instructions (isNonTerminatingJump), that's the first and only one
2024-02-16 01:25:31 +00:00
CODEGEN_ASSERT(isNonTerminatingJump(inst.cmd));
2023-05-05 20:57:12 +01:00
IrOp op = inst.a;
if (inst.b.kind == IrOpKind::Block)
op = inst.b;
else if (inst.c.kind == IrOpKind::Block)
op = inst.c;
else if (inst.d.kind == IrOpKind::Block)
op = inst.d;
else if (inst.e.kind == IrOpKind::Block)
op = inst.e;
else if (inst.f.kind == IrOpKind::Block)
op = inst.f;
if (op.kind == IrOpKind::Block && op.index < ctx.cfg.in.size())
{
const RegisterSet& inRs = ctx.cfg.in[op.index];
extraRs.regs = inRs.regs & ~defRs.regs;
if (inRs.varargSeq)
requireVariadicSequence(extraRs, defRs, inRs.varargStart);
}
return extraRs;
}
2024-01-27 02:30:40 +00:00
void toStringDetailed(
IrToStringContext& ctx, const IrBlock& block, uint32_t blockIdx, const IrInst& inst, uint32_t instIdx, IncludeUseInfo includeUseInfo)
2023-05-05 20:57:12 +01:00
{
size_t start = ctx.result.size();
toString(ctx, inst, instIdx);
2024-01-27 02:30:40 +00:00
if (includeUseInfo == IncludeUseInfo::Yes)
2023-05-05 20:57:12 +01:00
{
padToDetailColumn(ctx.result, start);
if (inst.useCount == 0 && hasSideEffects(inst.cmd))
{
if (isNonTerminatingJump(inst.cmd))
{
RegisterSet extraRs = getJumpTargetExtraLiveIn(ctx, block, blockIdx, inst);
if (extraRs.regs.any() || extraRs.varargSeq)
{
append(ctx.result, "; %%%u, extra in: ", instIdx);
appendRegisterSet(ctx, extraRs, ", ");
ctx.result.append("\n");
}
else
{
append(ctx.result, "; %%%u\n", instIdx);
}
}
else
{
append(ctx.result, "; %%%u\n", instIdx);
}
}
else
{
append(ctx.result, "; useCount: %d, lastUse: %%%u\n", inst.useCount, inst.lastUse);
}
}
else
{
ctx.result.append("\n");
}
}
2024-01-27 02:30:40 +00:00
void toStringDetailed(IrToStringContext& ctx, const IrBlock& block, uint32_t blockIdx, IncludeUseInfo includeUseInfo, IncludeCfgInfo includeCfgInfo,
IncludeRegFlowInfo includeRegFlowInfo)
2023-02-10 18:50:54 +00:00
{
2023-03-10 19:20:04 +00:00
// Report captured registers for entry block
2024-01-27 02:30:40 +00:00
if (includeRegFlowInfo == IncludeRegFlowInfo::Yes && block.useCount == 0 && block.kind != IrBlockKind::Dead && ctx.cfg.captured.regs.any())
2023-03-10 19:20:04 +00:00
{
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();
2024-01-19 15:13:08 +00:00
toString(ctx, block, blockIdx);
2023-03-10 19:20:04 +00:00
append(ctx.result, ":");
2024-01-27 02:30:40 +00:00
if (includeUseInfo == IncludeUseInfo::Yes)
2023-03-10 19:20:04 +00:00
{
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
2024-01-27 02:30:40 +00:00
if (includeCfgInfo == IncludeCfgInfo::Yes && blockIdx < ctx.cfg.predecessorsOffsets.size())
2023-03-10 19:20:04 +00:00
{
2024-01-19 15:13:08 +00:00
BlockIteratorWrapper pred = predecessors(ctx.cfg, blockIdx);
2023-03-10 19:20:04 +00:00
if (!pred.empty())
{
append(ctx.result, "; predecessors: ");
appendBlockSet(ctx, pred);
append(ctx.result, "\n");
}
}
// Successor list
2024-01-27 02:30:40 +00:00
if (includeCfgInfo == IncludeCfgInfo::Yes && blockIdx < ctx.cfg.successorsOffsets.size())
2023-03-10 19:20:04 +00:00
{
2024-01-19 15:13:08 +00:00
BlockIteratorWrapper succ = successors(ctx.cfg, blockIdx);
2023-03-10 19:20:04 +00:00
if (!succ.empty())
{
append(ctx.result, "; successors: ");
appendBlockSet(ctx, succ);
append(ctx.result, "\n");
}
}
// Live-in VM regs
2024-01-27 02:30:40 +00:00
if (includeRegFlowInfo == IncludeRegFlowInfo::Yes && blockIdx < ctx.cfg.in.size())
2023-03-10 19:20:04 +00:00
{
2024-01-19 15:13:08 +00:00
const RegisterSet& in = ctx.cfg.in[blockIdx];
2023-03-10 19:20:04 +00:00
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
2024-01-27 02:30:40 +00:00
if (includeRegFlowInfo == IncludeRegFlowInfo::Yes && blockIdx < ctx.cfg.out.size())
2023-03-10 19:20:04 +00:00
{
2024-01-19 15:13:08 +00:00
const RegisterSet& out = ctx.cfg.out[blockIdx];
2023-03-10 19:20:04 +00:00
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
}
2024-01-27 02:30:40 +00:00
std::string toString(const IrFunction& function, IncludeUseInfo 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;
2024-01-27 02:30:40 +00:00
toStringDetailed(ctx, block, uint32_t(i), includeUseInfo, IncludeCfgInfo::Yes, IncludeRegFlowInfo::Yes);
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-05-05 20:57:12 +01:00
toStringDetailed(ctx, block, uint32_t(i), 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
{
2024-01-27 02:30:40 +00:00
std::string result = toString(function, IncludeUseInfo::Yes);
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-06-16 18:01:18 +01:00
static void appendLabelRegset(IrToStringContext& ctx, const std::vector<RegisterSet>& regSets, size_t blockIdx, const char* name)
2023-03-17 14:59:30 +00:00
{
2023-06-16 18:01:18 +01:00
if (blockIdx < regSets.size())
{
const RegisterSet& rs = regSets[blockIdx];
2023-03-17 14:59:30 +00:00
2023-06-16 18:01:18 +01:00
if (rs.regs.any() || rs.varargSeq)
2023-03-17 14:59:30 +00:00
{
2023-06-16 18:01:18 +01:00
append(ctx.result, "|{%s|", name);
appendRegisterSet(ctx, rs, "|");
append(ctx.result, "}");
2023-03-17 14:59:30 +00:00
}
2023-06-16 18:01:18 +01:00
}
}
2023-03-17 14:59:30 +00:00
2023-06-16 18:01:18 +01:00
static void appendBlocks(IrToStringContext& ctx, const IrFunction& function, bool includeInst, bool includeIn, bool includeOut, bool includeDef)
{
2023-03-17 14:59:30 +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-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));
2023-06-16 18:01:18 +01:00
if (includeIn)
appendLabelRegset(ctx, ctx.cfg.in, i, "in");
2023-03-17 14:59:30 +00:00
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);
}
}
2023-06-16 18:01:18 +01:00
if (includeDef)
appendLabelRegset(ctx, ctx.cfg.def, i, "def");
if (includeOut)
appendLabelRegset(ctx, ctx.cfg.out, i, "out");
2023-03-17 14:59:30 +00:00
append(ctx.result, "}\"];\n");
}
2023-06-16 18:01:18 +01:00
}
std::string toDot(const IrFunction& function, bool includeInst)
{
std::string result;
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
append(ctx.result, "digraph CFG {\n");
append(ctx.result, "node[shape=record]\n");
appendBlocks(ctx, function, includeInst, /* includeIn */ true, /* includeOut */ true, /* includeDef */ true);
2023-03-17 14:59:30 +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-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-06-16 18:01:18 +01:00
std::string toDotCfg(const IrFunction& function)
{
std::string result;
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
append(ctx.result, "digraph CFG {\n");
append(ctx.result, "node[shape=record]\n");
appendBlocks(ctx, function, /* includeInst */ false, /* includeIn */ false, /* includeOut */ false, /* includeDef */ true);
for (size_t i = 0; i < function.blocks.size() && i < ctx.cfg.successorsOffsets.size(); i++)
{
BlockIteratorWrapper succ = successors(ctx.cfg, unsigned(i));
for (uint32_t target : succ)
append(ctx.result, "b%u -> b%u;\n", unsigned(i), target);
}
append(ctx.result, "}\n");
return result;
}
std::string toDotDjGraph(const IrFunction& function)
{
std::string result;
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
append(ctx.result, "digraph CFG {\n");
for (size_t i = 0; i < ctx.blocks.size(); i++)
{
const IrBlock& block = ctx.blocks[i];
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));
append(ctx.result, "\"];\n");
}
// Layer by depth in tree
uint32_t depth = 0;
bool found = true;
while (found)
{
found = false;
append(ctx.result, "{rank = same;");
for (size_t i = 0; i < ctx.cfg.domOrdering.size(); i++)
{
if (ctx.cfg.domOrdering[i].depth == depth)
{
append(ctx.result, "b%u;", unsigned(i));
found = true;
}
}
append(ctx.result, "}\n");
depth++;
}
for (size_t i = 0; i < ctx.cfg.domChildrenOffsets.size(); i++)
{
BlockIteratorWrapper dom = domChildren(ctx.cfg, unsigned(i));
for (uint32_t target : dom)
append(ctx.result, "b%u -> b%u;\n", unsigned(i), target);
// Join edges are all successor edges that do not strongly dominate
BlockIteratorWrapper succ = successors(ctx.cfg, unsigned(i));
for (uint32_t successor : succ)
{
bool found = false;
for (uint32_t target : dom)
{
if (target == successor)
{
found = true;
break;
}
}
if (!found)
append(ctx.result, "b%u -> b%u [style=dotted];\n", unsigned(i), successor);
}
}
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