2023-11-17 18:46:18 +00:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/BytecodeSummary.h"
|
2024-01-12 22:25:27 +00:00
|
|
|
#include "Luau/BytecodeUtils.h"
|
2023-11-17 18:46:18 +00:00
|
|
|
#include "CodeGenLower.h"
|
|
|
|
|
|
|
|
#include "lua.h"
|
|
|
|
#include "lapi.h"
|
|
|
|
#include "lobject.h"
|
|
|
|
#include "lstate.h"
|
|
|
|
|
2024-06-14 21:21:20 +01:00
|
|
|
LUAU_FASTFLAG(LuauNativeAttribute)
|
|
|
|
|
2023-11-17 18:46:18 +00:00
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace CodeGen
|
|
|
|
{
|
|
|
|
|
|
|
|
FunctionBytecodeSummary::FunctionBytecodeSummary(std::string source, std::string name, const int line, unsigned nestingLimit)
|
|
|
|
: source(std::move(source))
|
|
|
|
, name(std::move(name))
|
|
|
|
, line(line)
|
|
|
|
, nestingLimit(nestingLimit)
|
|
|
|
{
|
|
|
|
counts.reserve(nestingLimit);
|
|
|
|
for (unsigned i = 0; i < 1 + nestingLimit; ++i)
|
|
|
|
{
|
|
|
|
counts.push_back(std::vector<unsigned>(getOpLimit(), 0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FunctionBytecodeSummary FunctionBytecodeSummary::fromProto(Proto* proto, unsigned nestingLimit)
|
|
|
|
{
|
|
|
|
const char* source = getstr(proto->source);
|
|
|
|
source = (source[0] == '=' || source[0] == '@') ? source + 1 : "[string]";
|
|
|
|
|
|
|
|
const char* name = proto->debugname ? getstr(proto->debugname) : "";
|
|
|
|
|
|
|
|
int line = proto->linedefined;
|
|
|
|
|
|
|
|
FunctionBytecodeSummary summary(source, name, line, nestingLimit);
|
|
|
|
|
2024-01-12 22:25:27 +00:00
|
|
|
for (int i = 0; i < proto->sizecode;)
|
2023-11-17 18:46:18 +00:00
|
|
|
{
|
|
|
|
Instruction insn = proto->code[i];
|
|
|
|
uint8_t op = LUAU_INSN_OP(insn);
|
|
|
|
summary.incCount(0, op);
|
2024-01-12 22:25:27 +00:00
|
|
|
i += Luau::getOpLength(LuauOpcode(op));
|
2023-11-17 18:46:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return summary;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<FunctionBytecodeSummary> summarizeBytecode(lua_State* L, int idx, unsigned nestingLimit)
|
|
|
|
{
|
2024-02-16 02:04:39 +00:00
|
|
|
CODEGEN_ASSERT(lua_isLfunction(L, idx));
|
2023-11-17 18:46:18 +00:00
|
|
|
const TValue* func = luaA_toobject(L, idx);
|
|
|
|
|
|
|
|
Proto* root = clvalue(func)->l.p;
|
|
|
|
|
|
|
|
std::vector<Proto*> protos;
|
2024-06-14 21:21:20 +01:00
|
|
|
if (FFlag::LuauNativeAttribute)
|
|
|
|
gatherFunctions(protos, root, CodeGen_ColdFunctions, root->flags & LPF_NATIVE_FUNCTION);
|
|
|
|
else
|
|
|
|
gatherFunctions_DEPRECATED(protos, root, CodeGen_ColdFunctions);
|
2023-11-17 18:46:18 +00:00
|
|
|
|
|
|
|
std::vector<FunctionBytecodeSummary> summaries;
|
|
|
|
summaries.reserve(protos.size());
|
|
|
|
|
|
|
|
for (Proto* proto : protos)
|
|
|
|
{
|
2024-01-12 22:25:27 +00:00
|
|
|
if (proto)
|
|
|
|
summaries.push_back(FunctionBytecodeSummary::fromProto(proto, nestingLimit));
|
2023-11-17 18:46:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return summaries;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace CodeGen
|
|
|
|
} // namespace Luau
|