mirror of
https://github.com/luau-lang/luau.git
synced 2025-01-07 11:59:11 +00:00
Sync to upstream/release/654 (#1552)
# What's Changed * Support dead store elimination for `STORE_VECTOR` instruction * Fix parser hang when a separator is used between Luau class declaration properties * Provide properties and metatable for built-in vector type definition to fix type errors * Fix Fragment Autocomplete to ensure correct parentheses insertion behavior. * Add support for 'thread' and 'buffer' primitive types in user-defined type functions --------- Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Hunter Goldstein <hgoldstein@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
This commit is contained in:
parent
8cc289fae4
commit
d0222bb554
27 changed files with 825 additions and 95 deletions
|
@ -194,6 +194,7 @@ struct Frontend
|
||||||
);
|
);
|
||||||
|
|
||||||
std::optional<CheckResult> getCheckResult(const ModuleName& name, bool accumulateNested, bool forAutocomplete = false);
|
std::optional<CheckResult> getCheckResult(const ModuleName& name, bool accumulateNested, bool forAutocomplete = false);
|
||||||
|
std::vector<ModuleName> getRequiredScripts(const ModuleName& name);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ModulePtr check(
|
ModulePtr check(
|
||||||
|
|
|
@ -31,6 +31,8 @@ struct TypeFunctionPrimitiveType
|
||||||
Boolean,
|
Boolean,
|
||||||
Number,
|
Number,
|
||||||
String,
|
String,
|
||||||
|
Thread,
|
||||||
|
Buffer,
|
||||||
};
|
};
|
||||||
|
|
||||||
Type type;
|
Type type;
|
||||||
|
|
|
@ -1701,7 +1701,7 @@ AutocompleteResult autocomplete_(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
TypeArena* typeArena,
|
TypeArena* typeArena,
|
||||||
std::vector<AstNode*>& ancestry,
|
std::vector<AstNode*>& ancestry,
|
||||||
Scope* globalScope,
|
Scope* globalScope, // [TODO] This is unused argument, do we really need this?
|
||||||
const ScopePtr& scopeAtPosition,
|
const ScopePtr& scopeAtPosition,
|
||||||
Position position,
|
Position position,
|
||||||
FileResolver* fileResolver,
|
FileResolver* fileResolver,
|
||||||
|
|
|
@ -31,8 +31,8 @@
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauTypestateBuiltins2)
|
LUAU_FASTFLAGVARIABLE(LuauTypestateBuiltins2)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauStringFormatArityFix)
|
LUAU_FASTFLAGVARIABLE(LuauStringFormatArityFix)
|
||||||
|
LUAU_FASTFLAG(AutocompleteRequirePathSuggestions2)
|
||||||
LUAU_FASTFLAG(AutocompleteRequirePathSuggestions2);
|
LUAU_FASTFLAG(LuauVectorDefinitionsExtra)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -300,6 +300,31 @@ void registerBuiltinGlobals(Frontend& frontend, GlobalTypes& globals, bool typeC
|
||||||
|
|
||||||
addGlobalBinding(globals, "string", it->second.type(), "@luau");
|
addGlobalBinding(globals, "string", it->second.type(), "@luau");
|
||||||
|
|
||||||
|
// Setup 'vector' metatable
|
||||||
|
if (FFlag::LuauVectorDefinitionsExtra)
|
||||||
|
{
|
||||||
|
if (auto it = globals.globalScope->exportedTypeBindings.find("vector"); it != globals.globalScope->exportedTypeBindings.end())
|
||||||
|
{
|
||||||
|
TypeId vectorTy = it->second.type;
|
||||||
|
ClassType* vectorCls = getMutable<ClassType>(vectorTy);
|
||||||
|
|
||||||
|
vectorCls->metatable = arena.addType(TableType{{}, std::nullopt, TypeLevel{}, TableState::Sealed});
|
||||||
|
TableType* metatableTy = Luau::getMutable<TableType>(vectorCls->metatable);
|
||||||
|
|
||||||
|
metatableTy->props["__add"] = {makeFunction(arena, vectorTy, {vectorTy}, {vectorTy})};
|
||||||
|
metatableTy->props["__sub"] = {makeFunction(arena, vectorTy, {vectorTy}, {vectorTy})};
|
||||||
|
metatableTy->props["__unm"] = {makeFunction(arena, vectorTy, {}, {vectorTy})};
|
||||||
|
|
||||||
|
std::initializer_list<TypeId> mulOverloads{
|
||||||
|
makeFunction(arena, vectorTy, {vectorTy}, {vectorTy}),
|
||||||
|
makeFunction(arena, vectorTy, {builtinTypes->numberType}, {vectorTy}),
|
||||||
|
};
|
||||||
|
metatableTy->props["__mul"] = {makeIntersection(arena, mulOverloads)};
|
||||||
|
metatableTy->props["__div"] = {makeIntersection(arena, mulOverloads)};
|
||||||
|
metatableTy->props["__idiv"] = {makeIntersection(arena, mulOverloads)};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// next<K, V>(t: Table<K, V>, i: K?) -> (K?, V)
|
// next<K, V>(t: Table<K, V>, i: K?) -> (K?, V)
|
||||||
TypePackId nextArgsTypePack = arena.addTypePack(TypePack{{mapOfKtoV, makeOption(builtinTypes, arena, genericK)}});
|
TypePackId nextArgsTypePack = arena.addTypePack(TypePack{{mapOfKtoV, makeOption(builtinTypes, arena, genericK)}});
|
||||||
TypePackId nextRetsTypePack = arena.addTypePack(TypePack{{makeOption(builtinTypes, arena, genericK), genericV}});
|
TypePackId nextRetsTypePack = arena.addTypePack(TypePack{{makeOption(builtinTypes, arena, genericK), genericV}});
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
LUAU_FASTFLAG(LuauMathMap)
|
LUAU_FASTFLAG(LuauMathMap)
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitions)
|
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitions)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitionsExtra)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -452,7 +453,7 @@ declare buffer: {
|
||||||
|
|
||||||
)BUILTIN_SRC";
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
static const std::string kBuiltinDefinitionVectorSrc = R"BUILTIN_SRC(
|
static const std::string kBuiltinDefinitionVectorSrc_DEPRECATED = R"BUILTIN_SRC(
|
||||||
|
|
||||||
-- TODO: this will be replaced with a built-in primitive type
|
-- TODO: this will be replaced with a built-in primitive type
|
||||||
declare class vector end
|
declare class vector end
|
||||||
|
@ -478,12 +479,44 @@ declare vector: {
|
||||||
|
|
||||||
)BUILTIN_SRC";
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionVectorSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
-- While vector would have been better represented as a built-in primitive type, type solver class handling covers most of the properties
|
||||||
|
declare class vector
|
||||||
|
x: number
|
||||||
|
y: number
|
||||||
|
z: number
|
||||||
|
end
|
||||||
|
|
||||||
|
declare vector: {
|
||||||
|
create: @checked (x: number, y: number, z: number) -> vector,
|
||||||
|
magnitude: @checked (vec: vector) -> number,
|
||||||
|
normalize: @checked (vec: vector) -> vector,
|
||||||
|
cross: @checked (vec1: vector, vec2: vector) -> vector,
|
||||||
|
dot: @checked (vec1: vector, vec2: vector) -> number,
|
||||||
|
angle: @checked (vec1: vector, vec2: vector, axis: vector?) -> number,
|
||||||
|
floor: @checked (vec: vector) -> vector,
|
||||||
|
ceil: @checked (vec: vector) -> vector,
|
||||||
|
abs: @checked (vec: vector) -> vector,
|
||||||
|
sign: @checked (vec: vector) -> vector,
|
||||||
|
clamp: @checked (vec: vector, min: vector, max: vector) -> vector,
|
||||||
|
max: @checked (vector, ...vector) -> vector,
|
||||||
|
min: @checked (vector, ...vector) -> vector,
|
||||||
|
|
||||||
|
zero: vector,
|
||||||
|
one: vector,
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
std::string getBuiltinDefinitionSource()
|
std::string getBuiltinDefinitionSource()
|
||||||
{
|
{
|
||||||
std::string result = FFlag::LuauMathMap ? kBuiltinDefinitionLuaSrcChecked : kBuiltinDefinitionLuaSrcChecked_DEPRECATED;
|
std::string result = FFlag::LuauMathMap ? kBuiltinDefinitionLuaSrcChecked : kBuiltinDefinitionLuaSrcChecked_DEPRECATED;
|
||||||
|
|
||||||
if (FFlag::LuauVectorDefinitions)
|
if (FFlag::LuauVectorDefinitionsExtra)
|
||||||
result += kBuiltinDefinitionVectorSrc;
|
result += kBuiltinDefinitionVectorSrc;
|
||||||
|
else if (FFlag::LuauVectorDefinitions)
|
||||||
|
result += kBuiltinDefinitionVectorSrc_DEPRECATED;
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
|
@ -121,19 +121,13 @@ FragmentAutocompleteAncestryResult findAncestryForFragmentParse(AstStatBlock* ro
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get document offsets is a function that takes a source text document as well as a start position and end position(line, column) in that
|
* Get document offsets is a function that takes a source text document as well as a start position and end position(line, column) in that
|
||||||
* document and attempts to get the concrete text between those points. It returns a tuple of:
|
* document and attempts to get the concrete text between those points. It returns a pair of:
|
||||||
* - start offset that represents an index in the source `char*` corresponding to startPos
|
* - start offset that represents an index in the source `char*` corresponding to startPos
|
||||||
* - length, that represents how many more bytes to read to get to endPos.
|
* - length, that represents how many more bytes to read to get to endPos.
|
||||||
* - cursorPos, that represents the position of the cursor relative to the start offset.
|
* Example - your document is "foo bar baz" and getDocumentOffsets is passed (0, 4), (0, 8). This function returns the pair {3, 5}
|
||||||
* Example - your document is "foo bar baz" and getDocumentOffsets is passed (0, 4), (0, 7), (0, 8). This function returns the tuple {3, 5,
|
* which corresponds to the string " bar "
|
||||||
* Position{0, 4}}, which corresponds to the string " bar "
|
|
||||||
*/
|
*/
|
||||||
std::tuple<size_t, size_t, Position> getDocumentOffsets(
|
std::pair<size_t, size_t> getDocumentOffsets(const std::string_view& src, const Position& startPos, const Position& endPos)
|
||||||
const std::string_view& src,
|
|
||||||
const Position& startPos,
|
|
||||||
Position cursorPos,
|
|
||||||
const Position& endPos
|
|
||||||
)
|
|
||||||
{
|
{
|
||||||
size_t lineCount = 0;
|
size_t lineCount = 0;
|
||||||
size_t colCount = 0;
|
size_t colCount = 0;
|
||||||
|
@ -142,12 +136,8 @@ std::tuple<size_t, size_t, Position> getDocumentOffsets(
|
||||||
size_t startOffset = 0;
|
size_t startOffset = 0;
|
||||||
size_t endOffset = 0;
|
size_t endOffset = 0;
|
||||||
bool foundStart = false;
|
bool foundStart = false;
|
||||||
bool foundCursor = false;
|
|
||||||
bool foundEnd = false;
|
bool foundEnd = false;
|
||||||
|
|
||||||
unsigned int colOffsetFromStart = 0;
|
|
||||||
unsigned int lineOffsetFromStart = 0;
|
|
||||||
|
|
||||||
for (char c : src)
|
for (char c : src)
|
||||||
{
|
{
|
||||||
if (foundStart && foundEnd)
|
if (foundStart && foundEnd)
|
||||||
|
@ -159,15 +149,11 @@ std::tuple<size_t, size_t, Position> getDocumentOffsets(
|
||||||
startOffset = docOffset;
|
startOffset = docOffset;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cursorPos.line == lineCount && cursorPos.column == colCount)
|
|
||||||
{
|
|
||||||
foundCursor = true;
|
|
||||||
cursorPos = {lineOffsetFromStart, colOffsetFromStart};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (endPos.line == lineCount && endPos.column == colCount)
|
if (endPos.line == lineCount && endPos.column == colCount)
|
||||||
{
|
{
|
||||||
endOffset = docOffset;
|
endOffset = docOffset;
|
||||||
|
while (endOffset < src.size() && src[endOffset] != '\n')
|
||||||
|
endOffset++;
|
||||||
foundEnd = true;
|
foundEnd = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -180,18 +166,11 @@ std::tuple<size_t, size_t, Position> getDocumentOffsets(
|
||||||
|
|
||||||
if (c == '\n')
|
if (c == '\n')
|
||||||
{
|
{
|
||||||
if (foundStart)
|
|
||||||
{
|
|
||||||
lineOffsetFromStart++;
|
|
||||||
colOffsetFromStart = 0;
|
|
||||||
}
|
|
||||||
lineCount++;
|
lineCount++;
|
||||||
colCount = 0;
|
colCount = 0;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (foundStart)
|
|
||||||
colOffsetFromStart++;
|
|
||||||
colCount++;
|
colCount++;
|
||||||
}
|
}
|
||||||
docOffset++;
|
docOffset++;
|
||||||
|
@ -200,12 +179,9 @@ std::tuple<size_t, size_t, Position> getDocumentOffsets(
|
||||||
if (foundStart && !foundEnd)
|
if (foundStart && !foundEnd)
|
||||||
endOffset = src.length();
|
endOffset = src.length();
|
||||||
|
|
||||||
if (foundStart && !foundCursor)
|
|
||||||
cursorPos = {lineOffsetFromStart, colOffsetFromStart};
|
|
||||||
|
|
||||||
size_t min = std::min(startOffset, endOffset);
|
size_t min = std::min(startOffset, endOffset);
|
||||||
size_t len = std::max(startOffset, endOffset) - min;
|
size_t len = std::max(startOffset, endOffset) - min;
|
||||||
return {min, len, cursorPos};
|
return {min, len};
|
||||||
}
|
}
|
||||||
|
|
||||||
ScopePtr findClosestScope(const ModulePtr& module, const AstStat* nearestStatement)
|
ScopePtr findClosestScope(const ModulePtr& module, const AstStat* nearestStatement)
|
||||||
|
@ -232,10 +208,6 @@ FragmentParseResult parseFragment(
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
FragmentAutocompleteAncestryResult result = findAncestryForFragmentParse(srcModule.root, cursorPos);
|
FragmentAutocompleteAncestryResult result = findAncestryForFragmentParse(srcModule.root, cursorPos);
|
||||||
ParseOptions opts;
|
|
||||||
opts.allowDeclarationSyntax = false;
|
|
||||||
opts.captureComments = true;
|
|
||||||
opts.parseFragment = FragmentParseResumeSettings{std::move(result.localMap), std::move(result.localStack)};
|
|
||||||
AstStat* nearestStatement = result.nearestStatement;
|
AstStat* nearestStatement = result.nearestStatement;
|
||||||
|
|
||||||
const Location& rootSpan = srcModule.root->location;
|
const Location& rootSpan = srcModule.root->location;
|
||||||
|
@ -260,15 +232,18 @@ FragmentParseResult parseFragment(
|
||||||
else
|
else
|
||||||
startPos = nearestStatement->location.begin;
|
startPos = nearestStatement->location.begin;
|
||||||
|
|
||||||
auto [offsetStart, parseLength, cursorInFragment] = getDocumentOffsets(src, startPos, cursorPos, endPos);
|
auto [offsetStart, parseLength] = getDocumentOffsets(src, startPos, endPos);
|
||||||
|
|
||||||
|
|
||||||
const char* srcStart = src.data() + offsetStart;
|
const char* srcStart = src.data() + offsetStart;
|
||||||
std::string_view dbg = src.substr(offsetStart, parseLength);
|
std::string_view dbg = src.substr(offsetStart, parseLength);
|
||||||
const std::shared_ptr<AstNameTable>& nameTbl = srcModule.names;
|
const std::shared_ptr<AstNameTable>& nameTbl = srcModule.names;
|
||||||
FragmentParseResult fragmentResult;
|
FragmentParseResult fragmentResult;
|
||||||
fragmentResult.fragmentToParse = std::string(dbg.data(), parseLength);
|
fragmentResult.fragmentToParse = std::string(dbg.data(), parseLength);
|
||||||
// For the duration of the incremental parse, we want to allow the name table to re-use duplicate names
|
// For the duration of the incremental parse, we want to allow the name table to re-use duplicate names
|
||||||
|
|
||||||
|
ParseOptions opts;
|
||||||
|
opts.allowDeclarationSyntax = false;
|
||||||
|
opts.captureComments = true;
|
||||||
|
opts.parseFragment = FragmentParseResumeSettings{std::move(result.localMap), std::move(result.localStack), startPos};
|
||||||
ParseResult p = Luau::Parser::parse(srcStart, parseLength, *nameTbl, *fragmentResult.alloc.get(), opts);
|
ParseResult p = Luau::Parser::parse(srcStart, parseLength, *nameTbl, *fragmentResult.alloc.get(), opts);
|
||||||
|
|
||||||
std::vector<AstNode*> fabricatedAncestry = std::move(result.ancestry);
|
std::vector<AstNode*> fabricatedAncestry = std::move(result.ancestry);
|
||||||
|
@ -276,7 +251,7 @@ FragmentParseResult parseFragment(
|
||||||
// Get the ancestry for the fragment at the offset cursor position.
|
// Get the ancestry for the fragment at the offset cursor position.
|
||||||
// Consumers have the option to request with fragment end position, so we cannot just use the end position of our parse result as the
|
// Consumers have the option to request with fragment end position, so we cannot just use the end position of our parse result as the
|
||||||
// cursor position. Instead, use the cursor position calculated as an offset from our start position.
|
// cursor position. Instead, use the cursor position calculated as an offset from our start position.
|
||||||
std::vector<AstNode*> fragmentAncestry = findAncestryAtPositionForAutocomplete(p.root, cursorInFragment);
|
std::vector<AstNode*> fragmentAncestry = findAncestryAtPositionForAutocomplete(p.root, cursorPos);
|
||||||
fabricatedAncestry.insert(fabricatedAncestry.end(), fragmentAncestry.begin(), fragmentAncestry.end());
|
fabricatedAncestry.insert(fabricatedAncestry.end(), fragmentAncestry.begin(), fragmentAncestry.end());
|
||||||
if (nearestStatement == nullptr)
|
if (nearestStatement == nullptr)
|
||||||
nearestStatement = p.root;
|
nearestStatement = p.root;
|
||||||
|
@ -524,6 +499,7 @@ FragmentAutocompleteResult fragmentAutocomplete(
|
||||||
}
|
}
|
||||||
|
|
||||||
auto tcResult = typecheckFragment(frontend, moduleName, cursorPosition, opts, src, fragmentEndPosition);
|
auto tcResult = typecheckFragment(frontend, moduleName, cursorPosition, opts, src, fragmentEndPosition);
|
||||||
|
auto globalScope = (opts && opts->forAutocomplete) ? frontend.globalsForAutocomplete.globalScope.get() : frontend.globals.globalScope.get();
|
||||||
|
|
||||||
TypeArena arenaForFragmentAutocomplete;
|
TypeArena arenaForFragmentAutocomplete;
|
||||||
auto result = Luau::autocomplete_(
|
auto result = Luau::autocomplete_(
|
||||||
|
@ -531,7 +507,7 @@ FragmentAutocompleteResult fragmentAutocomplete(
|
||||||
frontend.builtinTypes,
|
frontend.builtinTypes,
|
||||||
&arenaForFragmentAutocomplete,
|
&arenaForFragmentAutocomplete,
|
||||||
tcResult.ancestry,
|
tcResult.ancestry,
|
||||||
frontend.globals.globalScope.get(),
|
globalScope,
|
||||||
tcResult.freshScope,
|
tcResult.freshScope,
|
||||||
cursorPosition,
|
cursorPosition,
|
||||||
frontend.fileResolver,
|
frontend.fileResolver,
|
||||||
|
|
|
@ -743,6 +743,32 @@ std::optional<CheckResult> Frontend::getCheckResult(const ModuleName& name, bool
|
||||||
return checkResult;
|
return checkResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<ModuleName> Frontend::getRequiredScripts(const ModuleName& name)
|
||||||
|
{
|
||||||
|
RequireTraceResult require = requireTrace[name];
|
||||||
|
if (isDirty(name))
|
||||||
|
{
|
||||||
|
std::optional<SourceCode> source = fileResolver->readSource(name);
|
||||||
|
if (!source)
|
||||||
|
{
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
const Config& config = configResolver->getConfig(name);
|
||||||
|
ParseOptions opts = config.parseOptions;
|
||||||
|
opts.captureComments = true;
|
||||||
|
SourceModule result = parse(name, source->source, opts);
|
||||||
|
result.type = source->type;
|
||||||
|
require = traceRequires(fileResolver, result.root, name);
|
||||||
|
}
|
||||||
|
std::vector<std::string> requiredModuleNames;
|
||||||
|
requiredModuleNames.reserve(require.requireList.size());
|
||||||
|
for (const auto& [moduleName, _] : require.requireList)
|
||||||
|
{
|
||||||
|
requiredModuleNames.push_back(moduleName);
|
||||||
|
}
|
||||||
|
return requiredModuleNames;
|
||||||
|
}
|
||||||
|
|
||||||
bool Frontend::parseGraph(
|
bool Frontend::parseGraph(
|
||||||
std::vector<ModuleName>& buildQueue,
|
std::vector<ModuleName>& buildQueue,
|
||||||
const ModuleName& root,
|
const ModuleName& root,
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixRegister)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixRegister)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixNoReadWrite)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixNoReadWrite)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunThreadBuffer)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -133,6 +134,12 @@ static std::string getTag(lua_State* L, TypeFunctionTypeId ty)
|
||||||
return "number";
|
return "number";
|
||||||
else if (auto s = get<TypeFunctionPrimitiveType>(ty); s && s->type == TypeFunctionPrimitiveType::Type::String)
|
else if (auto s = get<TypeFunctionPrimitiveType>(ty); s && s->type == TypeFunctionPrimitiveType::Type::String)
|
||||||
return "string";
|
return "string";
|
||||||
|
else if (auto s = get<TypeFunctionPrimitiveType>(ty);
|
||||||
|
FFlag::LuauUserTypeFunThreadBuffer && s && s->type == TypeFunctionPrimitiveType::Type::Thread)
|
||||||
|
return "thread";
|
||||||
|
else if (auto s = get<TypeFunctionPrimitiveType>(ty);
|
||||||
|
FFlag::LuauUserTypeFunThreadBuffer && s && s->type == TypeFunctionPrimitiveType::Type::Buffer)
|
||||||
|
return "buffer";
|
||||||
else if (get<TypeFunctionUnknownType>(ty))
|
else if (get<TypeFunctionUnknownType>(ty))
|
||||||
return "unknown";
|
return "unknown";
|
||||||
else if (get<TypeFunctionNeverType>(ty))
|
else if (get<TypeFunctionNeverType>(ty))
|
||||||
|
@ -212,6 +219,22 @@ static int createString(lua_State* L)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Luau: `type.thread`
|
||||||
|
static int createThread(lua_State* L)
|
||||||
|
{
|
||||||
|
allocTypeUserData(L, TypeFunctionPrimitiveType{TypeFunctionPrimitiveType::Thread});
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Luau: `type.buffer`
|
||||||
|
static int createBuffer(lua_State* L)
|
||||||
|
{
|
||||||
|
allocTypeUserData(L, TypeFunctionPrimitiveType{TypeFunctionPrimitiveType::Buffer});
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
// Luau: `type.singleton(value: string | boolean | nil) -> type`
|
// Luau: `type.singleton(value: string | boolean | nil) -> type`
|
||||||
// Returns the type instance representing string or boolean singleton or nil
|
// Returns the type instance representing string or boolean singleton or nil
|
||||||
static int createSingleton(lua_State* L)
|
static int createSingleton(lua_State* L)
|
||||||
|
@ -1394,6 +1417,8 @@ void registerTypesLibrary(lua_State* L)
|
||||||
{"boolean", createBoolean},
|
{"boolean", createBoolean},
|
||||||
{"number", createNumber},
|
{"number", createNumber},
|
||||||
{"string", createString},
|
{"string", createString},
|
||||||
|
{FFlag::LuauUserTypeFunThreadBuffer ? "thread" : nullptr, FFlag::LuauUserTypeFunThreadBuffer ? createThread : nullptr},
|
||||||
|
{FFlag::LuauUserTypeFunThreadBuffer ? "buffer" : nullptr, FFlag::LuauUserTypeFunThreadBuffer ? createBuffer : nullptr},
|
||||||
{nullptr, nullptr}
|
{nullptr, nullptr}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -2118,10 +2143,10 @@ private:
|
||||||
{
|
{
|
||||||
switch (p->type)
|
switch (p->type)
|
||||||
{
|
{
|
||||||
case TypeFunctionPrimitiveType::Type::NilType:
|
case TypeFunctionPrimitiveType::NilType:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::NilType));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::NilType));
|
||||||
break;
|
break;
|
||||||
case TypeFunctionPrimitiveType::Type::Boolean:
|
case TypeFunctionPrimitiveType::Boolean:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Boolean));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Boolean));
|
||||||
break;
|
break;
|
||||||
case TypeFunctionPrimitiveType::Number:
|
case TypeFunctionPrimitiveType::Number:
|
||||||
|
@ -2130,6 +2155,14 @@ private:
|
||||||
case TypeFunctionPrimitiveType::String:
|
case TypeFunctionPrimitiveType::String:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
||||||
break;
|
break;
|
||||||
|
case TypeFunctionPrimitiveType::Thread:
|
||||||
|
if (FFlag::LuauUserTypeFunThreadBuffer)
|
||||||
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Thread));
|
||||||
|
break;
|
||||||
|
case TypeFunctionPrimitiveType::Buffer:
|
||||||
|
if (FFlag::LuauUserTypeFunThreadBuffer)
|
||||||
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Buffer));
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,6 +21,7 @@
|
||||||
LUAU_DYNAMIC_FASTINTVARIABLE(LuauTypeFunctionSerdeIterationLimit, 100'000);
|
LUAU_DYNAMIC_FASTINTVARIABLE(LuauTypeFunctionSerdeIterationLimit, 100'000);
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixMetatable)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixMetatable)
|
||||||
|
LUAU_FASTFLAG(LuauUserTypeFunThreadBuffer)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -147,10 +148,10 @@ private:
|
||||||
{
|
{
|
||||||
switch (p->type)
|
switch (p->type)
|
||||||
{
|
{
|
||||||
case PrimitiveType::Type::NilType:
|
case PrimitiveType::NilType:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::NilType));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::NilType));
|
||||||
break;
|
break;
|
||||||
case PrimitiveType::Type::Boolean:
|
case PrimitiveType::Boolean:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Boolean));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Boolean));
|
||||||
break;
|
break;
|
||||||
case PrimitiveType::Number:
|
case PrimitiveType::Number:
|
||||||
|
@ -160,9 +161,29 @@ private:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
||||||
break;
|
break;
|
||||||
case PrimitiveType::Thread:
|
case PrimitiveType::Thread:
|
||||||
|
if (FFlag::LuauUserTypeFunThreadBuffer)
|
||||||
|
{
|
||||||
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Thread));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
std::string error = format("Argument of primitive type %s is not currently serializable by type functions", toString(ty).c_str());
|
||||||
|
state->errors.push_back(error);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case PrimitiveType::Buffer:
|
||||||
|
if (FFlag::LuauUserTypeFunThreadBuffer)
|
||||||
|
{
|
||||||
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Buffer));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
std::string error = format("Argument of primitive type %s is not currently serializable by type functions", toString(ty).c_str());
|
||||||
|
state->errors.push_back(error);
|
||||||
|
}
|
||||||
|
break;
|
||||||
case PrimitiveType::Function:
|
case PrimitiveType::Function:
|
||||||
case PrimitiveType::Table:
|
case PrimitiveType::Table:
|
||||||
case PrimitiveType::Buffer:
|
|
||||||
default:
|
default:
|
||||||
{
|
{
|
||||||
std::string error = format("Argument of primitive type %s is not currently serializable by type functions", toString(ty).c_str());
|
std::string error = format("Argument of primitive type %s is not currently serializable by type functions", toString(ty).c_str());
|
||||||
|
@ -565,6 +586,18 @@ private:
|
||||||
case TypeFunctionPrimitiveType::Type::String:
|
case TypeFunctionPrimitiveType::Type::String:
|
||||||
target = state->ctx->builtins->stringType;
|
target = state->ctx->builtins->stringType;
|
||||||
break;
|
break;
|
||||||
|
case TypeFunctionPrimitiveType::Type::Thread:
|
||||||
|
if (FFlag::LuauUserTypeFunThreadBuffer)
|
||||||
|
target = state->ctx->builtins->threadType;
|
||||||
|
else
|
||||||
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
|
break;
|
||||||
|
case TypeFunctionPrimitiveType::Type::Buffer:
|
||||||
|
if (FFlag::LuauUserTypeFunThreadBuffer)
|
||||||
|
target = state->ctx->builtins->bufferType;
|
||||||
|
else
|
||||||
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,7 +33,6 @@ LUAU_FASTFLAG(LuauKnowsTheDataModel3)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeDuringUnification)
|
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeDuringUnification)
|
||||||
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauMetatableFollow)
|
LUAU_FASTFLAGVARIABLE(LuauMetatableFollow)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauRequireCyclesDontAlwaysReturnAny)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -264,18 +263,7 @@ ModulePtr TypeChecker::checkWithoutRecursionCheck(const SourceModule& module, Mo
|
||||||
ScopePtr parentScope = environmentScope.value_or(globalScope);
|
ScopePtr parentScope = environmentScope.value_or(globalScope);
|
||||||
ScopePtr moduleScope = std::make_shared<Scope>(parentScope);
|
ScopePtr moduleScope = std::make_shared<Scope>(parentScope);
|
||||||
|
|
||||||
if (FFlag::LuauRequireCyclesDontAlwaysReturnAny)
|
|
||||||
{
|
|
||||||
moduleScope->returnType = freshTypePack(moduleScope);
|
moduleScope->returnType = freshTypePack(moduleScope);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (module.cyclic)
|
|
||||||
moduleScope->returnType = addTypePack(TypePack{{anyType}, std::nullopt});
|
|
||||||
else
|
|
||||||
moduleScope->returnType = freshTypePack(moduleScope);
|
|
||||||
}
|
|
||||||
|
|
||||||
moduleScope->varargPack = anyTypePack;
|
moduleScope->varargPack = anyTypePack;
|
||||||
|
|
||||||
currentModule->scopes.push_back(std::make_pair(module.root->location, moduleScope));
|
currentModule->scopes.push_back(std::make_pair(module.root->location, moduleScope));
|
||||||
|
|
|
@ -153,7 +153,7 @@ private:
|
||||||
class Lexer
|
class Lexer
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
Lexer(const char* buffer, std::size_t bufferSize, AstNameTable& names);
|
Lexer(const char* buffer, std::size_t bufferSize, AstNameTable& names, Position startPosition = {0, 0});
|
||||||
|
|
||||||
void setSkipComments(bool skip);
|
void setSkipComments(bool skip);
|
||||||
void setReadNames(bool read);
|
void setReadNames(bool read);
|
||||||
|
@ -230,6 +230,17 @@ private:
|
||||||
bool skipComments;
|
bool skipComments;
|
||||||
bool readNames;
|
bool readNames;
|
||||||
|
|
||||||
|
// This offset represents a column offset to be applied to any positions created by the lexer until the next new line.
|
||||||
|
// For example:
|
||||||
|
// local x = 4
|
||||||
|
// local y = 5
|
||||||
|
// If we start lexing from the position of `l` in `local x = 4`, the line number will be 1, and the column will be 4
|
||||||
|
// However, because the lexer calculates line offsets by 'index in source buffer where there is a newline', the column
|
||||||
|
// count will start at 0. For this reason, for just the first line, we'll need to store the offset.
|
||||||
|
unsigned int lexResumeOffset;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
enum class BraceType
|
enum class BraceType
|
||||||
{
|
{
|
||||||
InterpolatedString,
|
InterpolatedString,
|
||||||
|
|
|
@ -21,6 +21,7 @@ struct FragmentParseResumeSettings
|
||||||
{
|
{
|
||||||
DenseHashMap<AstName, AstLocal*> localMap{AstName()};
|
DenseHashMap<AstName, AstLocal*> localMap{AstName()};
|
||||||
std::vector<AstLocal*> localStack;
|
std::vector<AstLocal*> localStack;
|
||||||
|
Position resumePosition;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ParseOptions
|
struct ParseOptions
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
|
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(LexerResumesFromPosition)
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -303,16 +304,20 @@ static char unescape(char ch)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Lexer::Lexer(const char* buffer, size_t bufferSize, AstNameTable& names)
|
Lexer::Lexer(const char* buffer, size_t bufferSize, AstNameTable& names, Position startPosition)
|
||||||
: buffer(buffer)
|
: buffer(buffer)
|
||||||
, bufferSize(bufferSize)
|
, bufferSize(bufferSize)
|
||||||
, offset(0)
|
, offset(0)
|
||||||
, line(0)
|
, line(FFlag::LexerResumesFromPosition ? startPosition.line : 0)
|
||||||
, lineOffset(0)
|
, lineOffset(0)
|
||||||
, lexeme(Location(Position(0, 0), 0), Lexeme::Eof)
|
, lexeme(
|
||||||
|
(FFlag::LexerResumesFromPosition ? Location(Position(startPosition.line, startPosition.column), 0) : Location(Position(0, 0), 0)),
|
||||||
|
Lexeme::Eof
|
||||||
|
)
|
||||||
, names(names)
|
, names(names)
|
||||||
, skipComments(false)
|
, skipComments(false)
|
||||||
, readNames(true)
|
, readNames(true)
|
||||||
|
, lexResumeOffset(FFlag::LexerResumesFromPosition ? startPosition.column : 0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -367,6 +372,7 @@ Lexeme Lexer::lookahead()
|
||||||
Location currentPrevLocation = prevLocation;
|
Location currentPrevLocation = prevLocation;
|
||||||
size_t currentBraceStackSize = braceStack.size();
|
size_t currentBraceStackSize = braceStack.size();
|
||||||
BraceType currentBraceType = braceStack.empty() ? BraceType::Normal : braceStack.back();
|
BraceType currentBraceType = braceStack.empty() ? BraceType::Normal : braceStack.back();
|
||||||
|
unsigned int currentLexResumeOffset = lexResumeOffset;
|
||||||
|
|
||||||
Lexeme result = next();
|
Lexeme result = next();
|
||||||
|
|
||||||
|
@ -375,6 +381,7 @@ Lexeme Lexer::lookahead()
|
||||||
lineOffset = currentLineOffset;
|
lineOffset = currentLineOffset;
|
||||||
lexeme = currentLexeme;
|
lexeme = currentLexeme;
|
||||||
prevLocation = currentPrevLocation;
|
prevLocation = currentPrevLocation;
|
||||||
|
lexResumeOffset = currentLexResumeOffset;
|
||||||
|
|
||||||
if (braceStack.size() < currentBraceStackSize)
|
if (braceStack.size() < currentBraceStackSize)
|
||||||
braceStack.push_back(currentBraceType);
|
braceStack.push_back(currentBraceType);
|
||||||
|
@ -407,7 +414,7 @@ char Lexer::peekch(unsigned int lookahead) const
|
||||||
|
|
||||||
Position Lexer::position() const
|
Position Lexer::position() const
|
||||||
{
|
{
|
||||||
return Position(line, offset - lineOffset);
|
return Position(line, offset - lineOffset + (FFlag::LexerResumesFromPosition ? lexResumeOffset : 0));
|
||||||
}
|
}
|
||||||
|
|
||||||
LUAU_FORCEINLINE
|
LUAU_FORCEINLINE
|
||||||
|
@ -426,6 +433,9 @@ void Lexer::consumeAny()
|
||||||
{
|
{
|
||||||
line++;
|
line++;
|
||||||
lineOffset = offset + 1;
|
lineOffset = offset + 1;
|
||||||
|
// every new line, we reset
|
||||||
|
if (FFlag::LexerResumesFromPosition)
|
||||||
|
lexResumeOffset = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
offset++;
|
offset++;
|
||||||
|
|
|
@ -24,6 +24,7 @@ LUAU_FASTFLAGVARIABLE(LuauAllowFragmentParsing)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauPortableStringZeroCheck)
|
LUAU_FASTFLAGVARIABLE(LuauPortableStringZeroCheck)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauAllowComplexTypesInGenericParams)
|
LUAU_FASTFLAGVARIABLE(LuauAllowComplexTypesInGenericParams)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauErrorRecoveryForTableTypes)
|
LUAU_FASTFLAGVARIABLE(LuauErrorRecoveryForTableTypes)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauErrorRecoveryForClassNames)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -179,7 +180,7 @@ ParseResult Parser::parse(const char* buffer, size_t bufferSize, AstNameTable& n
|
||||||
|
|
||||||
Parser::Parser(const char* buffer, size_t bufferSize, AstNameTable& names, Allocator& allocator, const ParseOptions& options)
|
Parser::Parser(const char* buffer, size_t bufferSize, AstNameTable& names, Allocator& allocator, const ParseOptions& options)
|
||||||
: options(options)
|
: options(options)
|
||||||
, lexer(buffer, bufferSize, names)
|
, lexer(buffer, bufferSize, names, options.parseFragment ? options.parseFragment->resumePosition : Position(0, 0))
|
||||||
, allocator(allocator)
|
, allocator(allocator)
|
||||||
, recursionCounter(0)
|
, recursionCounter(0)
|
||||||
, endMismatchSuspect(Lexeme(Location(), Lexeme::Eof))
|
, endMismatchSuspect(Lexeme(Location(), Lexeme::Eof))
|
||||||
|
@ -1164,15 +1165,33 @@ AstStat* Parser::parseDeclaration(const Location& start, const AstArray<AstAttr*
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
|
if (FFlag::LuauErrorRecoveryForClassNames)
|
||||||
|
{
|
||||||
|
Location propStart = lexer.current().location;
|
||||||
|
std::optional<Name> propName = parseNameOpt("property name");
|
||||||
|
|
||||||
|
if (!propName)
|
||||||
|
break;
|
||||||
|
|
||||||
|
expectAndConsume(':', "property type annotation");
|
||||||
|
AstType* propType = parseType();
|
||||||
|
props.push_back(
|
||||||
|
AstDeclaredClassProp{propName->name, propName->location, propType, false, Location(propStart, lexer.previousLocation())}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
Location propStart = lexer.current().location;
|
Location propStart = lexer.current().location;
|
||||||
Name propName = parseName("property name");
|
Name propName = parseName("property name");
|
||||||
expectAndConsume(':', "property type annotation");
|
expectAndConsume(':', "property type annotation");
|
||||||
AstType* propType = parseType();
|
AstType* propType = parseType();
|
||||||
props.push_back(AstDeclaredClassProp{propName.name, propName.location, propType, false, Location(propStart, lexer.previousLocation())}
|
props.push_back(
|
||||||
|
AstDeclaredClassProp{propName.name, propName.location, propType, false, Location(propStart, lexer.previousLocation())}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Location classEnd = lexer.current().location;
|
Location classEnd = lexer.current().location;
|
||||||
nextLexeme(); // skip past `end`
|
nextLexeme(); // skip past `end`
|
||||||
|
|
|
@ -114,10 +114,12 @@ enum class IrCmd : uint8_t
|
||||||
STORE_INT,
|
STORE_INT,
|
||||||
|
|
||||||
// Store a vector into TValue
|
// Store a vector into TValue
|
||||||
|
// When optional 'E' tag is present, it is written out to the TValue as well
|
||||||
// A: Rn
|
// A: Rn
|
||||||
// B: double (x)
|
// B: double (x)
|
||||||
// C: double (y)
|
// C: double (y)
|
||||||
// D: double (z)
|
// D: double (z)
|
||||||
|
// E: tag (optional)
|
||||||
STORE_VECTOR,
|
STORE_VECTOR,
|
||||||
|
|
||||||
// Store a TValue into memory
|
// Store a TValue into memory
|
||||||
|
|
|
@ -11,7 +11,8 @@
|
||||||
#include "lstate.h"
|
#include "lstate.h"
|
||||||
#include "lgc.h"
|
#include "lgc.h"
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauVectorLibNativeDot);
|
LUAU_FASTFLAG(LuauVectorLibNativeDot)
|
||||||
|
LUAU_FASTFLAG(LuauCodeGenVectorDeadStoreElim)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -497,6 +498,13 @@ void IrLoweringA64::lowerInst(IrInst& inst, uint32_t index, const IrBlock& next)
|
||||||
build.str(temp4, AddressA64(addr.base, addr.data + 4));
|
build.str(temp4, AddressA64(addr.base, addr.data + 4));
|
||||||
build.fcvt(temp4, temp3);
|
build.fcvt(temp4, temp3);
|
||||||
build.str(temp4, AddressA64(addr.base, addr.data + 8));
|
build.str(temp4, AddressA64(addr.base, addr.data + 8));
|
||||||
|
|
||||||
|
if (FFlag::LuauCodeGenVectorDeadStoreElim && inst.e.kind != IrOpKind::None)
|
||||||
|
{
|
||||||
|
RegisterA64 temp = regs.allocTemp(KindA64::w);
|
||||||
|
build.mov(temp, tagOp(inst.e));
|
||||||
|
build.str(temp, tempAddr(inst.a, offsetof(TValue, tt)));
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IrCmd::STORE_TVALUE:
|
case IrCmd::STORE_TVALUE:
|
||||||
|
|
|
@ -15,7 +15,8 @@
|
||||||
#include "lstate.h"
|
#include "lstate.h"
|
||||||
#include "lgc.h"
|
#include "lgc.h"
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauVectorLibNativeDot);
|
LUAU_FASTFLAG(LuauVectorLibNativeDot)
|
||||||
|
LUAU_FASTFLAG(LuauCodeGenVectorDeadStoreElim)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -297,6 +298,9 @@ void IrLoweringX64::lowerInst(IrInst& inst, uint32_t index, const IrBlock& next)
|
||||||
storeDoubleAsFloat(luauRegValueVector(vmRegOp(inst.a), 0), inst.b);
|
storeDoubleAsFloat(luauRegValueVector(vmRegOp(inst.a), 0), inst.b);
|
||||||
storeDoubleAsFloat(luauRegValueVector(vmRegOp(inst.a), 1), inst.c);
|
storeDoubleAsFloat(luauRegValueVector(vmRegOp(inst.a), 1), inst.c);
|
||||||
storeDoubleAsFloat(luauRegValueVector(vmRegOp(inst.a), 2), inst.d);
|
storeDoubleAsFloat(luauRegValueVector(vmRegOp(inst.a), 2), inst.d);
|
||||||
|
|
||||||
|
if (FFlag::LuauCodeGenVectorDeadStoreElim && inst.e.kind != IrOpKind::None)
|
||||||
|
build.mov(luauRegTag(vmRegOp(inst.a)), tagOp(inst.e));
|
||||||
break;
|
break;
|
||||||
case IrCmd::STORE_TVALUE:
|
case IrCmd::STORE_TVALUE:
|
||||||
{
|
{
|
||||||
|
|
|
@ -364,7 +364,7 @@ struct ConstPropState
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// To avoid captured register invalidation tracking in lowering later, values from loads from captured registers are not propagated
|
// To avoid captured register invalidation tracking in lowering later, values from loads from captured registers are not propagated
|
||||||
// This prevents the case where load value location is linked to memory in case of a spill and is then cloberred in a user call
|
// This prevents the case where load value location is linked to memory in case of a spill and is then clobbered in a user call
|
||||||
if (function.cfg.captured.regs.test(vmRegOp(loadInst.a)))
|
if (function.cfg.captured.regs.test(vmRegOp(loadInst.a)))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -378,7 +378,7 @@ struct ConstPropState
|
||||||
if (!instLink.contains(*prevIdx))
|
if (!instLink.contains(*prevIdx))
|
||||||
createRegLink(*prevIdx, loadInst.a);
|
createRegLink(*prevIdx, loadInst.a);
|
||||||
|
|
||||||
// Substitute load instructon with the previous value
|
// Substitute load instruction with the previous value
|
||||||
substitute(function, loadInst, IrOp{IrOpKind::Inst, *prevIdx});
|
substitute(function, loadInst, IrOp{IrOpKind::Inst, *prevIdx});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -401,7 +401,7 @@ struct ConstPropState
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// To avoid captured register invalidation tracking in lowering later, values from stores into captured registers are not propagated
|
// To avoid captured register invalidation tracking in lowering later, values from stores into captured registers are not propagated
|
||||||
// This prevents the case where store creates an alternative value location in case of a spill and is then cloberred in a user call
|
// This prevents the case where store creates an alternative value location in case of a spill and is then clobbered in a user call
|
||||||
if (function.cfg.captured.regs.test(vmRegOp(storeInst.a)))
|
if (function.cfg.captured.regs.test(vmRegOp(storeInst.a)))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
|
|
@ -9,6 +9,8 @@
|
||||||
|
|
||||||
#include "lobject.h"
|
#include "lobject.h"
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauCodeGenVectorDeadStoreElim)
|
||||||
|
|
||||||
// TODO: optimization can be improved by knowing which registers are live in at each VM exit
|
// TODO: optimization can be improved by knowing which registers are live in at each VM exit
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
|
@ -323,10 +325,31 @@ static bool tryReplaceTagWithFullStore(
|
||||||
// If a 'nil' tag is being replaced by something else, we also keep 'STORE_TAG Rn, tag', expecting a value store to follow
|
// If a 'nil' tag is being replaced by something else, we also keep 'STORE_TAG Rn, tag', expecting a value store to follow
|
||||||
// And value store has to follow, as the pre-DSO code would not allow GC to observe an incomplete stack variable
|
// And value store has to follow, as the pre-DSO code would not allow GC to observe an incomplete stack variable
|
||||||
if (tag != LUA_TNIL && regInfo.valueInstIdx != ~0u)
|
if (tag != LUA_TNIL && regInfo.valueInstIdx != ~0u)
|
||||||
|
{
|
||||||
|
if (FFlag::LuauCodeGenVectorDeadStoreElim)
|
||||||
|
{
|
||||||
|
IrInst& prevValueInst = function.instructions[regInfo.valueInstIdx];
|
||||||
|
|
||||||
|
if (prevValueInst.cmd == IrCmd::STORE_VECTOR)
|
||||||
|
{
|
||||||
|
CODEGEN_ASSERT(prevValueInst.e.kind == IrOpKind::None);
|
||||||
|
IrOp prevValueX = prevValueInst.b;
|
||||||
|
IrOp prevValueY = prevValueInst.c;
|
||||||
|
IrOp prevValueZ = prevValueInst.d;
|
||||||
|
replace(function, block, instIndex, IrInst{IrCmd::STORE_VECTOR, targetOp, prevValueX, prevValueY, prevValueZ, tagOp});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
IrOp prevValueOp = prevValueInst.b;
|
||||||
|
replace(function, block, instIndex, IrInst{IrCmd::STORE_SPLIT_TVALUE, targetOp, tagOp, prevValueOp});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
IrOp prevValueOp = function.instructions[regInfo.valueInstIdx].b;
|
IrOp prevValueOp = function.instructions[regInfo.valueInstIdx].b;
|
||||||
replace(function, block, instIndex, IrInst{IrCmd::STORE_SPLIT_TVALUE, targetOp, tagOp, prevValueOp});
|
replace(function, block, instIndex, IrInst{IrCmd::STORE_SPLIT_TVALUE, targetOp, tagOp, prevValueOp});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
state.killTagStore(regInfo);
|
state.killTagStore(regInfo);
|
||||||
state.killValueStore(regInfo);
|
state.killValueStore(regInfo);
|
||||||
|
@ -356,6 +379,25 @@ static bool tryReplaceTagWithFullStore(
|
||||||
|
|
||||||
state.killTValueStore(regInfo);
|
state.killTValueStore(regInfo);
|
||||||
|
|
||||||
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
|
regInfo.maybeGco = isGCO(tag);
|
||||||
|
regInfo.knownTag = tag;
|
||||||
|
state.hasGcoToClear |= regInfo.maybeGco;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (FFlag::LuauCodeGenVectorDeadStoreElim && prev.cmd == IrCmd::STORE_VECTOR)
|
||||||
|
{
|
||||||
|
// If the 'nil' is stored, we keep 'STORE_TAG Rn, tnil' as it writes the 'full' TValue
|
||||||
|
if (tag != LUA_TNIL)
|
||||||
|
{
|
||||||
|
IrOp prevValueX = prev.b;
|
||||||
|
IrOp prevValueY = prev.c;
|
||||||
|
IrOp prevValueZ = prev.d;
|
||||||
|
replace(function, block, instIndex, IrInst{IrCmd::STORE_VECTOR, targetOp, prevValueX, prevValueY, prevValueZ, tagOp});
|
||||||
|
}
|
||||||
|
|
||||||
|
state.killTValueStore(regInfo);
|
||||||
|
|
||||||
regInfo.tvalueInstIdx = instIndex;
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
regInfo.maybeGco = isGCO(tag);
|
regInfo.maybeGco = isGCO(tag);
|
||||||
regInfo.knownTag = tag;
|
regInfo.knownTag = tag;
|
||||||
|
@ -410,6 +452,94 @@ static bool tryReplaceValueWithFullStore(
|
||||||
|
|
||||||
state.killTValueStore(regInfo);
|
state.killTValueStore(regInfo);
|
||||||
|
|
||||||
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (FFlag::LuauCodeGenVectorDeadStoreElim && prev.cmd == IrCmd::STORE_VECTOR)
|
||||||
|
{
|
||||||
|
IrOp prevTagOp = prev.e;
|
||||||
|
CODEGEN_ASSERT(prevTagOp.kind != IrOpKind::None);
|
||||||
|
uint8_t prevTag = function.tagOp(prevTagOp);
|
||||||
|
|
||||||
|
CODEGEN_ASSERT(regInfo.knownTag == prevTag);
|
||||||
|
replace(function, block, instIndex, IrInst{IrCmd::STORE_SPLIT_TVALUE, targetOp, prevTagOp, valueOp});
|
||||||
|
|
||||||
|
state.killTValueStore(regInfo);
|
||||||
|
|
||||||
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool tryReplaceVectorValueWithFullStore(
|
||||||
|
RemoveDeadStoreState& state,
|
||||||
|
IrBuilder& build,
|
||||||
|
IrFunction& function,
|
||||||
|
IrBlock& block,
|
||||||
|
uint32_t instIndex,
|
||||||
|
StoreRegInfo& regInfo
|
||||||
|
)
|
||||||
|
{
|
||||||
|
CODEGEN_ASSERT(FFlag::LuauCodeGenVectorDeadStoreElim);
|
||||||
|
|
||||||
|
// If the tag+value pair is established, we can mark both as dead and use a single split TValue store
|
||||||
|
if (regInfo.tagInstIdx != ~0u && regInfo.valueInstIdx != ~0u)
|
||||||
|
{
|
||||||
|
IrOp prevTagOp = function.instructions[regInfo.tagInstIdx].b;
|
||||||
|
uint8_t prevTag = function.tagOp(prevTagOp);
|
||||||
|
|
||||||
|
CODEGEN_ASSERT(regInfo.knownTag == prevTag);
|
||||||
|
|
||||||
|
IrInst& storeInst = function.instructions[instIndex];
|
||||||
|
CODEGEN_ASSERT(storeInst.cmd == IrCmd::STORE_VECTOR);
|
||||||
|
replace(function, storeInst.e, prevTagOp);
|
||||||
|
|
||||||
|
state.killTagStore(regInfo);
|
||||||
|
state.killValueStore(regInfo);
|
||||||
|
|
||||||
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We can also replace a dead split TValue store with a new one, while keeping the value the same
|
||||||
|
if (regInfo.tvalueInstIdx != ~0u)
|
||||||
|
{
|
||||||
|
IrInst& prev = function.instructions[regInfo.tvalueInstIdx];
|
||||||
|
|
||||||
|
if (prev.cmd == IrCmd::STORE_SPLIT_TVALUE)
|
||||||
|
{
|
||||||
|
IrOp prevTagOp = prev.b;
|
||||||
|
uint8_t prevTag = function.tagOp(prevTagOp);
|
||||||
|
|
||||||
|
CODEGEN_ASSERT(regInfo.knownTag == prevTag);
|
||||||
|
CODEGEN_ASSERT(prev.d.kind == IrOpKind::None);
|
||||||
|
|
||||||
|
IrInst& storeInst = function.instructions[instIndex];
|
||||||
|
CODEGEN_ASSERT(storeInst.cmd == IrCmd::STORE_VECTOR);
|
||||||
|
replace(function, storeInst.e, prevTagOp);
|
||||||
|
|
||||||
|
state.killTValueStore(regInfo);
|
||||||
|
|
||||||
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (prev.cmd == IrCmd::STORE_VECTOR)
|
||||||
|
{
|
||||||
|
IrOp prevTagOp = prev.e;
|
||||||
|
CODEGEN_ASSERT(prevTagOp.kind != IrOpKind::None);
|
||||||
|
uint8_t prevTag = function.tagOp(prevTagOp);
|
||||||
|
|
||||||
|
CODEGEN_ASSERT(regInfo.knownTag == prevTag);
|
||||||
|
|
||||||
|
IrInst& storeInst = function.instructions[instIndex];
|
||||||
|
CODEGEN_ASSERT(storeInst.cmd == IrCmd::STORE_VECTOR);
|
||||||
|
replace(function, storeInst.e, prevTagOp);
|
||||||
|
|
||||||
|
state.killTValueStore(regInfo);
|
||||||
|
|
||||||
regInfo.tvalueInstIdx = instIndex;
|
regInfo.tvalueInstIdx = instIndex;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -499,11 +629,32 @@ static void markDeadStoresInInst(RemoveDeadStoreState& state, IrBuilder& build,
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IrCmd::STORE_VECTOR:
|
case IrCmd::STORE_VECTOR:
|
||||||
// Partial vector value store cannot be combined into a STORE_SPLIT_TVALUE, so we skip dead store optimization for it
|
|
||||||
if (inst.a.kind == IrOpKind::VmReg)
|
if (inst.a.kind == IrOpKind::VmReg)
|
||||||
|
{
|
||||||
|
if (FFlag::LuauCodeGenVectorDeadStoreElim)
|
||||||
|
{
|
||||||
|
int reg = vmRegOp(inst.a);
|
||||||
|
|
||||||
|
if (function.cfg.captured.regs.test(reg))
|
||||||
|
return;
|
||||||
|
|
||||||
|
StoreRegInfo& regInfo = state.info[reg];
|
||||||
|
|
||||||
|
if (tryReplaceVectorValueWithFullStore(state, build, function, block, index, regInfo))
|
||||||
|
break;
|
||||||
|
|
||||||
|
// Partial value store can be removed by a new one if the tag is known
|
||||||
|
if (regInfo.knownTag != kUnknownTag)
|
||||||
|
state.killValueStore(regInfo);
|
||||||
|
|
||||||
|
regInfo.valueInstIdx = index;
|
||||||
|
regInfo.maybeGco = false;
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
state.useReg(vmRegOp(inst.a));
|
state.useReg(vmRegOp(inst.a));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
case IrCmd::STORE_TVALUE:
|
case IrCmd::STORE_TVALUE:
|
||||||
if (inst.a.kind == IrOpKind::VmReg)
|
if (inst.a.kind == IrOpKind::VmReg)
|
||||||
|
|
|
@ -27,6 +27,7 @@
|
||||||
|
|
||||||
LUAU_FASTFLAG(DebugLuauFreezeArena)
|
LUAU_FASTFLAG(DebugLuauFreezeArena)
|
||||||
LUAU_FASTFLAG(DebugLuauForceAllNewSolverTests)
|
LUAU_FASTFLAG(DebugLuauForceAllNewSolverTests)
|
||||||
|
LUAU_FASTFLAG(LuauVectorDefinitionsExtra)
|
||||||
|
|
||||||
#define DOES_NOT_PASS_NEW_SOLVER_GUARD_IMPL(line) \
|
#define DOES_NOT_PASS_NEW_SOLVER_GUARD_IMPL(line) \
|
||||||
ScopedFastFlag sff_##line{FFlag::LuauSolverV2, FFlag::DebugLuauForceAllNewSolverTests};
|
ScopedFastFlag sff_##line{FFlag::LuauSolverV2, FFlag::DebugLuauForceAllNewSolverTests};
|
||||||
|
@ -113,6 +114,8 @@ struct Fixture
|
||||||
// In that case, flag can be forced to 'true' using the example below:
|
// In that case, flag can be forced to 'true' using the example below:
|
||||||
// ScopedFastFlag sff_LuauExampleFlagDefinition{FFlag::LuauExampleFlagDefinition, true};
|
// ScopedFastFlag sff_LuauExampleFlagDefinition{FFlag::LuauExampleFlagDefinition, true};
|
||||||
|
|
||||||
|
ScopedFastFlag sff_LuauVectorDefinitionsExtra{FFlag::LuauVectorDefinitionsExtra, true};
|
||||||
|
|
||||||
// Arena freezing marks the `TypeArena`'s underlying memory as read-only, raising an access violation whenever you mutate it.
|
// Arena freezing marks the `TypeArena`'s underlying memory as read-only, raising an access violation whenever you mutate it.
|
||||||
// This is useful for tracking down violations of Luau's memory model.
|
// This is useful for tracking down violations of Luau's memory model.
|
||||||
ScopedFastFlag sff_DebugLuauFreezeArena{FFlag::DebugLuauFreezeArena, true};
|
ScopedFastFlag sff_DebugLuauFreezeArena{FFlag::DebugLuauFreezeArena, true};
|
||||||
|
|
|
@ -24,6 +24,7 @@ LUAU_FASTFLAG(LuauAllowFragmentParsing);
|
||||||
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
||||||
LUAU_FASTFLAG(LuauSymbolEquality);
|
LUAU_FASTFLAG(LuauSymbolEquality);
|
||||||
LUAU_FASTFLAG(LuauStoreSolverTypeOnModule);
|
LUAU_FASTFLAG(LuauStoreSolverTypeOnModule);
|
||||||
|
LUAU_FASTFLAG(LexerResumesFromPosition)
|
||||||
|
|
||||||
static std::optional<AutocompleteEntryMap> nullCallback(std::string tag, std::optional<const ClassType*> ptr, std::optional<std::string> contents)
|
static std::optional<AutocompleteEntryMap> nullCallback(std::string tag, std::optional<const ClassType*> ptr, std::optional<std::string> contents)
|
||||||
{
|
{
|
||||||
|
@ -46,11 +47,12 @@ static FrontendOptions getOptions()
|
||||||
template<class BaseType>
|
template<class BaseType>
|
||||||
struct FragmentAutocompleteFixtureImpl : BaseType
|
struct FragmentAutocompleteFixtureImpl : BaseType
|
||||||
{
|
{
|
||||||
ScopedFastFlag sffs[4] = {
|
ScopedFastFlag sffs[5] = {
|
||||||
{FFlag::LuauAllowFragmentParsing, true},
|
{FFlag::LuauAllowFragmentParsing, true},
|
||||||
{FFlag::LuauAutocompleteRefactorsForIncrementalAutocomplete, true},
|
{FFlag::LuauAutocompleteRefactorsForIncrementalAutocomplete, true},
|
||||||
{FFlag::LuauStoreSolverTypeOnModule, true},
|
{FFlag::LuauStoreSolverTypeOnModule, true},
|
||||||
{FFlag::LuauSymbolEquality, true},
|
{FFlag::LuauSymbolEquality, true},
|
||||||
|
{FFlag::LexerResumesFromPosition, true}
|
||||||
};
|
};
|
||||||
|
|
||||||
FragmentAutocompleteFixtureImpl()
|
FragmentAutocompleteFixtureImpl()
|
||||||
|
@ -288,6 +290,7 @@ TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "local_initializer")
|
||||||
check("local a =");
|
check("local a =");
|
||||||
auto fragment = parseFragment("local a =", Position(0, 10));
|
auto fragment = parseFragment("local a =", Position(0, 10));
|
||||||
CHECK_EQ("local a =", fragment.fragmentToParse);
|
CHECK_EQ("local a =", fragment.fragmentToParse);
|
||||||
|
CHECK_EQ(Location{Position{0, 0}, 9}, fragment.root->location);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "statement_in_empty_fragment_is_non_null")
|
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "statement_in_empty_fragment_is_non_null")
|
||||||
|
@ -334,6 +337,8 @@ local z = x + y
|
||||||
Position{3, 15}
|
Position{3, 15}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
CHECK_EQ(Location{Position{2, 0}, Position{3, 15}}, fragment.root->location);
|
||||||
|
|
||||||
CHECK_EQ("local y = 5\nlocal z = x + y", fragment.fragmentToParse);
|
CHECK_EQ("local y = 5\nlocal z = x + y", fragment.fragmentToParse);
|
||||||
CHECK_EQ(5, fragment.ancestry.size());
|
CHECK_EQ(5, fragment.ancestry.size());
|
||||||
REQUIRE(fragment.root);
|
REQUIRE(fragment.root);
|
||||||
|
@ -380,6 +385,7 @@ local y = 5
|
||||||
CHECK_EQ("local z = x + y", fragment.fragmentToParse);
|
CHECK_EQ("local z = x + y", fragment.fragmentToParse);
|
||||||
CHECK_EQ(5, fragment.ancestry.size());
|
CHECK_EQ(5, fragment.ancestry.size());
|
||||||
REQUIRE(fragment.root);
|
REQUIRE(fragment.root);
|
||||||
|
CHECK_EQ(Location{Position{2, 0}, Position{2, 15}}, fragment.root->location);
|
||||||
CHECK_EQ(1, fragment.root->body.size);
|
CHECK_EQ(1, fragment.root->body.size);
|
||||||
auto stat = fragment.root->body.data[0]->as<AstStatLocal>();
|
auto stat = fragment.root->body.data[0]->as<AstStatLocal>();
|
||||||
REQUIRE(stat);
|
REQUIRE(stat);
|
||||||
|
@ -465,7 +471,7 @@ abc("bar")
|
||||||
Position{1, 9}
|
Position{1, 9}
|
||||||
);
|
);
|
||||||
|
|
||||||
CHECK_EQ("function abc(foo: string) end\nabc(\"foo\"", stringFragment.fragmentToParse);
|
CHECK_EQ("function abc(foo: string) end\nabc(\"foo\")", stringFragment.fragmentToParse);
|
||||||
CHECK(stringFragment.nearestStatement->is<AstStatFunction>());
|
CHECK(stringFragment.nearestStatement->is<AstStatFunction>());
|
||||||
|
|
||||||
CHECK_GE(stringFragment.ancestry.size(), 1);
|
CHECK_GE(stringFragment.ancestry.size(), 1);
|
||||||
|
|
|
@ -13,7 +13,6 @@
|
||||||
using namespace Luau;
|
using namespace Luau;
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2);
|
LUAU_FASTFLAG(LuauSolverV2);
|
||||||
LUAU_FASTFLAG(LuauRequireCyclesDontAlwaysReturnAny);
|
|
||||||
LUAU_FASTFLAG(DebugLuauFreezeArena);
|
LUAU_FASTFLAG(DebugLuauFreezeArena);
|
||||||
LUAU_FASTFLAG(DebugLuauMagicTypes);
|
LUAU_FASTFLAG(DebugLuauMagicTypes);
|
||||||
|
|
||||||
|
@ -313,11 +312,9 @@ TEST_CASE_FIXTURE(FrontendFixture, "nocheck_cycle_used_by_checked")
|
||||||
REQUIRE(bool(cExports));
|
REQUIRE(bool(cExports));
|
||||||
|
|
||||||
if (FFlag::LuauSolverV2)
|
if (FFlag::LuauSolverV2)
|
||||||
CHECK_EQ("{ a: { hello: any }, b: { hello: any } }", toString(*cExports));
|
CHECK("{ a: { hello: any }, b: { hello: any } }" == toString(*cExports));
|
||||||
else if (FFlag::LuauRequireCyclesDontAlwaysReturnAny)
|
|
||||||
CHECK("{| a: any, b: any |}, {| a: {| hello: any |}, b: {| hello: any |} |}" == toString(*cExports));
|
|
||||||
else
|
else
|
||||||
CHECK_EQ("{| a: any, b: any |}", toString(*cExports));
|
CHECK("{| a: {| hello: any |}, b: {| hello: any |} |}" == toString(*cExports));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(FrontendFixture, "cycle_detection_disabled_in_nocheck")
|
TEST_CASE_FIXTURE(FrontendFixture, "cycle_detection_disabled_in_nocheck")
|
||||||
|
@ -1457,4 +1454,72 @@ TEST_CASE_FIXTURE(Fixture, "exported_tables_have_position_metadata")
|
||||||
CHECK(Location{Position{1, 17}, Position{1, 20}} == prop.location);
|
CHECK(Location{Position{1, 17}, Position{1, 20}} == prop.location);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(FrontendFixture, "get_required_scripts")
|
||||||
|
{
|
||||||
|
fileResolver.source["game/workspace/MyScript"] = R"(
|
||||||
|
local MyModuleScript = require(game.workspace.MyModuleScript)
|
||||||
|
local MyModuleScript2 = require(game.workspace.MyModuleScript2)
|
||||||
|
MyModuleScript.myPrint()
|
||||||
|
)";
|
||||||
|
|
||||||
|
fileResolver.source["game/workspace/MyModuleScript"] = R"(
|
||||||
|
local module = {}
|
||||||
|
function module.myPrint()
|
||||||
|
print("Hello World")
|
||||||
|
end
|
||||||
|
return module
|
||||||
|
)";
|
||||||
|
|
||||||
|
fileResolver.source["game/workspace/MyModuleScript2"] = R"(
|
||||||
|
local module = {}
|
||||||
|
return module
|
||||||
|
)";
|
||||||
|
|
||||||
|
// isDirty(name) is true, getRequiredScripts should not hit the cache.
|
||||||
|
frontend.markDirty("game/workspace/MyScript");
|
||||||
|
std::vector<ModuleName> requiredScripts = frontend.getRequiredScripts("game/workspace/MyScript");
|
||||||
|
REQUIRE(requiredScripts.size() == 2);
|
||||||
|
CHECK(requiredScripts[0] == "game/workspace/MyModuleScript");
|
||||||
|
CHECK(requiredScripts[1] == "game/workspace/MyModuleScript2");
|
||||||
|
|
||||||
|
// Call frontend.check first, then getRequiredScripts should hit the cache because isDirty(name) is false.
|
||||||
|
frontend.check("game/workspace/MyScript");
|
||||||
|
requiredScripts = frontend.getRequiredScripts("game/workspace/MyScript");
|
||||||
|
REQUIRE(requiredScripts.size() == 2);
|
||||||
|
CHECK(requiredScripts[0] == "game/workspace/MyModuleScript");
|
||||||
|
CHECK(requiredScripts[1] == "game/workspace/MyModuleScript2");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(FrontendFixture, "get_required_scripts_dirty")
|
||||||
|
{
|
||||||
|
fileResolver.source["game/workspace/MyScript"] = R"(
|
||||||
|
print("Hello World")
|
||||||
|
)";
|
||||||
|
|
||||||
|
fileResolver.source["game/workspace/MyModuleScript"] = R"(
|
||||||
|
local module = {}
|
||||||
|
function module.myPrint()
|
||||||
|
print("Hello World")
|
||||||
|
end
|
||||||
|
return module
|
||||||
|
)";
|
||||||
|
|
||||||
|
frontend.check("game/workspace/MyScript");
|
||||||
|
std::vector<ModuleName> requiredScripts = frontend.getRequiredScripts("game/workspace/MyScript");
|
||||||
|
REQUIRE(requiredScripts.size() == 0);
|
||||||
|
|
||||||
|
fileResolver.source["game/workspace/MyScript"] = R"(
|
||||||
|
local MyModuleScript = require(game.workspace.MyModuleScript)
|
||||||
|
MyModuleScript.myPrint()
|
||||||
|
)";
|
||||||
|
|
||||||
|
requiredScripts = frontend.getRequiredScripts("game/workspace/MyScript");
|
||||||
|
REQUIRE(requiredScripts.size() == 0);
|
||||||
|
|
||||||
|
frontend.markDirty("game/workspace/MyScript");
|
||||||
|
requiredScripts = frontend.getRequiredScripts("game/workspace/MyScript");
|
||||||
|
REQUIRE(requiredScripts.size() == 1);
|
||||||
|
CHECK(requiredScripts[0] == "game/workspace/MyModuleScript");
|
||||||
|
}
|
||||||
|
|
||||||
TEST_SUITE_END();
|
TEST_SUITE_END();
|
||||||
|
|
|
@ -13,6 +13,8 @@
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
LUAU_FASTFLAG(DebugLuauAbortingChecks)
|
LUAU_FASTFLAG(DebugLuauAbortingChecks)
|
||||||
|
LUAU_FASTFLAG(LuauCodeGenVectorDeadStoreElim)
|
||||||
|
LUAU_FASTFLAG(LuauCodeGenArithOpt)
|
||||||
|
|
||||||
using namespace Luau::CodeGen;
|
using namespace Luau::CodeGen;
|
||||||
|
|
||||||
|
@ -119,6 +121,7 @@ public:
|
||||||
static const int tnil = 0;
|
static const int tnil = 0;
|
||||||
static const int tboolean = 1;
|
static const int tboolean = 1;
|
||||||
static const int tnumber = 3;
|
static const int tnumber = 3;
|
||||||
|
static const int tvector = 4;
|
||||||
static const int tstring = 5;
|
static const int tstring = 5;
|
||||||
static const int ttable = 6;
|
static const int ttable = 6;
|
||||||
static const int tfunction = 7;
|
static const int tfunction = 7;
|
||||||
|
@ -1720,6 +1723,57 @@ bb_fallback_1:
|
||||||
)");
|
)");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "NumericSimplifications")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenArithOpt{FFlag::LuauCodeGenArithOpt, true};
|
||||||
|
|
||||||
|
IrOp block = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(block);
|
||||||
|
IrOp value = build.inst(IrCmd::LOAD_DOUBLE, build.vmReg(0));
|
||||||
|
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(1), build.inst(IrCmd::SUB_NUM, value, build.constDouble(0.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(2), build.inst(IrCmd::ADD_NUM, value, build.constDouble(-0.0)));
|
||||||
|
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(3), build.inst(IrCmd::MUL_NUM, value, build.constDouble(1.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(4), build.inst(IrCmd::MUL_NUM, value, build.constDouble(2.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(5), build.inst(IrCmd::MUL_NUM, value, build.constDouble(-1.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(6), build.inst(IrCmd::MUL_NUM, value, build.constDouble(3.0)));
|
||||||
|
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(7), build.inst(IrCmd::DIV_NUM, value, build.constDouble(1.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(8), build.inst(IrCmd::DIV_NUM, value, build.constDouble(-1.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(9), build.inst(IrCmd::DIV_NUM, value, build.constDouble(32.0)));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(10), build.inst(IrCmd::DIV_NUM, value, build.constDouble(6.0)));
|
||||||
|
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(1), build.constInt(9));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
constPropInBlockChains(build, true);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
%0 = LOAD_DOUBLE R0
|
||||||
|
STORE_DOUBLE R1, %0
|
||||||
|
STORE_DOUBLE R2, %0
|
||||||
|
STORE_DOUBLE R3, %0
|
||||||
|
%7 = ADD_NUM %0, %0
|
||||||
|
STORE_DOUBLE R4, %7
|
||||||
|
%9 = UNM_NUM %0
|
||||||
|
STORE_DOUBLE R5, %9
|
||||||
|
%11 = MUL_NUM %0, 3
|
||||||
|
STORE_DOUBLE R6, %11
|
||||||
|
STORE_DOUBLE R7, %0
|
||||||
|
%15 = UNM_NUM %0
|
||||||
|
STORE_DOUBLE R8, %15
|
||||||
|
%17 = MUL_NUM %0, 0.03125
|
||||||
|
STORE_DOUBLE R9, %17
|
||||||
|
%19 = DIV_NUM %0, 6
|
||||||
|
STORE_DOUBLE R10, %19
|
||||||
|
RETURN R1, 9i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
TEST_SUITE_END();
|
TEST_SUITE_END();
|
||||||
|
|
||||||
TEST_SUITE_BEGIN("LinearExecutionFlowExtraction");
|
TEST_SUITE_BEGIN("LinearExecutionFlowExtraction");
|
||||||
|
@ -4416,6 +4470,210 @@ bb_0:
|
||||||
)");
|
)");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "VectorOverNumber")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(2.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(1.0), build.constDouble(2.0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_VECTOR R0, 1, 2, 4, tvector
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "VectorOverVector")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(4.0), build.constDouble(2.0), build.constDouble(1.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(1.0), build.constDouble(2.0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_VECTOR R0, 1, 2, 4, tvector
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "NumberOverVector")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(1.0), build.constDouble(2.0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(2.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_SPLIT_TVALUE R0, tnumber, 2
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "NumberOverNil")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnil));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(2.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_SPLIT_TVALUE R0, tnumber, 2
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "VectorOverNil")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnil));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(1.0), build.constDouble(2.0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_VECTOR R0, 1, 2, 4, tvector
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "NumberOverCombinedVector")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(2.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(1.0), build.constDouble(2.0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(3.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_SPLIT_TVALUE R0, tnumber, 3
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "VectorOverCombinedVector")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(2.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(1.0), build.constDouble(2.0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(8.0), build.constDouble(16.0), build.constDouble(32.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_VECTOR R0, 8, 16, 32, tvector
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(IrBuilderFixture, "VectorOverCombinedNumber")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauCodeGenVectorDeadStoreElim{FFlag::LuauCodeGenVectorDeadStoreElim, true};
|
||||||
|
|
||||||
|
IrOp entry = build.block(IrBlockKind::Internal);
|
||||||
|
|
||||||
|
build.beginBlock(entry);
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(2.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(0), build.constDouble(4.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tnumber));
|
||||||
|
build.inst(IrCmd::STORE_VECTOR, build.vmReg(0), build.constDouble(8.0), build.constDouble(16.0), build.constDouble(32.0));
|
||||||
|
build.inst(IrCmd::STORE_TAG, build.vmReg(0), build.constTag(tvector));
|
||||||
|
build.inst(IrCmd::RETURN, build.vmReg(0), build.constInt(1));
|
||||||
|
|
||||||
|
updateUseCounts(build.function);
|
||||||
|
computeCfgInfo(build.function);
|
||||||
|
markDeadStoresInBlockChains(build);
|
||||||
|
|
||||||
|
CHECK("\n" + toString(build.function, IncludeUseInfo::No) == R"(
|
||||||
|
bb_0:
|
||||||
|
STORE_VECTOR R0, 8, 16, 32, tvector
|
||||||
|
RETURN R0, 1i
|
||||||
|
|
||||||
|
)");
|
||||||
|
}
|
||||||
|
|
||||||
TEST_SUITE_END();
|
TEST_SUITE_END();
|
||||||
|
|
||||||
TEST_SUITE_BEGIN("Dump");
|
TEST_SUITE_BEGIN("Dump");
|
||||||
|
|
|
@ -20,6 +20,7 @@ LUAU_FASTFLAG(LuauUserDefinedTypeFunctionsSyntax2)
|
||||||
LUAU_FASTFLAG(LuauUserDefinedTypeFunParseExport)
|
LUAU_FASTFLAG(LuauUserDefinedTypeFunParseExport)
|
||||||
LUAU_FASTFLAG(LuauAllowComplexTypesInGenericParams)
|
LUAU_FASTFLAG(LuauAllowComplexTypesInGenericParams)
|
||||||
LUAU_FASTFLAG(LuauErrorRecoveryForTableTypes)
|
LUAU_FASTFLAG(LuauErrorRecoveryForTableTypes)
|
||||||
|
LUAU_FASTFLAG(LuauErrorRecoveryForClassNames)
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
|
@ -2085,6 +2086,20 @@ TEST_CASE_FIXTURE(Fixture, "variadic_definition_parsing")
|
||||||
matchParseError("declare class Foo function a(self, ...) end", "All declaration parameters aside from 'self' must be annotated");
|
matchParseError("declare class Foo function a(self, ...) end", "All declaration parameters aside from 'self' must be annotated");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(Fixture, "missing_declaration_prop")
|
||||||
|
{
|
||||||
|
ScopedFastFlag luauErrorRecoveryForClassNames{FFlag::LuauErrorRecoveryForClassNames, true};
|
||||||
|
|
||||||
|
matchParseError(
|
||||||
|
R"(
|
||||||
|
declare class Foo
|
||||||
|
a: number,
|
||||||
|
end
|
||||||
|
)",
|
||||||
|
"Expected identifier when parsing property name, got ','"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(Fixture, "generic_pack_parsing")
|
TEST_CASE_FIXTURE(Fixture, "generic_pack_parsing")
|
||||||
{
|
{
|
||||||
ParseResult result = parseEx(R"(
|
ParseResult result = parseEx(R"(
|
||||||
|
|
|
@ -16,6 +16,7 @@ LUAU_FASTFLAG(LuauUserDefinedTypeFunctionResetState)
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunNonstrict)
|
LUAU_FASTFLAG(LuauUserTypeFunNonstrict)
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunExportedAndLocal)
|
LUAU_FASTFLAG(LuauUserTypeFunExportedAndLocal)
|
||||||
LUAU_FASTFLAG(LuauUserDefinedTypeFunParseExport)
|
LUAU_FASTFLAG(LuauUserDefinedTypeFunParseExport)
|
||||||
|
LUAU_FASTFLAG(LuauUserTypeFunThreadBuffer)
|
||||||
|
|
||||||
TEST_SUITE_BEGIN("UserDefinedTypeFunctionTests");
|
TEST_SUITE_BEGIN("UserDefinedTypeFunctionTests");
|
||||||
|
|
||||||
|
@ -235,6 +236,36 @@ TEST_CASE_FIXTURE(BuiltinsFixture, "udtf_number_methods_work")
|
||||||
LUAU_REQUIRE_NO_ERRORS(result);
|
LUAU_REQUIRE_NO_ERRORS(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(BuiltinsFixture, "thread_and_buffer_types")
|
||||||
|
{
|
||||||
|
ScopedFastFlag newSolver{FFlag::LuauSolverV2, true};
|
||||||
|
ScopedFastFlag udtfSyntax{FFlag::LuauUserDefinedTypeFunctionsSyntax2, true};
|
||||||
|
ScopedFastFlag luauUserTypeFunFixRegister{FFlag::LuauUserTypeFunFixRegister, true};
|
||||||
|
ScopedFastFlag luauUserTypeFunThreadBuffer{FFlag::LuauUserTypeFunThreadBuffer, true};
|
||||||
|
|
||||||
|
LUAU_REQUIRE_NO_ERRORS(check(R"(
|
||||||
|
type function work_with_thread(x)
|
||||||
|
if x:is("thread") then
|
||||||
|
return types.thread
|
||||||
|
end
|
||||||
|
return types.string
|
||||||
|
end
|
||||||
|
type X = thread
|
||||||
|
local function ok(idx: work_with_thread<X>): thread return idx end
|
||||||
|
)"));
|
||||||
|
|
||||||
|
LUAU_REQUIRE_NO_ERRORS(check(R"(
|
||||||
|
type function work_with_buffer(x)
|
||||||
|
if x:is("buffer") then
|
||||||
|
return types.buffer
|
||||||
|
end
|
||||||
|
return types.string
|
||||||
|
end
|
||||||
|
type X = buffer
|
||||||
|
local function ok(idx: work_with_buffer<X>): buffer return idx end
|
||||||
|
)"));
|
||||||
|
}
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(BuiltinsFixture, "udtf_string_serialization_works")
|
TEST_CASE_FIXTURE(BuiltinsFixture, "udtf_string_serialization_works")
|
||||||
{
|
{
|
||||||
ScopedFastFlag newSolver{FFlag::LuauSolverV2, true};
|
ScopedFastFlag newSolver{FFlag::LuauSolverV2, true};
|
||||||
|
|
|
@ -11,7 +11,6 @@
|
||||||
#include "doctest.h"
|
#include "doctest.h"
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
||||||
LUAU_FASTFLAG(LuauRequireCyclesDontAlwaysReturnAny)
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAG(LuauTypestateBuiltins2)
|
LUAU_FASTFLAG(LuauTypestateBuiltins2)
|
||||||
LUAU_FASTFLAG(LuauNewSolverPopulateTableLocations)
|
LUAU_FASTFLAG(LuauNewSolverPopulateTableLocations)
|
||||||
|
@ -756,8 +755,6 @@ TEST_CASE_FIXTURE(BuiltinsFixture, "spooky_blocked_type_laundered_by_bound_type"
|
||||||
|
|
||||||
TEST_CASE_FIXTURE(BuiltinsFixture, "cycles_dont_make_everything_any")
|
TEST_CASE_FIXTURE(BuiltinsFixture, "cycles_dont_make_everything_any")
|
||||||
{
|
{
|
||||||
ScopedFastFlag sff{FFlag::LuauRequireCyclesDontAlwaysReturnAny, true};
|
|
||||||
|
|
||||||
fileResolver.source["game/A"] = R"(
|
fileResolver.source["game/A"] = R"(
|
||||||
--!strict
|
--!strict
|
||||||
local module = {}
|
local module = {}
|
||||||
|
|
|
@ -12,6 +12,8 @@
|
||||||
|
|
||||||
#include "doctest.h"
|
#include "doctest.h"
|
||||||
|
|
||||||
|
LUAU_FASTFLAG(LuauVectorDefinitions)
|
||||||
|
|
||||||
using namespace Luau;
|
using namespace Luau;
|
||||||
|
|
||||||
TEST_SUITE_BEGIN("TypeInferPrimitives");
|
TEST_SUITE_BEGIN("TypeInferPrimitives");
|
||||||
|
@ -120,4 +122,34 @@ TEST_CASE_FIXTURE(BuiltinsFixture, "property_of_buffers")
|
||||||
LUAU_REQUIRE_ERROR_COUNT(1, result);
|
LUAU_REQUIRE_ERROR_COUNT(1, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_CASE_FIXTURE(BuiltinsFixture, "properties_of_vectors")
|
||||||
|
{
|
||||||
|
CheckResult result = check(R"(
|
||||||
|
local a = vector.create(1, 2, 3)
|
||||||
|
local b = vector.create(4, 5, 6)
|
||||||
|
|
||||||
|
local t1 = {
|
||||||
|
a + b,
|
||||||
|
a - b,
|
||||||
|
a * 3,
|
||||||
|
a * b,
|
||||||
|
3 * b,
|
||||||
|
a / 3,
|
||||||
|
a / b,
|
||||||
|
3 / b,
|
||||||
|
a // 4,
|
||||||
|
a // b,
|
||||||
|
4 // b,
|
||||||
|
-a,
|
||||||
|
}
|
||||||
|
local t2 = {
|
||||||
|
a.x,
|
||||||
|
a.y,
|
||||||
|
a.z,
|
||||||
|
}
|
||||||
|
)");
|
||||||
|
|
||||||
|
LUAU_REQUIRE_NO_ERRORS(result);
|
||||||
|
}
|
||||||
|
|
||||||
TEST_SUITE_END();
|
TEST_SUITE_END();
|
||||||
|
|
Loading…
Reference in a new issue