luau/CodeGen/include/Luau/CodeGen.h

168 lines
4.8 KiB
C
Raw Normal View History

2022-10-13 23:59:53 +01: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-20 21:36:26 +01:00
#include <algorithm>
2022-10-13 23:59:53 +01:00
#include <string>
#include <vector>
2022-10-13 23:59:53 +01:00
2023-09-15 17:27:45 +01:00
#include <stddef.h>
2023-05-12 13:15:01 +01:00
#include <stdint.h>
2022-10-13 23:59:53 +01:00
struct lua_State;
namespace Luau
{
namespace CodeGen
{
2023-07-28 12:37:00 +01:00
enum CodeGenFlags
{
// Only run native codegen for modules that have been marked with --!native
CodeGen_OnlyNativeModules = 1 << 0,
2023-10-06 18:31:16 +01:00
// Run native codegen for functions that the compiler considers not profitable
CodeGen_ColdFunctions = 1 << 1,
2023-07-28 12:37:00 +01:00
};
2023-08-25 16:25:09 +01: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 13:55:30 +01: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-13 23:59:53 +01:00
bool isSupported();
2023-08-11 13:55:30 +01:00
void create(lua_State* L, AllocationCallback* allocationCallback, void* allocationCallbackContext);
2022-10-13 23:59:53 +01:00
void create(lua_State* L);
// Builds target function and all inner functions
2023-08-25 16:25:09 +01:00
CodeGenCompilationResult compile(lua_State* L, int idx, unsigned int flags = 0, CompilationStats* stats = nullptr);
2022-10-13 23:59:53 +01:00
2023-05-12 13:15:01 +01:00
using AnnotatorFn = void (*)(void* context, std::string& result, int fid, int instpos);
2022-10-21 18:33:43 +01:00
struct AssemblyOptions
{
2023-06-24 06:33:44 +01:00
enum Target
{
Host,
A64,
A64_NoFeatures,
X64_Windows,
X64_SystemV,
};
Target target = Host;
2023-10-20 21:36:26 +01:00
unsigned int flags = 0;
2022-10-21 18:33:43 +01:00
bool outputBinary = false;
2023-01-27 21:28:45 +00:00
bool includeAssembly = false;
bool includeIr = false;
bool includeOutlinedCode = false;
2022-10-21 18:33:43 +01:00
// Optional annotator function can be provided to describe each instruction, it takes function id and sequential instruction id
2023-05-12 13:15:01 +01:00
AnnotatorFn annotator = nullptr;
2022-10-21 18:33:43 +01:00
void* annotatorContext = nullptr;
};
2023-11-03 19:47:28 +00: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;
}
};
struct FunctionStats
{
std::string name;
int line = -1;
unsigned bcodeCount = 0;
unsigned irCount = 0;
unsigned asmCount = 0;
};
2023-09-15 17:27:45 +01:00
struct LoweringStats
{
2023-10-20 21:36:26 +01:00
unsigned totalFunctions = 0;
unsigned skippedFunctions = 0;
2023-09-15 17:27:45 +01:00
int spillsToSlot = 0;
int spillsToRestore = 0;
unsigned maxSpillSlotsUsed = 0;
2023-10-20 21:36:26 +01:00
unsigned blocksPreOpt = 0;
unsigned blocksPostOpt = 0;
unsigned maxBlockInstructions = 0;
2023-09-15 17:27:45 +01:00
int regAllocErrors = 0;
int loweringErrors = 0;
2023-10-20 21:36:26 +01:00
2023-11-03 19:47:28 +00:00
BlockLinearizationStats blockLinearizationStats;
bool collectFunctionStats = false;
std::vector<FunctionStats> functions;
2023-10-20 21:36:26 +01: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-03 19:47:28 +00:00
this->blockLinearizationStats += that.blockLinearizationStats;
if (this->collectFunctionStats)
this->functions.insert(this->functions.end(), that.functions.begin(), that.functions.end());
2023-10-20 21:36:26 +01:00
return *this;
}
2023-09-15 17:27:45 +01:00
};
2022-10-21 18:33:43 +01:00
// Generates assembly for target function and all inner functions
2023-09-15 17:27:45 +01:00
std::string getAssembly(lua_State* L, int idx, AssemblyOptions options = {}, LoweringStats* stats = nullptr);
2022-10-13 23:59:53 +01:00
2023-05-12 13:15:01 +01:00
using PerfLogFn = void (*)(void* context, uintptr_t addr, unsigned size, const char* symbol);
void setPerfLog(void* context, PerfLogFn logFn);
2022-10-13 23:59:53 +01:00
} // namespace CodeGen
} // namespace Luau