2022-10-15 03:48:41 +08:00
|
|
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#pragma once
|
|
|
|
|
2023-10-21 09:10:30 +08:00
|
|
|
#include <algorithm>
|
2022-10-15 03:48:41 +08:00
|
|
|
#include <string>
|
2023-12-02 15:46:57 +08:00
|
|
|
#include <vector>
|
2022-10-15 03:48:41 +08:00
|
|
|
|
2023-09-16 01:26:59 +08:00
|
|
|
#include <stddef.h>
|
2023-05-13 01:50:47 +08:00
|
|
|
#include <stdint.h>
|
|
|
|
|
2022-10-15 03:48:41 +08:00
|
|
|
struct lua_State;
|
|
|
|
|
|
|
|
namespace Luau
|
|
|
|
{
|
|
|
|
namespace CodeGen
|
|
|
|
{
|
|
|
|
|
2023-07-28 23:13:53 +08:00
|
|
|
enum CodeGenFlags
|
|
|
|
{
|
|
|
|
// Only run native codegen for modules that have been marked with --!native
|
|
|
|
CodeGen_OnlyNativeModules = 1 << 0,
|
2023-10-07 03:02:32 +08:00
|
|
|
// Run native codegen for functions that the compiler considers not profitable
|
|
|
|
CodeGen_ColdFunctions = 1 << 1,
|
2023-07-28 23:13:53 +08:00
|
|
|
};
|
|
|
|
|
2023-08-26 01:23:55 +08:00
|
|
|
enum class CodeGenCompilationResult
|
|
|
|
{
|
|
|
|
Success, // Successfully generated code for at least one function
|
|
|
|
NothingToCompile, // There were no new functions to compile
|
|
|
|
|
|
|
|
CodeGenNotInitialized, // Native codegen system is not initialized
|
|
|
|
CodeGenFailed, // Native codegen failed due to an internal compiler error
|
|
|
|
AllocationFailed, // Native codegen failed due to an allocation error
|
|
|
|
};
|
|
|
|
|
2023-08-11 22:42:37 +08:00
|
|
|
struct CompilationStats
|
|
|
|
{
|
|
|
|
size_t bytecodeSizeBytes = 0;
|
|
|
|
size_t nativeCodeSizeBytes = 0;
|
|
|
|
size_t nativeDataSizeBytes = 0;
|
|
|
|
size_t nativeMetadataSizeBytes = 0;
|
|
|
|
|
|
|
|
uint32_t functionsCompiled = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
using AllocationCallback = void(void* context, void* oldPointer, size_t oldSize, void* newPointer, size_t newSize);
|
|
|
|
|
2022-10-15 03:48:41 +08:00
|
|
|
bool isSupported();
|
|
|
|
|
2023-08-11 22:42:37 +08:00
|
|
|
void create(lua_State* L, AllocationCallback* allocationCallback, void* allocationCallbackContext);
|
2022-10-15 03:48:41 +08:00
|
|
|
void create(lua_State* L);
|
|
|
|
|
|
|
|
// Builds target function and all inner functions
|
2023-08-26 01:23:55 +08:00
|
|
|
CodeGenCompilationResult compile(lua_State* L, int idx, unsigned int flags = 0, CompilationStats* stats = nullptr);
|
2022-10-15 03:48:41 +08:00
|
|
|
|
2023-05-13 01:50:47 +08:00
|
|
|
using AnnotatorFn = void (*)(void* context, std::string& result, int fid, int instpos);
|
2022-10-22 01:54:01 +08:00
|
|
|
|
|
|
|
struct AssemblyOptions
|
|
|
|
{
|
2023-06-24 14:19:39 +08:00
|
|
|
enum Target
|
|
|
|
{
|
|
|
|
Host,
|
|
|
|
A64,
|
|
|
|
A64_NoFeatures,
|
|
|
|
X64_Windows,
|
|
|
|
X64_SystemV,
|
|
|
|
};
|
|
|
|
|
|
|
|
Target target = Host;
|
|
|
|
|
2023-10-21 09:10:30 +08:00
|
|
|
unsigned int flags = 0;
|
|
|
|
|
2022-10-22 01:54:01 +08:00
|
|
|
bool outputBinary = false;
|
2023-01-28 06:28:31 +08:00
|
|
|
|
|
|
|
bool includeAssembly = false;
|
|
|
|
bool includeIr = false;
|
|
|
|
bool includeOutlinedCode = false;
|
2022-10-22 01:54:01 +08:00
|
|
|
|
|
|
|
// Optional annotator function can be provided to describe each instruction, it takes function id and sequential instruction id
|
2023-05-13 01:50:47 +08:00
|
|
|
AnnotatorFn annotator = nullptr;
|
2022-10-22 01:54:01 +08:00
|
|
|
void* annotatorContext = nullptr;
|
|
|
|
};
|
|
|
|
|
2023-11-04 07:45:04 +08:00
|
|
|
struct BlockLinearizationStats
|
|
|
|
{
|
|
|
|
unsigned int constPropInstructionCount = 0;
|
|
|
|
double timeSeconds = 0.0;
|
|
|
|
|
|
|
|
BlockLinearizationStats& operator+=(const BlockLinearizationStats& that)
|
|
|
|
{
|
|
|
|
this->constPropInstructionCount += that.constPropInstructionCount;
|
|
|
|
this->timeSeconds += that.timeSeconds;
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
BlockLinearizationStats operator+(const BlockLinearizationStats& other) const
|
|
|
|
{
|
|
|
|
BlockLinearizationStats result(*this);
|
|
|
|
result += other;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-01-13 06:25:27 +08:00
|
|
|
enum FunctionStatsFlags
|
|
|
|
{
|
|
|
|
// Enable stats collection per function
|
|
|
|
FunctionStats_Enable = 1 << 0,
|
|
|
|
// Compute function bytecode summary
|
|
|
|
FunctionStats_BytecodeSummary = 1 << 1,
|
|
|
|
};
|
|
|
|
|
2023-12-02 15:46:57 +08:00
|
|
|
struct FunctionStats
|
|
|
|
{
|
|
|
|
std::string name;
|
|
|
|
int line = -1;
|
|
|
|
unsigned bcodeCount = 0;
|
|
|
|
unsigned irCount = 0;
|
|
|
|
unsigned asmCount = 0;
|
2024-01-13 06:25:27 +08:00
|
|
|
unsigned asmSize = 0;
|
|
|
|
std::vector<std::vector<unsigned>> bytecodeSummary;
|
2023-12-02 15:46:57 +08:00
|
|
|
};
|
|
|
|
|
2023-09-16 01:26:59 +08:00
|
|
|
struct LoweringStats
|
|
|
|
{
|
2023-10-21 09:10:30 +08:00
|
|
|
unsigned totalFunctions = 0;
|
|
|
|
unsigned skippedFunctions = 0;
|
2023-09-16 01:26:59 +08:00
|
|
|
int spillsToSlot = 0;
|
|
|
|
int spillsToRestore = 0;
|
|
|
|
unsigned maxSpillSlotsUsed = 0;
|
2023-10-21 09:10:30 +08:00
|
|
|
unsigned blocksPreOpt = 0;
|
|
|
|
unsigned blocksPostOpt = 0;
|
|
|
|
unsigned maxBlockInstructions = 0;
|
2023-09-16 01:26:59 +08:00
|
|
|
|
|
|
|
int regAllocErrors = 0;
|
|
|
|
int loweringErrors = 0;
|
2023-10-21 09:10:30 +08:00
|
|
|
|
2023-11-04 07:45:04 +08:00
|
|
|
BlockLinearizationStats blockLinearizationStats;
|
|
|
|
|
2024-01-13 06:25:27 +08:00
|
|
|
unsigned functionStatsFlags = 0;
|
2023-12-02 15:46:57 +08:00
|
|
|
std::vector<FunctionStats> functions;
|
|
|
|
|
2023-10-21 09:10:30 +08:00
|
|
|
LoweringStats operator+(const LoweringStats& other) const
|
|
|
|
{
|
|
|
|
LoweringStats result(*this);
|
|
|
|
result += other;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
LoweringStats& operator+=(const LoweringStats& that)
|
|
|
|
{
|
|
|
|
this->totalFunctions += that.totalFunctions;
|
|
|
|
this->skippedFunctions += that.skippedFunctions;
|
|
|
|
this->spillsToSlot += that.spillsToSlot;
|
|
|
|
this->spillsToRestore += that.spillsToRestore;
|
|
|
|
this->maxSpillSlotsUsed = std::max(this->maxSpillSlotsUsed, that.maxSpillSlotsUsed);
|
|
|
|
this->blocksPreOpt += that.blocksPreOpt;
|
|
|
|
this->blocksPostOpt += that.blocksPostOpt;
|
|
|
|
this->maxBlockInstructions = std::max(this->maxBlockInstructions, that.maxBlockInstructions);
|
|
|
|
this->regAllocErrors += that.regAllocErrors;
|
|
|
|
this->loweringErrors += that.loweringErrors;
|
2023-11-04 07:45:04 +08:00
|
|
|
this->blockLinearizationStats += that.blockLinearizationStats;
|
2024-01-13 06:25:27 +08:00
|
|
|
if (this->functionStatsFlags & FunctionStats_Enable)
|
2023-12-02 15:46:57 +08:00
|
|
|
this->functions.insert(this->functions.end(), that.functions.begin(), that.functions.end());
|
2023-10-21 09:10:30 +08:00
|
|
|
return *this;
|
|
|
|
}
|
2023-09-16 01:26:59 +08:00
|
|
|
};
|
|
|
|
|
2022-10-22 01:54:01 +08:00
|
|
|
// Generates assembly for target function and all inner functions
|
2023-09-16 01:26:59 +08:00
|
|
|
std::string getAssembly(lua_State* L, int idx, AssemblyOptions options = {}, LoweringStats* stats = nullptr);
|
2022-10-15 03:48:41 +08:00
|
|
|
|
2023-05-13 01:50:47 +08:00
|
|
|
using PerfLogFn = void (*)(void* context, uintptr_t addr, unsigned size, const char* symbol);
|
|
|
|
|
|
|
|
void setPerfLog(void* context, PerfLogFn logFn);
|
|
|
|
|
2022-10-15 03:48:41 +08:00
|
|
|
} // namespace CodeGen
|
|
|
|
} // namespace Luau
|