mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 02:10:53 +01:00
Merge branch 'master' into petrih-vector2-ctor
This commit is contained in:
commit
f0a0b55992
120 changed files with 1649 additions and 816 deletions
|
@ -15,6 +15,12 @@ namespace Luau
|
|||
{
|
||||
struct FrontendOptions;
|
||||
|
||||
enum class FragmentTypeCheckStatus
|
||||
{
|
||||
Success,
|
||||
SkipAutocomplete,
|
||||
};
|
||||
|
||||
struct FragmentAutocompleteAncestryResult
|
||||
{
|
||||
DenseHashMap<AstName, AstLocal*> localMap{AstName()};
|
||||
|
@ -29,6 +35,7 @@ struct FragmentParseResult
|
|||
AstStatBlock* root = nullptr;
|
||||
std::vector<AstNode*> ancestry;
|
||||
AstStat* nearestStatement = nullptr;
|
||||
std::vector<Comment> commentLocations;
|
||||
std::unique_ptr<Allocator> alloc = std::make_unique<Allocator>();
|
||||
};
|
||||
|
||||
|
@ -56,7 +63,7 @@ FragmentParseResult parseFragment(
|
|||
std::optional<Position> fragmentEndPosition
|
||||
);
|
||||
|
||||
FragmentTypeCheckResult typecheckFragment(
|
||||
std::pair<FragmentTypeCheckStatus, FragmentTypeCheckResult> typecheckFragment(
|
||||
Frontend& frontend,
|
||||
const ModuleName& moduleName,
|
||||
const Position& cursorPos,
|
||||
|
|
|
@ -16,6 +16,8 @@
|
|||
#include <unordered_map>
|
||||
#include <optional>
|
||||
|
||||
LUAU_FASTFLAG(LuauIncrementalAutocompleteCommentDetection)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
||||
|
@ -55,6 +57,7 @@ struct SourceModule
|
|||
}
|
||||
};
|
||||
|
||||
bool isWithinComment(const std::vector<Comment>& commentLocations, Position pos);
|
||||
bool isWithinComment(const SourceModule& sourceModule, Position pos);
|
||||
bool isWithinComment(const ParseResult& result, Position pos);
|
||||
|
||||
|
|
|
@ -95,6 +95,8 @@ struct Scope
|
|||
// we need that the generic type `T` in both cases is the same, so we use a cache.
|
||||
std::unordered_map<Name, TypeId> typeAliasTypeParameters;
|
||||
std::unordered_map<Name, TypePackId> typeAliasTypePackParameters;
|
||||
|
||||
std::optional<std::vector<TypeId>> interiorFreeTypes;
|
||||
};
|
||||
|
||||
// Returns true iff the left scope encloses the right scope. A Scope* equal to
|
||||
|
|
|
@ -626,7 +626,7 @@ struct TypeFunctionInstanceType
|
|||
std::vector<TypeId> typeArguments;
|
||||
std::vector<TypePackId> packArguments;
|
||||
|
||||
std::optional<AstName> userFuncName; // Name of the user-defined type function; only available for UDTFs
|
||||
std::optional<AstName> userFuncName; // Name of the user-defined type function; only available for UDTFs
|
||||
UserDefinedFunctionData userFuncData;
|
||||
|
||||
TypeFunctionInstanceType(
|
||||
|
|
|
@ -71,7 +71,7 @@ struct TypeFunctionContext
|
|||
// The constraint being reduced in this run of the reduction
|
||||
const Constraint* constraint;
|
||||
|
||||
std::optional<AstName> userFuncName; // Name of the user-defined type function; only available for UDTFs
|
||||
std::optional<AstName> userFuncName; // Name of the user-defined type function; only available for UDTFs
|
||||
|
||||
TypeFunctionContext(NotNull<ConstraintSolver> cs, NotNull<Scope> scope, NotNull<const Constraint> constraint);
|
||||
|
||||
|
|
|
@ -269,8 +269,8 @@ bool isLiteral(const AstExpr* expr);
|
|||
std::vector<TypeId> findBlockedTypesIn(AstExprTable* expr, NotNull<DenseHashMap<const AstExpr*, TypeId>> astTypes);
|
||||
|
||||
/**
|
||||
* Given a function call and a mapping from expression to type, determine
|
||||
* whether the type of any argument in said call in depends on a blocked types.
|
||||
* Given a function call and a mapping from expression to type, determine
|
||||
* whether the type of any argument in said call in depends on a blocked types.
|
||||
* This is used as a precondition for bidirectional inference: be warned that
|
||||
* the behavior of this algorithm is tightly coupled to that of bidirectional
|
||||
* inference.
|
||||
|
@ -280,4 +280,13 @@ std::vector<TypeId> findBlockedTypesIn(AstExprTable* expr, NotNull<DenseHashMap<
|
|||
*/
|
||||
std::vector<TypeId> findBlockedArgTypesIn(AstExprCall* expr, NotNull<DenseHashMap<const AstExpr*, TypeId>> astTypes);
|
||||
|
||||
/**
|
||||
* Given a scope and a free type, find the closest parent that has a present
|
||||
* `interiorFreeTypes` and append the given type to said list. This list will
|
||||
* be generalized when the requiste `GeneralizationConstraint` is resolved.
|
||||
* @param scope Initial scope this free type was attached to
|
||||
* @param ty Free type to track.
|
||||
*/
|
||||
void trackInteriorFreeType(Scope* scope, TypeId ty);
|
||||
|
||||
} // namespace Luau
|
||||
|
|
|
@ -177,7 +177,6 @@ void AnyTypeSummary::visit(const Scope* scope, AstStatReturn* ret, const Module*
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
void AnyTypeSummary::visit(const Scope* scope, AstStatLocal* local, const Module* module, NotNull<BuiltinTypes> builtinTypes)
|
||||
|
|
|
@ -25,6 +25,7 @@ LUAU_FASTINT(LuauTypeInferIterationLimit)
|
|||
LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
||||
LUAU_FASTFLAGVARIABLE(LuauAutocompleteUseLimits)
|
||||
|
||||
static const std::unordered_set<std::string> kStatementStartingKeywords =
|
||||
{"while", "if", "local", "repeat", "function", "do", "for", "return", "break", "continue", "type", "export"};
|
||||
|
@ -177,6 +178,12 @@ static bool checkTypeMatch(TypeId subTy, TypeId superTy, NotNull<Scope> scope, T
|
|||
unifier.normalize = false;
|
||||
unifier.checkInhabited = false;
|
||||
|
||||
if (FFlag::LuauAutocompleteUseLimits)
|
||||
{
|
||||
unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit;
|
||||
unifierState.counters.iterationLimit = FInt::LuauTypeInferIterationLimit;
|
||||
}
|
||||
|
||||
return unifier.canUnify(subTy, superTy).empty();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -611,7 +611,9 @@ static void dcrMagicFunctionTypeCheckFormat(MagicFunctionTypeCheckContext contex
|
|||
if (!fmt)
|
||||
{
|
||||
if (FFlag::LuauStringFormatArityFix)
|
||||
context.typechecker->reportError(CountMismatch{1, std::nullopt, 0, CountMismatch::Arg, true, "string.format"}, context.callSite->location);
|
||||
context.typechecker->reportError(
|
||||
CountMismatch{1, std::nullopt, 0, CountMismatch::Arg, true, "string.format"}, context.callSite->location
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
|
|
|
@ -62,7 +62,6 @@ struct ReferenceCountInitializer : TypeOnceVisitor
|
|||
// of this type, hence:
|
||||
return !FFlag::LuauDontRefCountTypesInTypeFunctions;
|
||||
}
|
||||
|
||||
};
|
||||
|
||||
bool isReferenceCountedType(const TypeId typ)
|
||||
|
|
|
@ -31,15 +31,15 @@
|
|||
LUAU_FASTINT(LuauCheckRecursionLimit)
|
||||
LUAU_FASTFLAG(DebugLuauLogSolverToJson)
|
||||
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
||||
LUAU_FASTFLAG(DebugLuauEqSatSimplification)
|
||||
LUAU_FASTFLAG(LuauTypestateBuiltins2)
|
||||
LUAU_FASTFLAG(LuauUserTypeFunUpdateAllEnvs)
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverVisitErrorExprLvalues)
|
||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPrePopulateClasses)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunExportedAndLocal)
|
||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPrePopulateClasses)
|
||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPopulateTableLocations)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunNoExtraConstraint)
|
||||
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTypesOnScope)
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(InferGlobalTypes)
|
||||
|
||||
|
@ -233,8 +233,17 @@ void ConstraintGenerator::visitModuleRoot(AstStatBlock* block)
|
|||
Checkpoint end = checkpoint(this);
|
||||
|
||||
TypeId result = arena->addType(BlockedType{});
|
||||
NotNull<Constraint> genConstraint =
|
||||
addConstraint(scope, block->location, GeneralizationConstraint{result, moduleFnTy, std::move(interiorTypes.back())});
|
||||
NotNull<Constraint> genConstraint = addConstraint(
|
||||
scope,
|
||||
block->location,
|
||||
GeneralizationConstraint{
|
||||
result, moduleFnTy, FFlag::LuauTrackInteriorFreeTypesOnScope ? std::vector<TypeId>{} : std::move(interiorTypes.back())
|
||||
}
|
||||
);
|
||||
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
scope->interiorFreeTypes = std::move(interiorTypes.back());
|
||||
|
||||
getMutable<BlockedType>(result)->setOwner(genConstraint);
|
||||
forEachConstraint(
|
||||
start,
|
||||
|
@ -303,9 +312,19 @@ void ConstraintGenerator::visitFragmentRoot(const ScopePtr& resumeScope, AstStat
|
|||
}
|
||||
}
|
||||
|
||||
|
||||
TypeId ConstraintGenerator::freshType(const ScopePtr& scope)
|
||||
{
|
||||
return Luau::freshType(arena, builtinTypes, scope.get());
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
{
|
||||
auto ft = Luau::freshType(arena, builtinTypes, scope.get());
|
||||
interiorTypes.back().push_back(ft);
|
||||
return ft;
|
||||
}
|
||||
else
|
||||
{
|
||||
return Luau::freshType(arena, builtinTypes, scope.get());
|
||||
}
|
||||
}
|
||||
|
||||
TypePackId ConstraintGenerator::freshTypePack(const ScopePtr& scope)
|
||||
|
@ -2408,8 +2427,17 @@ Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprFunction* fun
|
|||
Checkpoint endCheckpoint = checkpoint(this);
|
||||
|
||||
TypeId generalizedTy = arena->addType(BlockedType{});
|
||||
NotNull<Constraint> gc =
|
||||
addConstraint(sig.signatureScope, func->location, GeneralizationConstraint{generalizedTy, sig.signature, std::move(interiorTypes.back())});
|
||||
NotNull<Constraint> gc = addConstraint(
|
||||
sig.signatureScope,
|
||||
func->location,
|
||||
GeneralizationConstraint{
|
||||
generalizedTy, sig.signature, FFlag::LuauTrackInteriorFreeTypesOnScope ? std::vector<TypeId>{} : std::move(interiorTypes.back())
|
||||
}
|
||||
);
|
||||
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
sig.signatureScope->interiorFreeTypes = std::move(interiorTypes.back());
|
||||
|
||||
getMutable<BlockedType>(generalizedTy)->setOwner(gc);
|
||||
interiorTypes.pop_back();
|
||||
|
||||
|
@ -2975,11 +3003,11 @@ Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprTable* expr,
|
|||
ty,
|
||||
expr,
|
||||
toBlock
|
||||
);
|
||||
// The visitor we ran prior should ensure that there are no
|
||||
// blocked types that we would encounter while matching on
|
||||
// this expression.
|
||||
LUAU_ASSERT(toBlock.empty());
|
||||
);
|
||||
// The visitor we ran prior should ensure that there are no
|
||||
// blocked types that we would encounter while matching on
|
||||
// this expression.
|
||||
LUAU_ASSERT(toBlock.empty());
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -3941,20 +3969,7 @@ TypeId ConstraintGenerator::createTypeFunctionInstance(
|
|||
|
||||
TypeId ConstraintGenerator::simplifyUnion(const ScopePtr& scope, Location location, TypeId left, TypeId right)
|
||||
{
|
||||
if (FFlag::DebugLuauEqSatSimplification)
|
||||
{
|
||||
TypeId ty = arena->addType(UnionType{{left, right}});
|
||||
std::optional<EqSatSimplificationResult> res = eqSatSimplify(simplifier, ty);
|
||||
if (!res)
|
||||
return ty;
|
||||
|
||||
for (TypeId tyFun : res->newTypeFunctions)
|
||||
addConstraint(scope, location, ReduceConstraint{tyFun});
|
||||
|
||||
return res->result;
|
||||
}
|
||||
else
|
||||
return ::Luau::simplifyUnion(builtinTypes, arena, left, right).result;
|
||||
return ::Luau::simplifyUnion(builtinTypes, arena, left, right).result;
|
||||
}
|
||||
|
||||
std::vector<NotNull<Constraint>> borrowConstraints(const std::vector<ConstraintPtr>& constraints)
|
||||
|
|
|
@ -36,6 +36,7 @@ LUAU_FASTFLAGVARIABLE(DebugLuauEqSatSimplification)
|
|||
LUAU_FASTFLAG(LuauNewSolverPopulateTableLocations)
|
||||
LUAU_FASTFLAGVARIABLE(LuauAllowNilAssignmentToIndexer)
|
||||
LUAU_FASTFLAG(LuauUserTypeFunNoExtraConstraint)
|
||||
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -724,8 +725,20 @@ bool ConstraintSolver::tryDispatch(const GeneralizationConstraint& c, NotNull<co
|
|||
bind(constraint, c.generalizedType, builtinTypes->errorRecoveryType());
|
||||
}
|
||||
|
||||
for (TypeId ty : c.interiorTypes)
|
||||
generalize(NotNull{arena}, builtinTypes, constraint->scope, generalizedTypes, ty, /* avoidSealingTables */ false);
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
{
|
||||
// We check if this member is initialized and then access it, but
|
||||
// clang-tidy doesn't understand this is safe.
|
||||
if (constraint->scope->interiorFreeTypes)
|
||||
for (TypeId ty : *constraint->scope->interiorFreeTypes) // NOLINT(bugprone-unchecked-optional-access)
|
||||
generalize(NotNull{arena}, builtinTypes, constraint->scope, generalizedTypes, ty, /* avoidSealingTables */ false);
|
||||
}
|
||||
else
|
||||
{
|
||||
for (TypeId ty : c.interiorTypes)
|
||||
generalize(NotNull{arena}, builtinTypes, constraint->scope, generalizedTypes, ty, /* avoidSealingTables */ false);
|
||||
}
|
||||
|
||||
|
||||
return true;
|
||||
}
|
||||
|
@ -801,6 +814,11 @@ bool ConstraintSolver::tryDispatch(const IterableConstraint& c, NotNull<const Co
|
|||
{
|
||||
TypeId keyTy = freshType(arena, builtinTypes, constraint->scope);
|
||||
TypeId valueTy = freshType(arena, builtinTypes, constraint->scope);
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
{
|
||||
trackInteriorFreeType(constraint->scope, keyTy);
|
||||
trackInteriorFreeType(constraint->scope, valueTy);
|
||||
}
|
||||
TypeId tableTy =
|
||||
arena->addType(TableType{TableType::Props{}, TableIndexer{keyTy, valueTy}, TypeLevel{}, constraint->scope, TableState::Free});
|
||||
|
||||
|
@ -2062,6 +2080,8 @@ bool ConstraintSolver::tryDispatch(const UnpackConstraint& c, NotNull<const Cons
|
|||
// constitute any meaningful constraint, so we replace it
|
||||
// with a free type.
|
||||
TypeId f = freshType(arena, builtinTypes, constraint->scope);
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
trackInteriorFreeType(constraint->scope, f);
|
||||
shiftReferences(resultTy, f);
|
||||
emplaceType<BoundType>(asMutable(resultTy), f);
|
||||
}
|
||||
|
@ -2197,6 +2217,11 @@ bool ConstraintSolver::tryDispatchIterableTable(TypeId iteratorTy, const Iterabl
|
|||
{
|
||||
TypeId keyTy = freshType(arena, builtinTypes, constraint->scope);
|
||||
TypeId valueTy = freshType(arena, builtinTypes, constraint->scope);
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
{
|
||||
trackInteriorFreeType(constraint->scope, keyTy);
|
||||
trackInteriorFreeType(constraint->scope, valueTy);
|
||||
}
|
||||
TypeId tableTy = arena->addType(TableType{TableState::Sealed, {}, constraint->scope});
|
||||
getMutable<TableType>(tableTy)->indexer = TableIndexer{keyTy, valueTy};
|
||||
|
||||
|
@ -2453,6 +2478,8 @@ TablePropLookupResult ConstraintSolver::lookupTableProp(
|
|||
if (ttv->state == TableState::Free)
|
||||
{
|
||||
TypeId result = freshType(arena, builtinTypes, ttv->scope);
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
trackInteriorFreeType(ttv->scope, result);
|
||||
switch (context)
|
||||
{
|
||||
case ValueContext::RValue:
|
||||
|
@ -2562,6 +2589,9 @@ TablePropLookupResult ConstraintSolver::lookupTableProp(
|
|||
LUAU_ASSERT(tt);
|
||||
TypeId propType = freshType(arena, builtinTypes, scope);
|
||||
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
trackInteriorFreeType(scope, propType);
|
||||
|
||||
switch (context)
|
||||
{
|
||||
case ValueContext::RValue:
|
||||
|
|
|
@ -1,237 +1,14 @@
|
|||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||
#include "Luau/BuiltinDefinitions.h"
|
||||
|
||||
LUAU_FASTFLAG(LuauMathMap)
|
||||
LUAU_FASTFLAG(LuauVector2Constructor)
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitions)
|
||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitionsExtra)
|
||||
LUAU_FASTFLAG(LuauBufferBitMethods)
|
||||
LUAU_FASTFLAG(LuauVector2Constructor)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
||||
// TODO: there has to be a better way, like splitting up per library
|
||||
static const std::string kBuiltinDefinitionLuaSrcChecked_DEPRECATED = R"BUILTIN_SRC(
|
||||
|
||||
declare bit32: {
|
||||
band: @checked (...number) -> number,
|
||||
bor: @checked (...number) -> number,
|
||||
bxor: @checked (...number) -> number,
|
||||
btest: @checked (number, ...number) -> boolean,
|
||||
rrotate: @checked (x: number, disp: number) -> number,
|
||||
lrotate: @checked (x: number, disp: number) -> number,
|
||||
lshift: @checked (x: number, disp: number) -> number,
|
||||
arshift: @checked (x: number, disp: number) -> number,
|
||||
rshift: @checked (x: number, disp: number) -> number,
|
||||
bnot: @checked (x: number) -> number,
|
||||
extract: @checked (n: number, field: number, width: number?) -> number,
|
||||
replace: @checked (n: number, v: number, field: number, width: number?) -> number,
|
||||
countlz: @checked (n: number) -> number,
|
||||
countrz: @checked (n: number) -> number,
|
||||
byteswap: @checked (n: number) -> number,
|
||||
}
|
||||
|
||||
declare math: {
|
||||
frexp: @checked (n: number) -> (number, number),
|
||||
ldexp: @checked (s: number, e: number) -> number,
|
||||
fmod: @checked (x: number, y: number) -> number,
|
||||
modf: @checked (n: number) -> (number, number),
|
||||
pow: @checked (x: number, y: number) -> number,
|
||||
exp: @checked (n: number) -> number,
|
||||
|
||||
ceil: @checked (n: number) -> number,
|
||||
floor: @checked (n: number) -> number,
|
||||
abs: @checked (n: number) -> number,
|
||||
sqrt: @checked (n: number) -> number,
|
||||
|
||||
log: @checked (n: number, base: number?) -> number,
|
||||
log10: @checked (n: number) -> number,
|
||||
|
||||
rad: @checked (n: number) -> number,
|
||||
deg: @checked (n: number) -> number,
|
||||
|
||||
sin: @checked (n: number) -> number,
|
||||
cos: @checked (n: number) -> number,
|
||||
tan: @checked (n: number) -> number,
|
||||
sinh: @checked (n: number) -> number,
|
||||
cosh: @checked (n: number) -> number,
|
||||
tanh: @checked (n: number) -> number,
|
||||
atan: @checked (n: number) -> number,
|
||||
acos: @checked (n: number) -> number,
|
||||
asin: @checked (n: number) -> number,
|
||||
atan2: @checked (y: number, x: number) -> number,
|
||||
|
||||
min: @checked (number, ...number) -> number,
|
||||
max: @checked (number, ...number) -> number,
|
||||
|
||||
pi: number,
|
||||
huge: number,
|
||||
|
||||
randomseed: @checked (seed: number) -> (),
|
||||
random: @checked (number?, number?) -> number,
|
||||
|
||||
sign: @checked (n: number) -> number,
|
||||
clamp: @checked (n: number, min: number, max: number) -> number,
|
||||
noise: @checked (x: number, y: number?, z: number?) -> number,
|
||||
round: @checked (n: number) -> number,
|
||||
}
|
||||
|
||||
type DateTypeArg = {
|
||||
year: number,
|
||||
month: number,
|
||||
day: number,
|
||||
hour: number?,
|
||||
min: number?,
|
||||
sec: number?,
|
||||
isdst: boolean?,
|
||||
}
|
||||
|
||||
type DateTypeResult = {
|
||||
year: number,
|
||||
month: number,
|
||||
wday: number,
|
||||
yday: number,
|
||||
day: number,
|
||||
hour: number,
|
||||
min: number,
|
||||
sec: number,
|
||||
isdst: boolean,
|
||||
}
|
||||
|
||||
declare os: {
|
||||
time: (time: DateTypeArg?) -> number,
|
||||
date: ((formatString: "*t" | "!*t", time: number?) -> DateTypeResult) & ((formatString: string?, time: number?) -> string),
|
||||
difftime: (t2: DateTypeResult | number, t1: DateTypeResult | number) -> number,
|
||||
clock: () -> number,
|
||||
}
|
||||
|
||||
@checked declare function require(target: any): any
|
||||
|
||||
@checked declare function getfenv(target: any): { [string]: any }
|
||||
|
||||
declare _G: any
|
||||
declare _VERSION: string
|
||||
|
||||
declare function gcinfo(): number
|
||||
|
||||
declare function print<T...>(...: T...)
|
||||
|
||||
declare function type<T>(value: T): string
|
||||
declare function typeof<T>(value: T): string
|
||||
|
||||
-- `assert` has a magic function attached that will give more detailed type information
|
||||
declare function assert<T>(value: T, errorMessage: string?): T
|
||||
declare function error<T>(message: T, level: number?): never
|
||||
|
||||
declare function tostring<T>(value: T): string
|
||||
declare function tonumber<T>(value: T, radix: number?): number?
|
||||
|
||||
declare function rawequal<T1, T2>(a: T1, b: T2): boolean
|
||||
declare function rawget<K, V>(tab: {[K]: V}, k: K): V
|
||||
declare function rawset<K, V>(tab: {[K]: V}, k: K, v: V): {[K]: V}
|
||||
declare function rawlen<K, V>(obj: {[K]: V} | string): number
|
||||
|
||||
declare function setfenv<T..., R...>(target: number | (T...) -> R..., env: {[string]: any}): ((T...) -> R...)?
|
||||
|
||||
declare function ipairs<V>(tab: {V}): (({V}, number) -> (number?, V), {V}, number)
|
||||
|
||||
declare function pcall<A..., R...>(f: (A...) -> R..., ...: A...): (boolean, R...)
|
||||
|
||||
-- FIXME: The actual type of `xpcall` is:
|
||||
-- <E, A..., R1..., R2...>(f: (A...) -> R1..., err: (E) -> R2..., A...) -> (true, R1...) | (false, R2...)
|
||||
-- Since we can't represent the return value, we use (boolean, R1...).
|
||||
declare function xpcall<E, A..., R1..., R2...>(f: (A...) -> R1..., err: (E) -> R2..., ...: A...): (boolean, R1...)
|
||||
|
||||
-- `select` has a magic function attached to provide more detailed type information
|
||||
declare function select<A...>(i: string | number, ...: A...): ...any
|
||||
|
||||
-- FIXME: This type is not entirely correct - `loadstring` returns a function or
|
||||
-- (nil, string).
|
||||
declare function loadstring<A...>(src: string, chunkname: string?): (((A...) -> any)?, string?)
|
||||
|
||||
@checked declare function newproxy(mt: boolean?): any
|
||||
|
||||
declare coroutine: {
|
||||
create: <A..., R...>(f: (A...) -> R...) -> thread,
|
||||
resume: <A..., R...>(co: thread, A...) -> (boolean, R...),
|
||||
running: () -> thread,
|
||||
status: @checked (co: thread) -> "dead" | "running" | "normal" | "suspended",
|
||||
wrap: <A..., R...>(f: (A...) -> R...) -> ((A...) -> R...),
|
||||
yield: <A..., R...>(A...) -> R...,
|
||||
isyieldable: () -> boolean,
|
||||
close: @checked (co: thread) -> (boolean, any)
|
||||
}
|
||||
|
||||
declare table: {
|
||||
concat: <V>(t: {V}, sep: string?, i: number?, j: number?) -> string,
|
||||
insert: (<V>(t: {V}, value: V) -> ()) & (<V>(t: {V}, pos: number, value: V) -> ()),
|
||||
maxn: <V>(t: {V}) -> number,
|
||||
remove: <V>(t: {V}, number?) -> V?,
|
||||
sort: <V>(t: {V}, comp: ((V, V) -> boolean)?) -> (),
|
||||
create: <V>(count: number, value: V?) -> {V},
|
||||
find: <V>(haystack: {V}, needle: V, init: number?) -> number?,
|
||||
|
||||
unpack: <V>(list: {V}, i: number?, j: number?) -> ...V,
|
||||
pack: <V>(...V) -> { n: number, [number]: V },
|
||||
|
||||
getn: <V>(t: {V}) -> number,
|
||||
foreach: <K, V>(t: {[K]: V}, f: (K, V) -> ()) -> (),
|
||||
foreachi: <V>({V}, (number, V) -> ()) -> (),
|
||||
|
||||
move: <V>(src: {V}, a: number, b: number, t: number, dst: {V}?) -> {V},
|
||||
clear: <K, V>(table: {[K]: V}) -> (),
|
||||
|
||||
isfrozen: <K, V>(t: {[K]: V}) -> boolean,
|
||||
}
|
||||
|
||||
declare debug: {
|
||||
info: (<R...>(thread: thread, level: number, options: string) -> R...) & (<R...>(level: number, options: string) -> R...) & (<A..., R1..., R2...>(func: (A...) -> R1..., options: string) -> R2...),
|
||||
traceback: ((message: string?, level: number?) -> string) & ((thread: thread, message: string?, level: number?) -> string),
|
||||
}
|
||||
|
||||
declare utf8: {
|
||||
char: @checked (...number) -> string,
|
||||
charpattern: string,
|
||||
codes: @checked (str: string) -> ((string, number) -> (number, number), string, number),
|
||||
codepoint: @checked (str: string, i: number?, j: number?) -> ...number,
|
||||
len: @checked (s: string, i: number?, j: number?) -> (number?, number?),
|
||||
offset: @checked (s: string, n: number?, i: number?) -> number,
|
||||
}
|
||||
|
||||
-- Cannot use `typeof` here because it will produce a polytype when we expect a monotype.
|
||||
declare function unpack<V>(tab: {V}, i: number?, j: number?): ...V
|
||||
|
||||
|
||||
--- Buffer API
|
||||
declare buffer: {
|
||||
create: @checked (size: number) -> buffer,
|
||||
fromstring: @checked (str: string) -> buffer,
|
||||
tostring: @checked (b: buffer) -> string,
|
||||
len: @checked (b: buffer) -> number,
|
||||
copy: @checked (target: buffer, targetOffset: number, source: buffer, sourceOffset: number?, count: number?) -> (),
|
||||
fill: @checked (b: buffer, offset: number, value: number, count: number?) -> (),
|
||||
readi8: @checked (b: buffer, offset: number) -> number,
|
||||
readu8: @checked (b: buffer, offset: number) -> number,
|
||||
readi16: @checked (b: buffer, offset: number) -> number,
|
||||
readu16: @checked (b: buffer, offset: number) -> number,
|
||||
readi32: @checked (b: buffer, offset: number) -> number,
|
||||
readu32: @checked (b: buffer, offset: number) -> number,
|
||||
readf32: @checked (b: buffer, offset: number) -> number,
|
||||
readf64: @checked (b: buffer, offset: number) -> number,
|
||||
writei8: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writeu8: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writei16: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writeu16: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writei32: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writeu32: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writef32: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writef64: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
readstring: @checked (b: buffer, offset: number, count: number) -> string,
|
||||
writestring: @checked (b: buffer, offset: number, value: string, count: number?) -> (),
|
||||
}
|
||||
|
||||
)BUILTIN_SRC";
|
||||
|
||||
static const std::string kBuiltinDefinitionLuaSrcChecked = R"BUILTIN_SRC(
|
||||
|
||||
declare bit32: {
|
||||
|
@ -423,7 +200,9 @@ declare utf8: {
|
|||
-- Cannot use `typeof` here because it will produce a polytype when we expect a monotype.
|
||||
declare function unpack<V>(tab: {V}, i: number?, j: number?): ...V
|
||||
|
||||
)BUILTIN_SRC";
|
||||
|
||||
static const std::string kBuiltinDefinitionBufferSrc_DEPRECATED = R"BUILTIN_SRC(
|
||||
--- Buffer API
|
||||
declare buffer: {
|
||||
create: @checked (size: number) -> buffer,
|
||||
|
@ -454,6 +233,39 @@ declare buffer: {
|
|||
|
||||
)BUILTIN_SRC";
|
||||
|
||||
static const std::string kBuiltinDefinitionBufferSrc = R"BUILTIN_SRC(
|
||||
--- Buffer API
|
||||
declare buffer: {
|
||||
create: @checked (size: number) -> buffer,
|
||||
fromstring: @checked (str: string) -> buffer,
|
||||
tostring: @checked (b: buffer) -> string,
|
||||
len: @checked (b: buffer) -> number,
|
||||
copy: @checked (target: buffer, targetOffset: number, source: buffer, sourceOffset: number?, count: number?) -> (),
|
||||
fill: @checked (b: buffer, offset: number, value: number, count: number?) -> (),
|
||||
readi8: @checked (b: buffer, offset: number) -> number,
|
||||
readu8: @checked (b: buffer, offset: number) -> number,
|
||||
readi16: @checked (b: buffer, offset: number) -> number,
|
||||
readu16: @checked (b: buffer, offset: number) -> number,
|
||||
readi32: @checked (b: buffer, offset: number) -> number,
|
||||
readu32: @checked (b: buffer, offset: number) -> number,
|
||||
readf32: @checked (b: buffer, offset: number) -> number,
|
||||
readf64: @checked (b: buffer, offset: number) -> number,
|
||||
writei8: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writeu8: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writei16: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writeu16: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writei32: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writeu32: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writef32: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
writef64: @checked (b: buffer, offset: number, value: number) -> (),
|
||||
readstring: @checked (b: buffer, offset: number, count: number) -> string,
|
||||
writestring: @checked (b: buffer, offset: number, value: string, count: number?) -> (),
|
||||
readbits: @checked (b: buffer, bitOffset: number, bitCount: number) -> number,
|
||||
writebits: @checked (b: buffer, bitOffset: number, bitCount: number, value: number) -> (),
|
||||
}
|
||||
|
||||
)BUILTIN_SRC";
|
||||
|
||||
static const std::string kBuiltinDefinitionVectorSrc_NoExtra_NoVector2Ctor_DEPRECATED = R"BUILTIN_SRC(
|
||||
|
||||
-- TODO: this will be replaced with a built-in primitive type
|
||||
|
@ -568,16 +380,24 @@ declare vector: {
|
|||
|
||||
std::string getBuiltinDefinitionSource()
|
||||
{
|
||||
std::string result = FFlag::LuauMathMap ? kBuiltinDefinitionLuaSrcChecked : kBuiltinDefinitionLuaSrcChecked_DEPRECATED;
|
||||
std::string result = kBuiltinDefinitionLuaSrcChecked;
|
||||
|
||||
if (FFlag::LuauVectorDefinitionsExtra && FFlag::LuauVector2Constructor)
|
||||
result += kBuiltinDefinitionVectorSrc;
|
||||
else if (FFlag::LuauVectorDefinitionsExtra && !FFlag::LuauVector2Constructor)
|
||||
result += kBuiltinDefinitionVectorSrc_NoVector2Ctor_DEPRECATED;
|
||||
else if (FFlag::LuauVectorDefinitions && FFlag::LuauVector2Constructor)
|
||||
result += kBuiltinDefinitionVectorSrc_NoExtra_DEPRECATED;
|
||||
else if (FFlag::LuauVectorDefinitions && !FFlag::LuauVector2Constructor)
|
||||
result += kBuiltinDefinitionVectorSrc_NoExtra_NoVector2Ctor_DEPRECATED;
|
||||
result += FFlag::LuauBufferBitMethods ? kBuiltinDefinitionBufferSrc : kBuiltinDefinitionBufferSrc_DEPRECATED;
|
||||
|
||||
if (FFlag::LuauVectorDefinitionsExtra)
|
||||
{
|
||||
if (FFlag::LuauVector2Constructor)
|
||||
result += kBuiltinDefinitionVectorSrc;
|
||||
else
|
||||
result += kBuiltinDefinitionVectorSrc_NoVector2Ctor_DEPRECATED;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (FFlag::LuauVector2Constructor)
|
||||
result += kBuiltinDefinitionVectorSrc_NoExtra_DEPRECATED;
|
||||
else
|
||||
result += kBuiltinDefinitionVectorSrc_NoExtra_NoVector2Ctor_DEPRECATED;
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
|
|
@ -193,9 +193,8 @@ static bool areTerminalAndDefinitelyDisjoint(const EType& lhs, const EType& rhs)
|
|||
// - Whether one of the enodes is a large semantic set such as TAny,
|
||||
// TUnknown, or TError.
|
||||
return !(
|
||||
lhs.index() == rhs.index() ||
|
||||
lhs.get<TUnknown>() || rhs.get<TUnknown>() || lhs.get<TAny>() || rhs.get<TAny>() || lhs.get<TNoRefine>() || rhs.get<TNoRefine>() ||
|
||||
lhs.get<TError>() || rhs.get<TError>() || lhs.get<TOpaque>() || rhs.get<TOpaque>()
|
||||
lhs.index() == rhs.index() || lhs.get<TUnknown>() || rhs.get<TUnknown>() || lhs.get<TAny>() || rhs.get<TAny>() || lhs.get<TNoRefine>() ||
|
||||
rhs.get<TNoRefine>() || lhs.get<TError>() || rhs.get<TError>() || lhs.get<TOpaque>() || rhs.get<TOpaque>()
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -694,7 +693,8 @@ TypeId flattenTableNode(
|
|||
StringId propName = t->propNames[i];
|
||||
const Id propType = t->propTypes()[i];
|
||||
|
||||
resultTable.props[strings.asString(propName)] = Property{fromId(egraph, strings, builtinTypes, arena, bestNodes, seen, newTypeFunctions, propType)};
|
||||
resultTable.props[strings.asString(propName)] =
|
||||
Property{fromId(egraph, strings, builtinTypes, arena, bestNodes, seen, newTypeFunctions, propType)};
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -937,12 +937,20 @@ std::string mkDesc(
|
|||
const int RULE_PADDING = 35;
|
||||
const std::string rulePadding(std::max<size_t>(0, RULE_PADDING - rule.size()), ' ');
|
||||
const std::string fromIdStr = ""; // "(" + std::to_string(uint32_t(from)) + ") ";
|
||||
const std::string toIdStr = ""; // "(" + std::to_string(uint32_t(to)) + ") ";
|
||||
const std::string toIdStr = ""; // "(" + std::to_string(uint32_t(to)) + ") ";
|
||||
|
||||
return rule + ":" + rulePadding + fromIdStr + toString(fromTy, opts) + " <=> " + toIdStr + toString(toTy, opts);
|
||||
}
|
||||
|
||||
std::string mkDesc(EGraph& egraph, const StringCache& strings, NotNull<TypeArena> arena, NotNull<BuiltinTypes> builtinTypes, Id from, Id to, const std::string& rule)
|
||||
std::string mkDesc(
|
||||
EGraph& egraph,
|
||||
const StringCache& strings,
|
||||
NotNull<TypeArena> arena,
|
||||
NotNull<BuiltinTypes> builtinTypes,
|
||||
Id from,
|
||||
Id to,
|
||||
const std::string& rule
|
||||
)
|
||||
{
|
||||
if (!FFlag::DebugLuauLogSimplification)
|
||||
return "";
|
||||
|
@ -1879,7 +1887,12 @@ void Simplifier::intersectWithNegatedClass(Id id)
|
|||
isTag<SBoolean>(iNode) || isTag<SString>(iNode) || isTag<TFunction>(iNode) || isTag<TNever>(iNode))
|
||||
{
|
||||
// eg string & ~SomeClass
|
||||
subst(id, iId, "intersectClassWithNegatedClass", {{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}});
|
||||
subst(
|
||||
id,
|
||||
iId,
|
||||
"intersectClassWithNegatedClass",
|
||||
{{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}}
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
|
@ -1887,27 +1900,37 @@ void Simplifier::intersectWithNegatedClass(Id id)
|
|||
{
|
||||
switch (relateClasses(class_, negatedClass))
|
||||
{
|
||||
case LeftSuper:
|
||||
// eg Instance & ~Part
|
||||
// This cannot be meaningfully reduced.
|
||||
continue;
|
||||
case RightSuper:
|
||||
subst(id, egraph.add(TNever{}), "intersectClassWithNegatedClass", {{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}});
|
||||
return;
|
||||
case Unrelated:
|
||||
// Part & ~Folder == Part
|
||||
case LeftSuper:
|
||||
// eg Instance & ~Part
|
||||
// This cannot be meaningfully reduced.
|
||||
continue;
|
||||
case RightSuper:
|
||||
subst(
|
||||
id,
|
||||
egraph.add(TNever{}),
|
||||
"intersectClassWithNegatedClass",
|
||||
{{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}}
|
||||
);
|
||||
return;
|
||||
case Unrelated:
|
||||
// Part & ~Folder == Part
|
||||
{
|
||||
std::vector<Id> newParts;
|
||||
newParts.reserve(intersection->operands().size() - 1);
|
||||
for (Id part : intersection->operands())
|
||||
{
|
||||
std::vector<Id> newParts;
|
||||
newParts.reserve(intersection->operands().size() - 1);
|
||||
for (Id part : intersection->operands())
|
||||
{
|
||||
if (part != jId)
|
||||
newParts.push_back(part);
|
||||
}
|
||||
|
||||
Id substId = egraph.add(Intersection{newParts.begin(), newParts.end()});
|
||||
subst(id, substId, "intersectClassWithNegatedClass", {{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}});
|
||||
if (part != jId)
|
||||
newParts.push_back(part);
|
||||
}
|
||||
|
||||
Id substId = egraph.add(Intersection{newParts.begin(), newParts.end()});
|
||||
subst(
|
||||
id,
|
||||
substId,
|
||||
"intersectClassWithNegatedClass",
|
||||
{{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}}
|
||||
);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -245,7 +245,6 @@ FragmentParseResult parseFragment(
|
|||
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);
|
||||
|
||||
std::vector<AstNode*> fabricatedAncestry = std::move(result.ancestry);
|
||||
|
||||
// Get the ancestry for the fragment at the offset cursor position.
|
||||
|
@ -258,6 +257,7 @@ FragmentParseResult parseFragment(
|
|||
fragmentResult.root = std::move(p.root);
|
||||
fragmentResult.ancestry = std::move(fabricatedAncestry);
|
||||
fragmentResult.nearestStatement = nearestStatement;
|
||||
fragmentResult.commentLocations = std::move(p.commentLocations);
|
||||
return fragmentResult;
|
||||
}
|
||||
|
||||
|
@ -444,7 +444,7 @@ FragmentTypeCheckResult typecheckFragment_(
|
|||
}
|
||||
|
||||
|
||||
FragmentTypeCheckResult typecheckFragment(
|
||||
std::pair<FragmentTypeCheckStatus, FragmentTypeCheckResult> typecheckFragment(
|
||||
Frontend& frontend,
|
||||
const ModuleName& moduleName,
|
||||
const Position& cursorPos,
|
||||
|
@ -469,12 +469,15 @@ FragmentTypeCheckResult typecheckFragment(
|
|||
}
|
||||
|
||||
FragmentParseResult parseResult = parseFragment(*sourceModule, src, cursorPos, fragmentEndPosition);
|
||||
if (isWithinComment(parseResult.commentLocations, fragmentEndPosition.value_or(cursorPos)))
|
||||
return {FragmentTypeCheckStatus::SkipAutocomplete, {}};
|
||||
|
||||
FrontendOptions frontendOptions = opts.value_or(frontend.options);
|
||||
const ScopePtr& closestScope = findClosestScope(module, parseResult.nearestStatement);
|
||||
FragmentTypeCheckResult result =
|
||||
typecheckFragment_(frontend, parseResult.root, module, closestScope, cursorPos, std::move(parseResult.alloc), frontendOptions);
|
||||
result.ancestry = std::move(parseResult.ancestry);
|
||||
return result;
|
||||
return {FragmentTypeCheckStatus::Success, result};
|
||||
}
|
||||
|
||||
|
||||
|
@ -498,7 +501,14 @@ FragmentAutocompleteResult fragmentAutocomplete(
|
|||
return {};
|
||||
}
|
||||
|
||||
auto tcResult = typecheckFragment(frontend, moduleName, cursorPosition, opts, src, fragmentEndPosition);
|
||||
// If the cursor is within a comment in the stale source module we should avoid providing a recommendation
|
||||
if (isWithinComment(*sourceModule, fragmentEndPosition.value_or(cursorPosition)))
|
||||
return {};
|
||||
|
||||
auto [tcStatus, tcResult] = typecheckFragment(frontend, moduleName, cursorPosition, opts, src, fragmentEndPosition);
|
||||
if (tcStatus == FragmentTypeCheckStatus::SkipAutocomplete)
|
||||
return {};
|
||||
|
||||
auto globalScope = (opts && opts->forAutocomplete) ? frontend.globalsForAutocomplete.globalScope.get() : frontend.globals.globalScope.get();
|
||||
|
||||
TypeArena arenaForFragmentAutocomplete;
|
||||
|
|
|
@ -977,7 +977,8 @@ struct TypeCacher : TypeOnceVisitor
|
|||
return false;
|
||||
}
|
||||
|
||||
bool visit(TypePackId tp, const BoundTypePack& btp) override {
|
||||
bool visit(TypePackId tp, const BoundTypePack& btp) override
|
||||
{
|
||||
traverse(btp.boundTo);
|
||||
if (isUncacheable(btp.boundTo))
|
||||
markUncacheable(tp);
|
||||
|
|
|
@ -15,11 +15,12 @@
|
|||
#include <algorithm>
|
||||
|
||||
LUAU_FASTFLAG(LuauSolverV2);
|
||||
LUAU_FASTFLAGVARIABLE(LuauIncrementalAutocompleteCommentDetection)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
||||
static bool contains(Position pos, Comment comment)
|
||||
static bool contains_DEPRECATED(Position pos, Comment comment)
|
||||
{
|
||||
if (comment.location.contains(pos))
|
||||
return true;
|
||||
|
@ -32,7 +33,22 @@ static bool contains(Position pos, Comment comment)
|
|||
return false;
|
||||
}
|
||||
|
||||
static bool isWithinComment(const std::vector<Comment>& commentLocations, Position pos)
|
||||
static bool contains(Position pos, Comment comment)
|
||||
{
|
||||
if (comment.location.contains(pos))
|
||||
return true;
|
||||
else if (comment.type == Lexeme::BrokenComment && comment.location.begin <= pos) // Broken comments are broken specifically because they don't
|
||||
// have an end
|
||||
return true;
|
||||
// comments actually span the whole line - in incremental mode, we could pass a cursor outside of the current parsed comment range span, but it
|
||||
// would still be 'within' the comment So, the cursor must be on the same line and the comment itself must come strictly after the `begin`
|
||||
else if (comment.type == Lexeme::Comment && comment.location.end.line == pos.line && comment.location.begin <= pos)
|
||||
return true;
|
||||
else
|
||||
return false;
|
||||
}
|
||||
|
||||
bool isWithinComment(const std::vector<Comment>& commentLocations, Position pos)
|
||||
{
|
||||
auto iter = std::lower_bound(
|
||||
commentLocations.begin(),
|
||||
|
@ -40,6 +56,11 @@ static bool isWithinComment(const std::vector<Comment>& commentLocations, Positi
|
|||
Comment{Lexeme::Comment, Location{pos, pos}},
|
||||
[](const Comment& a, const Comment& b)
|
||||
{
|
||||
if (FFlag::LuauIncrementalAutocompleteCommentDetection)
|
||||
{
|
||||
if (a.type == Lexeme::Comment)
|
||||
return a.location.end.line < b.location.end.line;
|
||||
}
|
||||
return a.location.end < b.location.end;
|
||||
}
|
||||
);
|
||||
|
@ -47,7 +68,7 @@ static bool isWithinComment(const std::vector<Comment>& commentLocations, Positi
|
|||
if (iter == commentLocations.end())
|
||||
return false;
|
||||
|
||||
if (contains(pos, *iter))
|
||||
if (FFlag::LuauIncrementalAutocompleteCommentDetection ? contains(pos, *iter) : contains_DEPRECATED(pos, *iter))
|
||||
return true;
|
||||
|
||||
// Due to the nature of std::lower_bound, it is possible that iter points at a comment that ends
|
||||
|
@ -149,9 +170,9 @@ struct ClonePublicInterface : Substitution
|
|||
freety->scope->location,
|
||||
module->name,
|
||||
InternalError{"Free type is escaping its module; please report this bug at "
|
||||
"https://github.com/luau-lang/luau/issues"}
|
||||
);
|
||||
result = builtinTypes->errorRecoveryType();
|
||||
"https://github.com/luau-lang/luau/issues"}
|
||||
);
|
||||
result = builtinTypes->errorRecoveryType();
|
||||
}
|
||||
else if (auto genericty = getMutable<GenericType>(result))
|
||||
{
|
||||
|
@ -173,8 +194,8 @@ struct ClonePublicInterface : Substitution
|
|||
ftp->scope->location,
|
||||
module->name,
|
||||
InternalError{"Free type pack is escaping its module; please report this bug at "
|
||||
"https://github.com/luau-lang/luau/issues"}
|
||||
);
|
||||
"https://github.com/luau-lang/luau/issues"}
|
||||
);
|
||||
clonedTp = builtinTypes->errorRecoveryTypePack();
|
||||
}
|
||||
else if (auto gtp = getMutable<GenericTypePack>(clonedTp))
|
||||
|
|
|
@ -1809,7 +1809,8 @@ NormalizationResult Normalizer::unionNormalWithTy(
|
|||
}
|
||||
else if (get<UnknownType>(here.tops))
|
||||
return NormalizationResult::True;
|
||||
else if (get<GenericType>(there) || get<FreeType>(there) || get<BlockedType>(there) || get<PendingExpansionType>(there) || get<TypeFunctionInstanceType>(there))
|
||||
else if (get<GenericType>(there) || get<FreeType>(there) || get<BlockedType>(there) || get<PendingExpansionType>(there) ||
|
||||
get<TypeFunctionInstanceType>(there))
|
||||
{
|
||||
if (tyvarIndex(there) <= ignoreSmallerTyvars)
|
||||
return NormalizationResult::True;
|
||||
|
@ -3162,7 +3163,8 @@ NormalizationResult Normalizer::intersectNormalWithTy(
|
|||
}
|
||||
return NormalizationResult::True;
|
||||
}
|
||||
else if (get<GenericType>(there) || get<FreeType>(there) || get<BlockedType>(there) || get<PendingExpansionType>(there) || get<TypeFunctionInstanceType>(there))
|
||||
else if (get<GenericType>(there) || get<FreeType>(there) || get<BlockedType>(there) || get<PendingExpansionType>(there) ||
|
||||
get<TypeFunctionInstanceType>(there))
|
||||
{
|
||||
NormalizedType thereNorm{builtinTypes};
|
||||
NormalizedType topNorm{builtinTypes};
|
||||
|
|
|
@ -420,7 +420,8 @@ static std::optional<TypeId> selectOverload(
|
|||
TypePackId argsPack
|
||||
)
|
||||
{
|
||||
auto resolver = std::make_unique<OverloadResolver>(builtinTypes, arena, simplifier, normalizer, typeFunctionRuntime, scope, iceReporter, limits, location);
|
||||
auto resolver =
|
||||
std::make_unique<OverloadResolver>(builtinTypes, arena, simplifier, normalizer, typeFunctionRuntime, scope, iceReporter, limits, location);
|
||||
auto [status, overload] = resolver->selectOverload(fn, argsPack);
|
||||
|
||||
if (status == OverloadResolver::Analysis::Ok)
|
||||
|
|
|
@ -1480,9 +1480,8 @@ SubtypingResult Subtyping::isCovariantWith(
|
|||
|
||||
if (auto variadic = get<VariadicTypePack>(tail); variadic && variadic->hidden)
|
||||
{
|
||||
result.orElse(
|
||||
isContravariantWith(env, subFunction->argTypes, arena->addTypePack(TypePack{arguments}), scope).withBothComponent(TypePath::PackField::Arguments)
|
||||
);
|
||||
result.orElse(isContravariantWith(env, subFunction->argTypes, arena->addTypePack(TypePack{arguments}), scope)
|
||||
.withBothComponent(TypePath::PackField::Arguments));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1020,7 +1020,8 @@ void TypeChecker2::visit(AstStatForIn* forInStatement)
|
|||
{
|
||||
reportError(OptionalValueAccess{iteratorTy}, forInStatement->values.data[0]->location);
|
||||
}
|
||||
else if (std::optional<TypeId> iterMmTy = findMetatableEntry(builtinTypes, module->errors, iteratorTy, "__iter", forInStatement->values.data[0]->location))
|
||||
else if (std::optional<TypeId> iterMmTy =
|
||||
findMetatableEntry(builtinTypes, module->errors, iteratorTy, "__iter", forInStatement->values.data[0]->location))
|
||||
{
|
||||
Instantiation instantiation{TxnLog::empty(), &arena, builtinTypes, TypeLevel{}, scope};
|
||||
|
||||
|
|
|
@ -832,7 +832,12 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
|||
{
|
||||
if (FFlag::LuauUserTypeFunPrintToError)
|
||||
return {
|
||||
std::nullopt, Reduction::Erroneous, {}, {}, format("'%s' type function: returned a non-type value", name.value), ctx->typeFunctionRuntime->messages
|
||||
std::nullopt,
|
||||
Reduction::Erroneous,
|
||||
{},
|
||||
{},
|
||||
format("'%s' type function: returned a non-type value", name.value),
|
||||
ctx->typeFunctionRuntime->messages
|
||||
};
|
||||
else
|
||||
return {std::nullopt, Reduction::Erroneous, {}, {}, format("'%s' type function: returned a non-type value", name.value)};
|
||||
|
@ -2064,7 +2069,7 @@ TypeFunctionReductionResult<TypeId> refineTypeFunction(
|
|||
if (ctx->solver)
|
||||
{
|
||||
for (TypeId newTf : simplifyResult->newTypeFunctions)
|
||||
ctx->solver->pushConstraint(ctx->scope, ctx->constraint->location, ReduceConstraint{newTf});
|
||||
ctx->pushConstraint(ReduceConstraint{newTf});
|
||||
}
|
||||
|
||||
return {simplifyResult->result, {}};
|
||||
|
|
|
@ -14,6 +14,7 @@
|
|||
#include <vector>
|
||||
|
||||
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixInner)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunPrintToError)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixNoReadWrite)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunThreadBuffer)
|
||||
|
@ -413,10 +414,21 @@ static int getNegatedValue(lua_State* L)
|
|||
luaL_error(L, "type.inner: expected 1 argument, but got %d", argumentCount);
|
||||
|
||||
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
||||
if (auto tfnt = get<TypeFunctionNegationType>(self); !tfnt)
|
||||
allocTypeUserData(L, tfnt->type->type);
|
||||
|
||||
if (FFlag::LuauUserTypeFunFixInner)
|
||||
{
|
||||
if (auto tfnt = get<TypeFunctionNegationType>(self); tfnt)
|
||||
allocTypeUserData(L, tfnt->type->type);
|
||||
else
|
||||
luaL_error(L, "type.inner: cannot call inner method on non-negation type: `%s` type", getTag(L, self).c_str());
|
||||
}
|
||||
else
|
||||
luaL_error(L, "type.inner: cannot call inner method on non-negation type: `%s` type", getTag(L, self).c_str());
|
||||
{
|
||||
if (auto tfnt = get<TypeFunctionNegationType>(self); !tfnt)
|
||||
allocTypeUserData(L, tfnt->type->type);
|
||||
else
|
||||
luaL_error(L, "type.inner: cannot call inner method on non-negation type: `%s` type", getTag(L, self).c_str());
|
||||
}
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
|
|
@ -464,7 +464,9 @@ public:
|
|||
, typeFunctionRuntime(state->ctx->typeFunctionRuntime)
|
||||
, queue({})
|
||||
, types({})
|
||||
, packs({}){};
|
||||
, packs({})
|
||||
{
|
||||
}
|
||||
|
||||
TypeId deserialize(TypeFunctionTypeId ty)
|
||||
{
|
||||
|
|
|
@ -2799,10 +2799,10 @@ TypeId TypeChecker::checkRelationalOperation(
|
|||
reportError(
|
||||
expr.location,
|
||||
GenericError{
|
||||
format("Type '%s' cannot be compared with relational operator %s", toString(leftType).c_str(), toString(expr.op).c_str())
|
||||
}
|
||||
);
|
||||
}
|
||||
format("Type '%s' cannot be compared with relational operator %s", toString(leftType).c_str(), toString(expr.op).c_str())
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
return booleanType;
|
||||
}
|
||||
|
|
|
@ -11,6 +11,7 @@
|
|||
|
||||
LUAU_FASTFLAG(LuauSolverV2);
|
||||
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete);
|
||||
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope);
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -318,6 +319,8 @@ TypePack extendTypePack(
|
|||
{
|
||||
FreeType ft{ftp->scope, builtinTypes->neverType, builtinTypes->unknownType};
|
||||
t = arena.addType(ft);
|
||||
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||
trackInteriorFreeType(ftp->scope, t);
|
||||
}
|
||||
else
|
||||
t = arena.freshType(ftp->scope);
|
||||
|
@ -533,7 +536,7 @@ std::vector<TypeId> findBlockedArgTypesIn(AstExprCall* expr, NotNull<DenseHashMa
|
|||
{
|
||||
std::vector<TypeId> toBlock;
|
||||
BlockedTypeInLiteralVisitor v{astTypes, NotNull{&toBlock}};
|
||||
for (auto arg: expr->args)
|
||||
for (auto arg : expr->args)
|
||||
{
|
||||
if (isLiteral(arg) || arg->is<AstExprGroup>())
|
||||
{
|
||||
|
@ -543,5 +546,21 @@ std::vector<TypeId> findBlockedArgTypesIn(AstExprCall* expr, NotNull<DenseHashMa
|
|||
return toBlock;
|
||||
}
|
||||
|
||||
void trackInteriorFreeType(Scope* scope, TypeId ty)
|
||||
{
|
||||
LUAU_ASSERT(FFlag::LuauSolverV2 && FFlag::LuauTrackInteriorFreeTypesOnScope);
|
||||
for (; scope; scope = scope->parent.get())
|
||||
{
|
||||
if (scope->interiorFreeTypes)
|
||||
{
|
||||
scope->interiorFreeTypes->push_back(ty);
|
||||
return;
|
||||
}
|
||||
}
|
||||
// There should at least be *one* generalization constraint per module
|
||||
// where `interiorFreeTypes` is present, which would be the one made
|
||||
// by ConstraintGenerator::visitModuleRoot.
|
||||
LUAU_ASSERT(!"No scopes in parent chain had a present `interiorFreeTypes` member.");
|
||||
}
|
||||
|
||||
} // namespace Luau
|
||||
|
|
|
@ -45,4 +45,4 @@ private:
|
|||
size_t offset;
|
||||
};
|
||||
|
||||
}
|
||||
} // namespace Luau
|
||||
|
|
|
@ -14,12 +14,37 @@ struct Position
|
|||
{
|
||||
}
|
||||
|
||||
bool operator==(const Position& rhs) const;
|
||||
bool operator!=(const Position& rhs) const;
|
||||
bool operator<(const Position& rhs) const;
|
||||
bool operator>(const Position& rhs) const;
|
||||
bool operator<=(const Position& rhs) const;
|
||||
bool operator>=(const Position& rhs) const;
|
||||
bool operator==(const Position& rhs) const
|
||||
{
|
||||
return this->column == rhs.column && this->line == rhs.line;
|
||||
}
|
||||
|
||||
bool operator!=(const Position& rhs) const
|
||||
{
|
||||
return !(*this == rhs);
|
||||
}
|
||||
bool operator<(const Position& rhs) const
|
||||
{
|
||||
if (line == rhs.line)
|
||||
return column < rhs.column;
|
||||
else
|
||||
return line < rhs.line;
|
||||
}
|
||||
bool operator>(const Position& rhs) const
|
||||
{
|
||||
if (line == rhs.line)
|
||||
return column > rhs.column;
|
||||
else
|
||||
return line > rhs.line;
|
||||
}
|
||||
bool operator<=(const Position& rhs) const
|
||||
{
|
||||
return *this == rhs || *this < rhs;
|
||||
}
|
||||
bool operator>=(const Position& rhs) const
|
||||
{
|
||||
return *this == rhs || *this > rhs;
|
||||
}
|
||||
|
||||
void shift(const Position& start, const Position& oldEnd, const Position& newEnd);
|
||||
};
|
||||
|
@ -52,8 +77,14 @@ struct Location
|
|||
{
|
||||
}
|
||||
|
||||
bool operator==(const Location& rhs) const;
|
||||
bool operator!=(const Location& rhs) const;
|
||||
bool operator==(const Location& rhs) const
|
||||
{
|
||||
return this->begin == rhs.begin && this->end == rhs.end;
|
||||
}
|
||||
bool operator!=(const Location& rhs) const
|
||||
{
|
||||
return !(*this == rhs);
|
||||
}
|
||||
|
||||
bool encloses(const Location& l) const;
|
||||
bool overlaps(const Location& l) const;
|
||||
|
|
|
@ -63,4 +63,4 @@ void* Allocator::allocate(size_t size)
|
|||
return page->data;
|
||||
}
|
||||
|
||||
}
|
||||
} // namespace Luau
|
||||
|
|
|
@ -1151,10 +1151,7 @@ void AstTypePackGeneric::visit(AstVisitor* visitor)
|
|||
|
||||
bool isLValue(const AstExpr* expr)
|
||||
{
|
||||
return expr->is<AstExprLocal>()
|
||||
|| expr->is<AstExprGlobal>()
|
||||
|| expr->is<AstExprIndexName>()
|
||||
|| expr->is<AstExprIndexExpr>();
|
||||
return expr->is<AstExprLocal>() || expr->is<AstExprGlobal>() || expr->is<AstExprIndexName>() || expr->is<AstExprIndexExpr>();
|
||||
}
|
||||
|
||||
AstName getIdentifier(AstExpr* node)
|
||||
|
|
|
@ -9,6 +9,8 @@
|
|||
#include <limits.h>
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LexerResumesFromPosition2)
|
||||
LUAU_FASTFLAGVARIABLE(LexerFixInterpStringStart)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
||||
|
@ -759,7 +761,7 @@ Lexeme Lexer::readNext()
|
|||
return Lexeme(Location(start, 1), '}');
|
||||
}
|
||||
|
||||
return readInterpolatedStringSection(position(), Lexeme::InterpStringMid, Lexeme::InterpStringEnd);
|
||||
return readInterpolatedStringSection(FFlag::LexerFixInterpStringStart ? start : position(), Lexeme::InterpStringMid, Lexeme::InterpStringEnd);
|
||||
}
|
||||
|
||||
case '=':
|
||||
|
|
|
@ -4,42 +4,6 @@
|
|||
namespace Luau
|
||||
{
|
||||
|
||||
bool Position::operator==(const Position& rhs) const
|
||||
{
|
||||
return this->column == rhs.column && this->line == rhs.line;
|
||||
}
|
||||
|
||||
bool Position::operator!=(const Position& rhs) const
|
||||
{
|
||||
return !(*this == rhs);
|
||||
}
|
||||
|
||||
bool Position::operator<(const Position& rhs) const
|
||||
{
|
||||
if (line == rhs.line)
|
||||
return column < rhs.column;
|
||||
else
|
||||
return line < rhs.line;
|
||||
}
|
||||
|
||||
bool Position::operator>(const Position& rhs) const
|
||||
{
|
||||
if (line == rhs.line)
|
||||
return column > rhs.column;
|
||||
else
|
||||
return line > rhs.line;
|
||||
}
|
||||
|
||||
bool Position::operator<=(const Position& rhs) const
|
||||
{
|
||||
return *this == rhs || *this < rhs;
|
||||
}
|
||||
|
||||
bool Position::operator>=(const Position& rhs) const
|
||||
{
|
||||
return *this == rhs || *this > rhs;
|
||||
}
|
||||
|
||||
void Position::shift(const Position& start, const Position& oldEnd, const Position& newEnd)
|
||||
{
|
||||
if (*this >= start)
|
||||
|
@ -54,16 +18,6 @@ void Position::shift(const Position& start, const Position& oldEnd, const Positi
|
|||
}
|
||||
}
|
||||
|
||||
bool Location::operator==(const Location& rhs) const
|
||||
{
|
||||
return this->begin == rhs.begin && this->end == rhs.end;
|
||||
}
|
||||
|
||||
bool Location::operator!=(const Location& rhs) const
|
||||
{
|
||||
return !(*this == rhs);
|
||||
}
|
||||
|
||||
bool Location::encloses(const Location& l) const
|
||||
{
|
||||
return begin <= l.begin && end >= l.end;
|
||||
|
|
|
@ -18,7 +18,6 @@ LUAU_FASTINTVARIABLE(LuauParseErrorLimit, 100)
|
|||
// flag so that we don't break production games by reverting syntax changes.
|
||||
// See docs/SyntaxChanges.md for an explanation.
|
||||
LUAU_FASTFLAGVARIABLE(LuauSolverV2)
|
||||
LUAU_FASTFLAGVARIABLE(LuauUserDefinedTypeFunParseExport)
|
||||
LUAU_FASTFLAGVARIABLE(LuauAllowFragmentParsing)
|
||||
LUAU_FASTFLAGVARIABLE(LuauAllowComplexTypesInGenericParams)
|
||||
LUAU_FASTFLAGVARIABLE(LuauErrorRecoveryForTableTypes)
|
||||
|
@ -936,12 +935,6 @@ AstStat* Parser::parseTypeFunction(const Location& start, bool exported)
|
|||
Lexeme matchFn = lexer.current();
|
||||
nextLexeme();
|
||||
|
||||
if (!FFlag::LuauUserDefinedTypeFunParseExport)
|
||||
{
|
||||
if (exported)
|
||||
report(start, "Type function cannot be exported");
|
||||
}
|
||||
|
||||
// parse the name of the type function
|
||||
std::optional<Name> fnName = parseNameOpt("type function name");
|
||||
if (!fnName)
|
||||
|
@ -2239,7 +2232,8 @@ std::optional<AstExprBinary::Op> Parser::checkBinaryConfusables(const BinaryOpPr
|
|||
report(Location(start, next.location), "Unexpected '||'; did you mean 'or'?");
|
||||
return AstExprBinary::Or;
|
||||
}
|
||||
else if (curr.type == '!' && next.type == '=' && curr.location.end == next.location.begin && binaryPriority[AstExprBinary::CompareNe].left > limit)
|
||||
else if (curr.type == '!' && next.type == '=' && curr.location.end == next.location.begin &&
|
||||
binaryPriority[AstExprBinary::CompareNe].left > limit)
|
||||
{
|
||||
nextLexeme();
|
||||
report(Location(start, next.location), "Unexpected '!='; did you mean '~='?");
|
||||
|
@ -2587,7 +2581,8 @@ AstExpr* Parser::parseSimpleExpr()
|
|||
{
|
||||
return parseNumber();
|
||||
}
|
||||
else if (lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString || lexer.current().type == Lexeme::InterpStringSimple)
|
||||
else if (lexer.current().type == Lexeme::RawString || lexer.current().type == Lexeme::QuotedString ||
|
||||
lexer.current().type == Lexeme::InterpStringSimple)
|
||||
{
|
||||
return parseString();
|
||||
}
|
||||
|
|
|
@ -7,9 +7,9 @@
|
|||
#include "Luau/TypeAttach.h"
|
||||
#include "Luau/Transpiler.h"
|
||||
|
||||
#include "FileUtils.h"
|
||||
#include "Flags.h"
|
||||
#include "Require.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
#include "Luau/Flags.h"
|
||||
#include "Luau/Require.h"
|
||||
|
||||
#include <condition_variable>
|
||||
#include <functional>
|
|
@ -8,7 +8,7 @@
|
|||
#include "Luau/ParseOptions.h"
|
||||
#include "Luau/ToString.h"
|
||||
|
||||
#include "FileUtils.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
|
||||
static void displayHelp(const char* argv0)
|
||||
{
|
|
@ -7,8 +7,8 @@
|
|||
#include "Luau/BytecodeBuilder.h"
|
||||
#include "Luau/Parser.h"
|
||||
#include "Luau/BytecodeSummary.h"
|
||||
#include "FileUtils.h"
|
||||
#include "Flags.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
#include "Luau/Flags.h"
|
||||
|
||||
#include <memory>
|
||||
|
|
@ -8,8 +8,8 @@
|
|||
#include "Luau/Parser.h"
|
||||
#include "Luau/TimeTrace.h"
|
||||
|
||||
#include "FileUtils.h"
|
||||
#include "Flags.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
#include "Luau/Flags.h"
|
||||
|
||||
#include <memory>
|
||||
|
||||
|
@ -341,7 +341,8 @@ static bool compileFile(const char* name, CompileFormat format, Luau::CodeGen::A
|
|||
bcb.setDumpFlags(Luau::BytecodeBuilder::Dump_Source | Luau::BytecodeBuilder::Dump_Remarks);
|
||||
bcb.setDumpSource(*source);
|
||||
}
|
||||
else if (format == CompileFormat::Codegen || format == CompileFormat::CodegenAsm || format == CompileFormat::CodegenIr || format == CompileFormat::CodegenVerbose)
|
||||
else if (format == CompileFormat::Codegen || format == CompileFormat::CodegenAsm || format == CompileFormat::CodegenIr ||
|
||||
format == CompileFormat::CodegenVerbose)
|
||||
{
|
||||
bcb.setDumpFlags(
|
||||
Luau::BytecodeBuilder::Dump_Code | Luau::BytecodeBuilder::Dump_Source | Luau::BytecodeBuilder::Dump_Locals |
|
|
@ -1,5 +1,5 @@
|
|||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||
#include "Coverage.h"
|
||||
#include "Luau/Coverage.h"
|
||||
|
||||
#include "lua.h"
|
||||
|
|
@ -1,5 +1,5 @@
|
|||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||
#include "FileUtils.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
|
||||
#include "Luau/Common.h"
|
||||
|
||||
|
@ -147,7 +147,7 @@ std::string resolvePath(std::string_view path, std::string_view baseFilePath)
|
|||
if (baseFilePathComponents.empty())
|
||||
{
|
||||
if (isResolvedPathRelative)
|
||||
numPrependedParents++; // "../" will later be added to the beginning of the resolved path
|
||||
numPrependedParents++; // "../" will later be added to the beginning of the resolved path
|
||||
}
|
||||
else if (baseFilePathComponents.back() != "..")
|
||||
{
|
|
@ -5,7 +5,7 @@
|
|||
#include "Luau/Parser.h"
|
||||
#include "Luau/Transpiler.h"
|
||||
|
||||
#include "FileUtils.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
|
||||
#include <algorithm>
|
||||
#include <stdio.h>
|
|
@ -1,5 +1,5 @@
|
|||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||
#include "Repl.h"
|
||||
#include "Luau/Repl.h"
|
||||
|
||||
#include "Luau/Common.h"
|
||||
#include "lua.h"
|
||||
|
@ -10,11 +10,11 @@
|
|||
#include "Luau/Parser.h"
|
||||
#include "Luau/TimeTrace.h"
|
||||
|
||||
#include "Coverage.h"
|
||||
#include "FileUtils.h"
|
||||
#include "Flags.h"
|
||||
#include "Profiler.h"
|
||||
#include "Require.h"
|
||||
#include "Luau/Coverage.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
#include "Luau/Flags.h"
|
||||
#include "Luau/Profiler.h"
|
||||
#include "Luau/Require.h"
|
||||
|
||||
#include "isocline.h"
|
||||
|
|
@ -1,5 +1,5 @@
|
|||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||
#include "Repl.h"
|
||||
#include "Luau/Repl.h"
|
||||
|
||||
int main(int argc, char** argv)
|
||||
{
|
|
@ -1,7 +1,7 @@
|
|||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||
#include "Require.h"
|
||||
#include "Luau/Require.h"
|
||||
|
||||
#include "FileUtils.h"
|
||||
#include "Luau/FileUtils.h"
|
||||
#include "Luau/Common.h"
|
||||
#include "Luau/Config.h"
|
||||
|
||||
|
@ -301,4 +301,4 @@ bool RequireResolver::parseConfigInDirectory(const std::string& directory)
|
|||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
|
@ -68,11 +68,12 @@ include(Sources.cmake)
|
|||
target_include_directories(Luau.Common INTERFACE Common/include)
|
||||
|
||||
target_compile_features(Luau.CLI.lib PUBLIC cxx_std_17)
|
||||
target_link_libraries(Luau.CLI.lib PRIVATE Luau.Common)
|
||||
target_include_directories(Luau.CLI.lib PUBLIC CLI/include)
|
||||
target_link_libraries(Luau.CLI.lib PRIVATE Luau.Common Luau.Config)
|
||||
|
||||
target_compile_features(Luau.Ast PUBLIC cxx_std_17)
|
||||
target_include_directories(Luau.Ast PUBLIC Ast/include)
|
||||
target_link_libraries(Luau.Ast PUBLIC Luau.Common Luau.CLI.lib)
|
||||
target_link_libraries(Luau.Ast PUBLIC Luau.Common)
|
||||
|
||||
target_compile_features(Luau.Compiler PUBLIC cxx_std_17)
|
||||
target_include_directories(Luau.Compiler PUBLIC Compiler/include)
|
||||
|
|
|
@ -160,6 +160,7 @@ public:
|
|||
void vmaxsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
||||
void vminsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
||||
|
||||
void vcmpeqsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
||||
void vcmpltsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
||||
|
||||
void vblendvpd(RegisterX64 dst, RegisterX64 src1, OperandX64 mask, RegisterX64 src3);
|
||||
|
|
|
@ -185,6 +185,11 @@ enum class IrCmd : uint8_t
|
|||
// A: double
|
||||
SIGN_NUM,
|
||||
|
||||
// Select B if C == D, otherwise select A
|
||||
// A, B: double (endpoints)
|
||||
// C, D: double (condition arguments)
|
||||
SELECT_NUM,
|
||||
|
||||
// Add/Sub/Mul/Div/Idiv two vectors
|
||||
// A, B: TValue
|
||||
ADD_VEC,
|
||||
|
|
|
@ -7,6 +7,8 @@
|
|||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
struct Proto;
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
namespace CodeGen
|
||||
|
@ -23,6 +25,7 @@ struct IrToStringContext
|
|||
const std::vector<IrBlock>& blocks;
|
||||
const std::vector<IrConst>& constants;
|
||||
const CfgInfo& cfg;
|
||||
Proto* proto = nullptr;
|
||||
};
|
||||
|
||||
void toString(IrToStringContext& ctx, const IrInst& inst, uint32_t index);
|
||||
|
|
|
@ -174,6 +174,7 @@ inline bool hasResult(IrCmd cmd)
|
|||
case IrCmd::SQRT_NUM:
|
||||
case IrCmd::ABS_NUM:
|
||||
case IrCmd::SIGN_NUM:
|
||||
case IrCmd::SELECT_NUM:
|
||||
case IrCmd::ADD_VEC:
|
||||
case IrCmd::SUB_VEC:
|
||||
case IrCmd::MUL_VEC:
|
||||
|
|
|
@ -927,6 +927,11 @@ void AssemblyBuilderX64::vminsd(OperandX64 dst, OperandX64 src1, OperandX64 src2
|
|||
placeAvx("vminsd", dst, src1, src2, 0x5d, false, AVX_0F, AVX_F2);
|
||||
}
|
||||
|
||||
void AssemblyBuilderX64::vcmpeqsd(OperandX64 dst, OperandX64 src1, OperandX64 src2)
|
||||
{
|
||||
placeAvx("vcmpeqsd", dst, src1, src2, 0x00, 0xc2, false, AVX_0F, AVX_F2);
|
||||
}
|
||||
|
||||
void AssemblyBuilderX64::vcmpltsd(OperandX64 dst, OperandX64 src1, OperandX64 src2)
|
||||
{
|
||||
placeAvx("vcmpltsd", dst, src1, src2, 0x01, 0xc2, false, AVX_0F, AVX_F2);
|
||||
|
|
|
@ -235,7 +235,7 @@ static uint8_t getBytecodeConstantTag(Proto* proto, unsigned ki)
|
|||
return LBC_TYPE_ANY;
|
||||
}
|
||||
|
||||
static void applyBuiltinCall(int bfid, BytecodeTypes& types)
|
||||
static void applyBuiltinCall(LuauBuiltinFunction bfid, BytecodeTypes& types)
|
||||
{
|
||||
switch (bfid)
|
||||
{
|
||||
|
@ -549,6 +549,12 @@ static void applyBuiltinCall(int bfid, BytecodeTypes& types)
|
|||
types.b = LBC_TYPE_VECTOR;
|
||||
types.c = LBC_TYPE_VECTOR; // We can mark optional arguments
|
||||
break;
|
||||
case LBF_MATH_LERP:
|
||||
types.result = LBC_TYPE_NUMBER;
|
||||
types.a = LBC_TYPE_NUMBER;
|
||||
types.b = LBC_TYPE_NUMBER;
|
||||
types.c = LBC_TYPE_NUMBER;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -842,7 +848,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
regTags[ra] = LBC_TYPE_NUMBER;
|
||||
else if (bcType.a == LBC_TYPE_VECTOR && bcType.b == LBC_TYPE_VECTOR)
|
||||
regTags[ra] = LBC_TYPE_VECTOR;
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
|
||||
bcType.result = regTags[ra];
|
||||
|
@ -873,7 +880,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
if (bcType.b == LBC_TYPE_NUMBER || bcType.b == LBC_TYPE_VECTOR)
|
||||
regTags[ra] = LBC_TYPE_VECTOR;
|
||||
}
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
{
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
}
|
||||
|
@ -895,7 +903,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
|
||||
if (bcType.a == LBC_TYPE_NUMBER && bcType.b == LBC_TYPE_NUMBER)
|
||||
regTags[ra] = LBC_TYPE_NUMBER;
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
|
||||
bcType.result = regTags[ra];
|
||||
|
@ -917,7 +926,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
regTags[ra] = LBC_TYPE_NUMBER;
|
||||
else if (bcType.a == LBC_TYPE_VECTOR && bcType.b == LBC_TYPE_VECTOR)
|
||||
regTags[ra] = LBC_TYPE_VECTOR;
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
|
||||
bcType.result = regTags[ra];
|
||||
|
@ -948,7 +958,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
if (bcType.b == LBC_TYPE_NUMBER || bcType.b == LBC_TYPE_VECTOR)
|
||||
regTags[ra] = LBC_TYPE_VECTOR;
|
||||
}
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
{
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
}
|
||||
|
@ -970,7 +981,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
|
||||
if (bcType.a == LBC_TYPE_NUMBER && bcType.b == LBC_TYPE_NUMBER)
|
||||
regTags[ra] = LBC_TYPE_NUMBER;
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
|
||||
bcType.result = regTags[ra];
|
||||
|
@ -991,7 +1003,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
regTags[ra] = LBC_TYPE_NUMBER;
|
||||
else if (bcType.a == LBC_TYPE_VECTOR && bcType.b == LBC_TYPE_VECTOR)
|
||||
regTags[ra] = LBC_TYPE_VECTOR;
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
|
||||
bcType.result = regTags[ra];
|
||||
|
@ -1020,7 +1033,8 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
if (bcType.b == LBC_TYPE_NUMBER || bcType.b == LBC_TYPE_VECTOR)
|
||||
regTags[ra] = LBC_TYPE_VECTOR;
|
||||
}
|
||||
else if (hostHooks.userdataMetamethodBytecodeType && (isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
else if (hostHooks.userdataMetamethodBytecodeType &&
|
||||
(isCustomUserdataBytecodeType(bcType.a) || isCustomUserdataBytecodeType(bcType.b)))
|
||||
{
|
||||
regTags[ra] = hostHooks.userdataMetamethodBytecodeType(bcType.a, bcType.b, opcodeToHostMetamethod(op));
|
||||
}
|
||||
|
@ -1086,7 +1100,7 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
CODEGEN_ASSERT(LUAU_INSN_OP(call) == LOP_CALL);
|
||||
int ra = LUAU_INSN_A(call);
|
||||
|
||||
applyBuiltinCall(bfid, bcType);
|
||||
applyBuiltinCall(LuauBuiltinFunction(bfid), bcType);
|
||||
regTags[ra + 1] = bcType.a;
|
||||
regTags[ra + 2] = bcType.b;
|
||||
regTags[ra + 3] = bcType.c;
|
||||
|
@ -1105,7 +1119,7 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
CODEGEN_ASSERT(LUAU_INSN_OP(call) == LOP_CALL);
|
||||
int ra = LUAU_INSN_A(call);
|
||||
|
||||
applyBuiltinCall(bfid, bcType);
|
||||
applyBuiltinCall(LuauBuiltinFunction(bfid), bcType);
|
||||
|
||||
regTags[LUAU_INSN_B(*pc)] = bcType.a;
|
||||
regTags[ra] = bcType.result;
|
||||
|
@ -1122,7 +1136,7 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
CODEGEN_ASSERT(LUAU_INSN_OP(call) == LOP_CALL);
|
||||
int ra = LUAU_INSN_A(call);
|
||||
|
||||
applyBuiltinCall(bfid, bcType);
|
||||
applyBuiltinCall(LuauBuiltinFunction(bfid), bcType);
|
||||
|
||||
regTags[LUAU_INSN_B(*pc)] = bcType.a;
|
||||
regTags[int(pc[1])] = bcType.b;
|
||||
|
@ -1141,7 +1155,7 @@ void analyzeBytecodeTypes(IrFunction& function, const HostIrHooks& hostHooks)
|
|||
CODEGEN_ASSERT(LUAU_INSN_OP(call) == LOP_CALL);
|
||||
int ra = LUAU_INSN_A(call);
|
||||
|
||||
applyBuiltinCall(bfid, bcType);
|
||||
applyBuiltinCall(LuauBuiltinFunction(bfid), bcType);
|
||||
|
||||
regTags[LUAU_INSN_B(*pc)] = bcType.a;
|
||||
regTags[aux & 0xff] = bcType.b;
|
||||
|
|
|
@ -166,7 +166,7 @@ bool isSupported()
|
|||
if (sizeof(LuaNode) != 32)
|
||||
return false;
|
||||
|
||||
// Windows CRT uses stack unwinding in longjmp so we have to use unwind data; on other platforms, it's only necessary for C++ EH.
|
||||
// Windows CRT uses stack unwinding in longjmp so we have to use unwind data; on other platforms, it's only necessary for C++ EH.
|
||||
#if defined(_WIN32)
|
||||
if (!isUnwindSupported())
|
||||
return false;
|
||||
|
|
|
@ -101,7 +101,7 @@ inline bool lowerImpl(
|
|||
|
||||
bool outputEnabled = options.includeAssembly || options.includeIr;
|
||||
|
||||
IrToStringContext ctx{build.text, function.blocks, function.constants, function.cfg};
|
||||
IrToStringContext ctx{build.text, function.blocks, function.constants, function.cfg, function.proto};
|
||||
|
||||
// We use this to skip outlined fallback blocks from IR/asm text output
|
||||
size_t textSize = build.text.length();
|
||||
|
|
|
@ -18,6 +18,8 @@
|
|||
|
||||
#include <string.h>
|
||||
|
||||
LUAU_DYNAMIC_FASTFLAG(LuauPopIncompleteCi)
|
||||
|
||||
// All external function calls that can cause stack realloc or Lua calls have to be wrapped in VM_PROTECT
|
||||
// This makes sure that we save the pc (in case the Lua call needs to generate a backtrace) before the call,
|
||||
// and restores the stack pointer after in case stack gets reallocated
|
||||
|
@ -191,7 +193,14 @@ Closure* callProlog(lua_State* L, TValue* ra, StkId argtop, int nresults)
|
|||
// note: this reallocs stack, but we don't need to VM_PROTECT this
|
||||
// this is because we're going to modify base/savedpc manually anyhow
|
||||
// crucially, we can't use ra/argtop after this line
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
if (DFFlag::LuauPopIncompleteCi)
|
||||
{
|
||||
luaD_checkstackfornewci(L, ccl->stacksize);
|
||||
}
|
||||
else
|
||||
{
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
}
|
||||
|
||||
return ccl;
|
||||
}
|
||||
|
@ -261,7 +270,14 @@ Closure* callFallback(lua_State* L, StkId ra, StkId argtop, int nresults)
|
|||
// note: this reallocs stack, but we don't need to VM_PROTECT this
|
||||
// this is because we're going to modify base/savedpc manually anyhow
|
||||
// crucially, we can't use ra/argtop after this line
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
if (DFFlag::LuauPopIncompleteCi)
|
||||
{
|
||||
luaD_checkstackfornewci(L, ccl->stacksize);
|
||||
}
|
||||
else
|
||||
{
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
}
|
||||
|
||||
LUAU_ASSERT(ci->top <= L->stack_last);
|
||||
|
||||
|
|
|
@ -684,7 +684,7 @@ void computeCfgDominanceTreeChildren(IrFunction& function)
|
|||
info.domChildrenOffsets[domParent]++;
|
||||
}
|
||||
|
||||
// Convert counds to offsets using prefix sum
|
||||
// Convert counts to offsets using prefix sum
|
||||
uint32_t total = 0;
|
||||
|
||||
for (size_t blockIdx = 0; blockIdx < function.blocks.size(); blockIdx++)
|
||||
|
|
|
@ -4,6 +4,8 @@
|
|||
#include "Luau/IrUtils.h"
|
||||
|
||||
#include "lua.h"
|
||||
#include "lobject.h"
|
||||
#include "lstate.h"
|
||||
|
||||
#include <stdarg.h>
|
||||
|
||||
|
@ -19,6 +21,7 @@ static const char* textForCondition[] =
|
|||
static_assert(sizeof(textForCondition) / sizeof(textForCondition[0]) == size_t(IrCondition::Count), "all conditions have to be covered");
|
||||
|
||||
const int kDetailsAlignColumn = 60;
|
||||
const unsigned kMaxStringConstantPrintLength = 16;
|
||||
|
||||
LUAU_PRINTF_ATTR(2, 3)
|
||||
static void append(std::string& result, const char* fmt, ...)
|
||||
|
@ -39,6 +42,17 @@ static void padToDetailColumn(std::string& result, size_t lineStart)
|
|||
result.append(pad, ' ');
|
||||
}
|
||||
|
||||
static bool isPrintableStringConstant(const char* str, size_t len)
|
||||
{
|
||||
for (size_t i = 0; i < len; ++i)
|
||||
{
|
||||
if (unsigned(str[i]) < ' ')
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
static const char* getTagName(uint8_t tag)
|
||||
{
|
||||
switch (tag)
|
||||
|
@ -155,6 +169,8 @@ const char* getCmdName(IrCmd cmd)
|
|||
return "ABS_NUM";
|
||||
case IrCmd::SIGN_NUM:
|
||||
return "SIGN_NUM";
|
||||
case IrCmd::SELECT_NUM:
|
||||
return "SELECT_NUM";
|
||||
case IrCmd::ADD_VEC:
|
||||
return "ADD_VEC";
|
||||
case IrCmd::SUB_VEC:
|
||||
|
@ -431,6 +447,53 @@ void toString(IrToStringContext& ctx, const IrBlock& block, uint32_t index)
|
|||
append(ctx.result, "%s_%u", getBlockKindName(block.kind), index);
|
||||
}
|
||||
|
||||
static void appendVmConstant(std::string& result, Proto* proto, int index)
|
||||
{
|
||||
TValue constant = proto->k[index];
|
||||
|
||||
if (constant.tt == LUA_TNIL)
|
||||
{
|
||||
append(result, "nil");
|
||||
}
|
||||
else if (constant.tt == LUA_TBOOLEAN)
|
||||
{
|
||||
append(result, constant.value.b != 0 ? "true" : "false");
|
||||
}
|
||||
else if (constant.tt == LUA_TNUMBER)
|
||||
{
|
||||
if (constant.value.n != constant.value.n)
|
||||
append(result, "nan");
|
||||
else
|
||||
append(result, "%.17g", constant.value.n);
|
||||
}
|
||||
else if (constant.tt == LUA_TSTRING)
|
||||
{
|
||||
TString* str = gco2ts(constant.value.gc);
|
||||
const char* data = getstr(str);
|
||||
|
||||
if (isPrintableStringConstant(data, str->len))
|
||||
{
|
||||
if (str->len < kMaxStringConstantPrintLength)
|
||||
append(result, "'%.*s'", int(str->len), data);
|
||||
else
|
||||
append(result, "'%.*s'...", int(kMaxStringConstantPrintLength), data);
|
||||
}
|
||||
}
|
||||
else if (constant.tt == LUA_TVECTOR)
|
||||
{
|
||||
const float* v = constant.value.v;
|
||||
|
||||
#if LUA_VECTOR_SIZE == 4
|
||||
if (v[3] != 0)
|
||||
append(result, "%.9g, %.9g, %.9g, %.9g", v[0], v[1], v[2], v[3]);
|
||||
else
|
||||
append(result, "%.9g, %.9g, %.9g", v[0], v[1], v[2]);
|
||||
#else
|
||||
append(result, "%.9g, %.9g, %.9g", v[0], v[1], v[2]);
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
void toString(IrToStringContext& ctx, IrOp op)
|
||||
{
|
||||
switch (op.kind)
|
||||
|
@ -458,6 +521,14 @@ void toString(IrToStringContext& ctx, IrOp op)
|
|||
break;
|
||||
case IrOpKind::VmConst:
|
||||
append(ctx.result, "K%d", vmConstOp(op));
|
||||
|
||||
if (ctx.proto)
|
||||
{
|
||||
append(ctx.result, " (");
|
||||
appendVmConstant(ctx.result, ctx.proto, vmConstOp(op));
|
||||
append(ctx.result, ")");
|
||||
}
|
||||
|
||||
break;
|
||||
case IrOpKind::VmUpvalue:
|
||||
append(ctx.result, "U%d", vmUpvalueOp(op));
|
||||
|
@ -770,7 +841,7 @@ void toStringDetailed(
|
|||
std::string toString(const IrFunction& function, IncludeUseInfo includeUseInfo)
|
||||
{
|
||||
std::string result;
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg, function.proto};
|
||||
|
||||
for (size_t i = 0; i < function.blocks.size(); i++)
|
||||
{
|
||||
|
@ -877,7 +948,7 @@ static void appendBlocks(IrToStringContext& ctx, const IrFunction& function, boo
|
|||
std::string toDot(const IrFunction& function, bool includeInst)
|
||||
{
|
||||
std::string result;
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg, function.proto};
|
||||
|
||||
append(ctx.result, "digraph CFG {\n");
|
||||
append(ctx.result, "node[shape=record]\n");
|
||||
|
@ -924,7 +995,7 @@ std::string toDot(const IrFunction& function, bool includeInst)
|
|||
std::string toDotCfg(const IrFunction& function)
|
||||
{
|
||||
std::string result;
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg, function.proto};
|
||||
|
||||
append(ctx.result, "digraph CFG {\n");
|
||||
append(ctx.result, "node[shape=record]\n");
|
||||
|
@ -947,7 +1018,7 @@ std::string toDotCfg(const IrFunction& function)
|
|||
std::string toDotDjGraph(const IrFunction& function)
|
||||
{
|
||||
std::string result;
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg};
|
||||
IrToStringContext ctx{result, function.blocks, function.constants, function.cfg, function.proto};
|
||||
|
||||
append(ctx.result, "digraph CFG {\n");
|
||||
|
||||
|
|
|
@ -13,6 +13,7 @@
|
|||
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeDot)
|
||||
LUAU_FASTFLAG(LuauCodeGenVectorDeadStoreElim)
|
||||
LUAU_FASTFLAG(LuauCodeGenLerp)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -703,6 +704,20 @@ void IrLoweringA64::lowerInst(IrInst& inst, uint32_t index, const IrBlock& next)
|
|||
build.fcsel(inst.regA64, temp1, inst.regA64, getConditionFP(IrCondition::Less));
|
||||
break;
|
||||
}
|
||||
case IrCmd::SELECT_NUM:
|
||||
{
|
||||
LUAU_ASSERT(FFlag::LuauCodeGenLerp);
|
||||
inst.regA64 = regs.allocReuse(KindA64::d, index, {inst.a, inst.b, inst.c, inst.d});
|
||||
|
||||
RegisterA64 temp1 = tempDouble(inst.a);
|
||||
RegisterA64 temp2 = tempDouble(inst.b);
|
||||
RegisterA64 temp3 = tempDouble(inst.c);
|
||||
RegisterA64 temp4 = tempDouble(inst.d);
|
||||
|
||||
build.fcmp(temp3, temp4);
|
||||
build.fcsel(inst.regA64, temp2, temp1, getConditionFP(IrCondition::Equal));
|
||||
break;
|
||||
}
|
||||
case IrCmd::ADD_VEC:
|
||||
{
|
||||
inst.regA64 = regs.allocReuse(KindA64::q, index, {inst.a, inst.b});
|
||||
|
|
|
@ -17,6 +17,7 @@
|
|||
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeDot)
|
||||
LUAU_FASTFLAG(LuauCodeGenVectorDeadStoreElim)
|
||||
LUAU_FASTFLAG(LuauCodeGenLerp)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -622,6 +623,30 @@ void IrLoweringX64::lowerInst(IrInst& inst, uint32_t index, const IrBlock& next)
|
|||
build.vblendvpd(inst.regX64, tmp1.reg, build.f64x2(1, 1), inst.regX64);
|
||||
break;
|
||||
}
|
||||
case IrCmd::SELECT_NUM:
|
||||
{
|
||||
LUAU_ASSERT(FFlag::LuauCodeGenLerp);
|
||||
inst.regX64 = regs.allocRegOrReuse(SizeX64::xmmword, index, {inst.a, inst.c, inst.d}); // can't reuse b if a is a memory operand
|
||||
|
||||
ScopedRegX64 tmp{regs, SizeX64::xmmword};
|
||||
|
||||
if (inst.c.kind == IrOpKind::Inst)
|
||||
build.vcmpeqsd(tmp.reg, regOp(inst.c), memRegDoubleOp(inst.d));
|
||||
else
|
||||
{
|
||||
build.vmovsd(tmp.reg, memRegDoubleOp(inst.c));
|
||||
build.vcmpeqsd(tmp.reg, tmp.reg, memRegDoubleOp(inst.d));
|
||||
}
|
||||
|
||||
if (inst.a.kind == IrOpKind::Inst)
|
||||
build.vblendvpd(inst.regX64, regOp(inst.a), memRegDoubleOp(inst.b), tmp.reg);
|
||||
else
|
||||
{
|
||||
build.vmovsd(inst.regX64, memRegDoubleOp(inst.a));
|
||||
build.vblendvpd(inst.regX64, inst.regX64, memRegDoubleOp(inst.b), tmp.reg);
|
||||
}
|
||||
break;
|
||||
}
|
||||
case IrCmd::ADD_VEC:
|
||||
{
|
||||
inst.regX64 = regs.allocRegOrReuse(SizeX64::xmmword, index, {inst.a, inst.b});
|
||||
|
|
|
@ -15,6 +15,7 @@ static const int kBit32BinaryOpUnrolledParams = 5;
|
|||
|
||||
LUAU_FASTFLAGVARIABLE(LuauVectorLibNativeCodegen);
|
||||
LUAU_FASTFLAGVARIABLE(LuauVectorLibNativeDot);
|
||||
LUAU_FASTFLAGVARIABLE(LuauCodeGenLerp);
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -284,6 +285,42 @@ static BuiltinImplResult translateBuiltinMathClamp(
|
|||
return {BuiltinImplType::UsesFallback, 1};
|
||||
}
|
||||
|
||||
static BuiltinImplResult translateBuiltinMathLerp(
|
||||
IrBuilder& build,
|
||||
int nparams,
|
||||
int ra,
|
||||
int arg,
|
||||
IrOp args,
|
||||
IrOp arg3,
|
||||
int nresults,
|
||||
IrOp fallback,
|
||||
int pcpos
|
||||
)
|
||||
{
|
||||
LUAU_ASSERT(FFlag::LuauCodeGenLerp);
|
||||
|
||||
if (nparams < 3 || nresults > 1)
|
||||
return {BuiltinImplType::None, -1};
|
||||
|
||||
builtinCheckDouble(build, build.vmReg(arg), pcpos);
|
||||
builtinCheckDouble(build, args, pcpos);
|
||||
builtinCheckDouble(build, arg3, pcpos);
|
||||
|
||||
IrOp a = builtinLoadDouble(build, build.vmReg(arg));
|
||||
IrOp b = builtinLoadDouble(build, args);
|
||||
IrOp t = builtinLoadDouble(build, arg3);
|
||||
|
||||
IrOp l = build.inst(IrCmd::ADD_NUM, a, build.inst(IrCmd::MUL_NUM, build.inst(IrCmd::SUB_NUM, b, a), t));
|
||||
IrOp r = build.inst(IrCmd::SELECT_NUM, l, b, t, build.constDouble(1.0)); // select on t==1.0
|
||||
|
||||
build.inst(IrCmd::STORE_DOUBLE, build.vmReg(ra), r);
|
||||
|
||||
if (ra != arg)
|
||||
build.inst(IrCmd::STORE_TAG, build.vmReg(ra), build.constTag(LUA_TNUMBER));
|
||||
|
||||
return {BuiltinImplType::Full, 1};
|
||||
}
|
||||
|
||||
static BuiltinImplResult translateBuiltinMathUnary(IrBuilder& build, IrCmd cmd, int nparams, int ra, int arg, int nresults, int pcpos)
|
||||
{
|
||||
if (nparams < 1 || nresults > 1)
|
||||
|
@ -1387,6 +1424,8 @@ BuiltinImplResult translateBuiltin(
|
|||
case LBF_VECTOR_MAX:
|
||||
return FFlag::LuauVectorLibNativeCodegen ? translateBuiltinVectorMap2(build, IrCmd::MAX_NUM, nparams, ra, arg, args, arg3, nresults, pcpos)
|
||||
: noneResult;
|
||||
case LBF_MATH_LERP:
|
||||
return FFlag::LuauCodeGenLerp ? translateBuiltinMathLerp(build, nparams, ra, arg, args, arg3, nresults, fallback, pcpos) : noneResult;
|
||||
default:
|
||||
return {BuiltinImplType::None, -1};
|
||||
}
|
||||
|
|
|
@ -13,6 +13,7 @@
|
|||
#include <math.h>
|
||||
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeDot);
|
||||
LUAU_FASTFLAG(LuauCodeGenLerp);
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -70,6 +71,7 @@ IrValueKind getCmdValueKind(IrCmd cmd)
|
|||
case IrCmd::SQRT_NUM:
|
||||
case IrCmd::ABS_NUM:
|
||||
case IrCmd::SIGN_NUM:
|
||||
case IrCmd::SELECT_NUM:
|
||||
return IrValueKind::Double;
|
||||
case IrCmd::ADD_VEC:
|
||||
case IrCmd::SUB_VEC:
|
||||
|
@ -656,6 +658,16 @@ void foldConstants(IrBuilder& build, IrFunction& function, IrBlock& block, uint3
|
|||
substitute(function, inst, build.constDouble(v > 0.0 ? 1.0 : v < 0.0 ? -1.0 : 0.0));
|
||||
}
|
||||
break;
|
||||
case IrCmd::SELECT_NUM:
|
||||
LUAU_ASSERT(FFlag::LuauCodeGenLerp);
|
||||
if (inst.c.kind == IrOpKind::Constant && inst.d.kind == IrOpKind::Constant)
|
||||
{
|
||||
double c = function.doubleOp(inst.c);
|
||||
double d = function.doubleOp(inst.d);
|
||||
|
||||
substitute(function, inst, c == d ? inst.b : inst.a);
|
||||
}
|
||||
break;
|
||||
case IrCmd::NOT_ANY:
|
||||
if (inst.a.kind == IrOpKind::Constant)
|
||||
{
|
||||
|
|
|
@ -11,6 +11,7 @@
|
|||
#include <limits.h>
|
||||
#include <math.h>
|
||||
|
||||
#include <algorithm>
|
||||
#include <array>
|
||||
#include <utility>
|
||||
#include <vector>
|
||||
|
@ -18,9 +19,11 @@
|
|||
LUAU_FASTINTVARIABLE(LuauCodeGenMinLinearBlockPath, 3)
|
||||
LUAU_FASTINTVARIABLE(LuauCodeGenReuseSlotLimit, 64)
|
||||
LUAU_FASTINTVARIABLE(LuauCodeGenReuseUdataTagLimit, 64)
|
||||
LUAU_FASTINTVARIABLE(LuauCodeGenLiveSlotReuseLimit, 8)
|
||||
LUAU_FASTFLAGVARIABLE(DebugLuauAbortingChecks)
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeDot);
|
||||
LUAU_FASTFLAGVARIABLE(LuauCodeGenArithOpt);
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeDot)
|
||||
LUAU_FASTFLAGVARIABLE(LuauCodeGenArithOpt)
|
||||
LUAU_FASTFLAGVARIABLE(LuauCodeGenLimitLiveSlotReuse)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -50,6 +53,14 @@ struct RegisterLink
|
|||
uint32_t version = 0;
|
||||
};
|
||||
|
||||
// Reference to an instruction together with the position of that instruction in the current block chain and the last position of reuse
|
||||
struct NumberedInstruction
|
||||
{
|
||||
uint32_t instIdx = 0;
|
||||
uint32_t startPos = 0;
|
||||
uint32_t finishPos = 0;
|
||||
};
|
||||
|
||||
// Data we know about the current VM state
|
||||
struct ConstPropState
|
||||
{
|
||||
|
@ -190,7 +201,11 @@ struct ConstPropState
|
|||
// Same goes for table array elements as well
|
||||
void invalidateHeapTableData()
|
||||
{
|
||||
getSlotNodeCache.clear();
|
||||
if (FFlag::LuauCodeGenLimitLiveSlotReuse)
|
||||
getSlotNodeCache.clear();
|
||||
else
|
||||
getSlotNodeCache_DEPRECATED.clear();
|
||||
|
||||
checkSlotMatchCache.clear();
|
||||
|
||||
getArrAddrCache.clear();
|
||||
|
@ -409,6 +424,64 @@ struct ConstPropState
|
|||
valueMap[versionedVmRegLoad(loadCmd, storeInst.a)] = storeInst.b.index;
|
||||
}
|
||||
|
||||
// Used to compute the pressure of the cached value 'set' on the spill registers
|
||||
// We want to find out the maximum live range intersection count between the cached value at 'slot' and current instruction
|
||||
// Note that this pressure is approximate, as some values that might have been live at one point could have been marked dead later
|
||||
int getMaxInternalOverlap(std::vector<NumberedInstruction>& set, size_t slot)
|
||||
{
|
||||
CODEGEN_ASSERT(FFlag::LuauCodeGenLimitLiveSlotReuse);
|
||||
|
||||
// Start with one live range for the slot we want to reuse
|
||||
int curr = 1;
|
||||
|
||||
// For any slots where lifetime began before the slot of interest, mark as live if lifetime end is still active
|
||||
// This saves us from processing slots [0; slot] in the range sweep later, which requires sorting the lifetime end points
|
||||
for (size_t i = 0; i < slot; i++)
|
||||
{
|
||||
if (set[i].finishPos >= set[slot].startPos)
|
||||
curr++;
|
||||
}
|
||||
|
||||
int max = curr;
|
||||
|
||||
// Collect lifetime end points and sort them
|
||||
rangeEndTemp.clear();
|
||||
|
||||
for (size_t i = slot + 1; i < set.size(); i++)
|
||||
rangeEndTemp.push_back(set[i].finishPos);
|
||||
|
||||
std::sort(rangeEndTemp.begin(), rangeEndTemp.end());
|
||||
|
||||
// Go over the lifetime begin/end ranges that we store as separate array and walk based on the smallest of values
|
||||
for (size_t i1 = slot + 1, i2 = 0; i1 < set.size() && i2 < rangeEndTemp.size();)
|
||||
{
|
||||
if (rangeEndTemp[i2] == set[i1].startPos)
|
||||
{
|
||||
i1++;
|
||||
i2++;
|
||||
}
|
||||
else if (rangeEndTemp[i2] < set[i1].startPos)
|
||||
{
|
||||
CODEGEN_ASSERT(curr > 0);
|
||||
|
||||
curr--;
|
||||
i2++;
|
||||
}
|
||||
else
|
||||
{
|
||||
curr++;
|
||||
i1++;
|
||||
|
||||
if (curr > max)
|
||||
max = curr;
|
||||
}
|
||||
}
|
||||
|
||||
// We might have unprocessed lifetime end entries, but we will never have unprocessed lifetime start entries
|
||||
// Not that lifetime end entries can only decrease the current value and do not affect the end result (maximum)
|
||||
return max;
|
||||
}
|
||||
|
||||
void clear()
|
||||
{
|
||||
for (int i = 0; i <= maxReg; ++i)
|
||||
|
@ -416,6 +489,9 @@ struct ConstPropState
|
|||
|
||||
maxReg = 0;
|
||||
|
||||
if (FFlag::LuauCodeGenLimitLiveSlotReuse)
|
||||
instPos = 0u;
|
||||
|
||||
inSafeEnv = false;
|
||||
checkedGc = false;
|
||||
|
||||
|
@ -436,6 +512,9 @@ struct ConstPropState
|
|||
// For range/full invalidations, we only want to visit a limited number of data that we have recorded
|
||||
int maxReg = 0;
|
||||
|
||||
// Number of the instruction being processed
|
||||
uint32_t instPos = 0;
|
||||
|
||||
bool inSafeEnv = false;
|
||||
bool checkedGc = false;
|
||||
|
||||
|
@ -447,7 +526,8 @@ struct ConstPropState
|
|||
std::vector<uint32_t> tryNumToIndexCache; // Fallback block argument might be different
|
||||
|
||||
// Heap changes might affect table state
|
||||
std::vector<uint32_t> getSlotNodeCache; // Additionally, pcpos argument might be different
|
||||
std::vector<NumberedInstruction> getSlotNodeCache; // Additionally, pcpos argument might be different
|
||||
std::vector<uint32_t> getSlotNodeCache_DEPRECATED; // Additionally, pcpos argument might be different
|
||||
std::vector<uint32_t> checkSlotMatchCache; // Additionally, fallback block argument might be different
|
||||
|
||||
std::vector<uint32_t> getArrAddrCache;
|
||||
|
@ -457,6 +537,8 @@ struct ConstPropState
|
|||
|
||||
// Userdata tag cache can point to both NEW_USERDATA and CHECK_USERDATA_TAG instructions
|
||||
std::vector<uint32_t> useradataTagCache; // Additionally, fallback block argument might be different
|
||||
|
||||
std::vector<uint32_t> rangeEndTemp;
|
||||
};
|
||||
|
||||
static void handleBuiltinEffects(ConstPropState& state, LuauBuiltinFunction bfid, uint32_t firstReturnReg, int nresults)
|
||||
|
@ -550,6 +632,7 @@ static void handleBuiltinEffects(ConstPropState& state, LuauBuiltinFunction bfid
|
|||
case LBF_VECTOR_CLAMP:
|
||||
case LBF_VECTOR_MIN:
|
||||
case LBF_VECTOR_MAX:
|
||||
case LBF_MATH_LERP:
|
||||
break;
|
||||
case LBF_TABLE_INSERT:
|
||||
state.invalidateHeap();
|
||||
|
@ -569,6 +652,9 @@ static void handleBuiltinEffects(ConstPropState& state, LuauBuiltinFunction bfid
|
|||
|
||||
static void constPropInInst(ConstPropState& state, IrBuilder& build, IrFunction& function, IrBlock& block, IrInst& inst, uint32_t index)
|
||||
{
|
||||
if (FFlag::LuauCodeGenLimitLiveSlotReuse)
|
||||
state.instPos++;
|
||||
|
||||
switch (inst.cmd)
|
||||
{
|
||||
case IrCmd::LOAD_TAG:
|
||||
|
@ -1175,19 +1261,49 @@ static void constPropInInst(ConstPropState& state, IrBuilder& build, IrFunction&
|
|||
state.getArrAddrCache.push_back(index);
|
||||
break;
|
||||
case IrCmd::GET_SLOT_NODE_ADDR:
|
||||
for (uint32_t prevIdx : state.getSlotNodeCache)
|
||||
if (FFlag::LuauCodeGenLimitLiveSlotReuse)
|
||||
{
|
||||
const IrInst& prev = function.instructions[prevIdx];
|
||||
|
||||
if (prev.a == inst.a && prev.c == inst.c)
|
||||
for (size_t i = 0; i < state.getSlotNodeCache.size(); i++)
|
||||
{
|
||||
substitute(function, inst, IrOp{IrOpKind::Inst, prevIdx});
|
||||
return; // Break out from both the loop and the switch
|
||||
}
|
||||
}
|
||||
auto&& [prevIdx, num, lastNum] = state.getSlotNodeCache[i];
|
||||
|
||||
if (int(state.getSlotNodeCache.size()) < FInt::LuauCodeGenReuseSlotLimit)
|
||||
state.getSlotNodeCache.push_back(index);
|
||||
const IrInst& prev = function.instructions[prevIdx];
|
||||
|
||||
if (prev.a == inst.a && prev.c == inst.c)
|
||||
{
|
||||
// Check if this reuse will increase the overall register pressure over the limit
|
||||
int limit = FInt::LuauCodeGenLiveSlotReuseLimit;
|
||||
|
||||
if (int(state.getSlotNodeCache.size()) > limit && state.getMaxInternalOverlap(state.getSlotNodeCache, i) > limit)
|
||||
return;
|
||||
|
||||
// Update live range of the value from the optimization standpoint
|
||||
lastNum = state.instPos;
|
||||
|
||||
substitute(function, inst, IrOp{IrOpKind::Inst, prevIdx});
|
||||
return; // Break out from both the loop and the switch
|
||||
}
|
||||
}
|
||||
|
||||
if (int(state.getSlotNodeCache.size()) < FInt::LuauCodeGenReuseSlotLimit)
|
||||
state.getSlotNodeCache.push_back({index, state.instPos, state.instPos});
|
||||
}
|
||||
else
|
||||
{
|
||||
for (uint32_t prevIdx : state.getSlotNodeCache_DEPRECATED)
|
||||
{
|
||||
const IrInst& prev = function.instructions[prevIdx];
|
||||
|
||||
if (prev.a == inst.a && prev.c == inst.c)
|
||||
{
|
||||
substitute(function, inst, IrOp{IrOpKind::Inst, prevIdx});
|
||||
return; // Break out from both the loop and the switch
|
||||
}
|
||||
}
|
||||
|
||||
if (int(state.getSlotNodeCache_DEPRECATED.size()) < FInt::LuauCodeGenReuseSlotLimit)
|
||||
state.getSlotNodeCache_DEPRECATED.push_back(index);
|
||||
}
|
||||
break;
|
||||
case IrCmd::GET_HASH_NODE_ADDR:
|
||||
case IrCmd::GET_CLOSURE_UPVAL_ADDR:
|
||||
|
@ -1266,6 +1382,7 @@ static void constPropInInst(ConstPropState& state, IrBuilder& build, IrFunction&
|
|||
case IrCmd::SQRT_NUM:
|
||||
case IrCmd::ABS_NUM:
|
||||
case IrCmd::SIGN_NUM:
|
||||
case IrCmd::SELECT_NUM:
|
||||
case IrCmd::NOT_ANY:
|
||||
state.substituteOrRecord(inst, index);
|
||||
break;
|
||||
|
|
|
@ -613,6 +613,9 @@ enum LuauBuiltinFunction
|
|||
LBF_VECTOR_CLAMP,
|
||||
LBF_VECTOR_MIN,
|
||||
LBF_VECTOR_MAX,
|
||||
|
||||
// math.lerp
|
||||
LBF_MATH_LERP,
|
||||
};
|
||||
|
||||
// Capture type, used in LOP_CAPTURE
|
||||
|
|
|
@ -13,7 +13,7 @@ inline bool isFlagExperimental(const char* flag)
|
|||
static const char* const kList[] = {
|
||||
"LuauInstantiateInSubtyping", // requires some fixes to lua-apps code
|
||||
"LuauFixIndexerSubtypingOrdering", // requires some small fixes to lua-apps code since this fixes a false negative
|
||||
"StudioReportLuauAny2", // takes telemetry data for usage of any types
|
||||
"StudioReportLuauAny2", // takes telemetry data for usage of any types
|
||||
"LuauSolverV2",
|
||||
// makes sure we always have at least one entry
|
||||
nullptr,
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
#include <math.h>
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauVector2Constants)
|
||||
LUAU_FASTFLAG(LuauCompileMathLerp)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -483,6 +484,19 @@ Constant foldBuiltin(int bfid, const Constant* args, size_t count)
|
|||
return cvector(args[0].valueNumber, args[1].valueNumber, args[2].valueNumber, args[3].valueNumber);
|
||||
}
|
||||
break;
|
||||
|
||||
case LBF_MATH_LERP:
|
||||
if (FFlag::LuauCompileMathLerp && count == 3 && args[0].type == Constant::Type_Number && args[1].type == Constant::Type_Number &&
|
||||
args[2].type == Constant::Type_Number)
|
||||
{
|
||||
double a = args[0].valueNumber;
|
||||
double b = args[1].valueNumber;
|
||||
double t = args[2].valueNumber;
|
||||
|
||||
double v = (t == 1.0) ? b : a + (b - a) * t;
|
||||
return cnum(v);
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
return cvar();
|
||||
|
|
|
@ -7,8 +7,8 @@
|
|||
|
||||
#include <array>
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauVectorBuiltins)
|
||||
LUAU_FASTFLAGVARIABLE(LuauCompileDisabledBuiltins)
|
||||
LUAU_FASTFLAGVARIABLE(LuauCompileMathLerp)
|
||||
|
||||
namespace Luau
|
||||
{
|
||||
|
@ -140,6 +140,8 @@ static int getBuiltinFunctionId(const Builtin& builtin, const CompileOptions& op
|
|||
return LBF_MATH_SIGN;
|
||||
if (builtin.method == "round")
|
||||
return LBF_MATH_ROUND;
|
||||
if (FFlag::LuauCompileMathLerp && builtin.method == "lerp")
|
||||
return LBF_MATH_LERP;
|
||||
}
|
||||
|
||||
if (builtin.object == "bit32")
|
||||
|
@ -226,7 +228,7 @@ static int getBuiltinFunctionId(const Builtin& builtin, const CompileOptions& op
|
|||
return LBF_BUFFER_WRITEF64;
|
||||
}
|
||||
|
||||
if (FFlag::LuauVectorBuiltins && builtin.object == "vector")
|
||||
if (builtin.object == "vector")
|
||||
{
|
||||
if (builtin.method == "create")
|
||||
return LBF_VECTOR;
|
||||
|
@ -556,6 +558,10 @@ BuiltinInfo getBuiltinInfo(int bfid)
|
|||
case LBF_VECTOR_MIN:
|
||||
case LBF_VECTOR_MAX:
|
||||
return {-1, 1}; // variadic
|
||||
|
||||
case LBF_MATH_LERP:
|
||||
LUAU_ASSERT(FFlag::LuauCompileMathLerp);
|
||||
return {3, 1, BuiltinInfo::Flag_NoneSafe};
|
||||
}
|
||||
|
||||
LUAU_UNREACHABLE();
|
||||
|
|
|
@ -1751,7 +1751,8 @@ void BytecodeBuilder::validateVariadic() const
|
|||
// variadic sequence since they are never executed if FASTCALL does anything, so it's okay to skip their validation until CALL
|
||||
// (we can't simply start a variadic sequence here because that would trigger assertions during linked CALL validation)
|
||||
}
|
||||
else if (op == LOP_CLOSEUPVALS || op == LOP_NAMECALL || op == LOP_GETIMPORT || op == LOP_MOVE || op == LOP_GETUPVAL || op == LOP_GETGLOBAL || op == LOP_GETTABLEKS || op == LOP_COVERAGE)
|
||||
else if (op == LOP_CLOSEUPVALS || op == LOP_NAMECALL || op == LOP_GETIMPORT || op == LOP_MOVE || op == LOP_GETUPVAL || op == LOP_GETGLOBAL ||
|
||||
op == LOP_GETTABLEKS || op == LOP_COVERAGE)
|
||||
{
|
||||
// instructions inside a variadic sequence must be neutral (can't change L->top)
|
||||
// while there are many neutral instructions like this, here we check that the instruction is one of the few
|
||||
|
|
|
@ -1624,7 +1624,8 @@ struct Compiler
|
|||
return;
|
||||
}
|
||||
}
|
||||
else if (FFlag::LuauCompileOptimizeRevArith && options.optimizationLevel >= 2 && (expr->op == AstExprBinary::Add || expr->op == AstExprBinary::Mul))
|
||||
else if (FFlag::LuauCompileOptimizeRevArith && options.optimizationLevel >= 2 &&
|
||||
(expr->op == AstExprBinary::Add || expr->op == AstExprBinary::Mul))
|
||||
{
|
||||
// Optimization: replace k*r with r*k when r is known to be a number (otherwise metamethods may be called)
|
||||
if (LuauBytecodeType* ty = exprTypes.find(expr); ty && *ty == LBC_TYPE_NUMBER)
|
||||
|
@ -4408,7 +4409,7 @@ void setCompileConstantString(CompileConstant* constant, const char* s, size_t l
|
|||
CompileError::raise({}, "Exceeded custom string constant length limit");
|
||||
|
||||
target->type = Compile::Constant::Type_String;
|
||||
target->stringLength = l;
|
||||
target->stringLength = unsigned(l);
|
||||
target->valueString = s;
|
||||
}
|
||||
|
||||
|
|
|
@ -130,7 +130,8 @@ struct CostVisitor : AstVisitor
|
|||
{
|
||||
return model(expr->expr);
|
||||
}
|
||||
else if (node->is<AstExprConstantNil>() || node->is<AstExprConstantBool>() || node->is<AstExprConstantNumber>() || node->is<AstExprConstantString>())
|
||||
else if (node->is<AstExprConstantNil>() || node->is<AstExprConstantBool>() || node->is<AstExprConstantNumber>() ||
|
||||
node->is<AstExprConstantString>())
|
||||
{
|
||||
return Cost(0, Cost::kLiteral);
|
||||
}
|
||||
|
|
|
@ -3,7 +3,6 @@
|
|||
|
||||
#include "Luau/BytecodeBuilder.h"
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauCompileVectorTypeInfo)
|
||||
LUAU_FASTFLAG(LuauCompileLibraryConstants)
|
||||
|
||||
namespace Luau
|
||||
|
@ -32,7 +31,7 @@ static LuauBytecodeType getPrimitiveType(AstName name)
|
|||
return LBC_TYPE_THREAD;
|
||||
else if (name == "buffer")
|
||||
return LBC_TYPE_BUFFER;
|
||||
else if (FFlag::LuauCompileVectorTypeInfo && name == "vector")
|
||||
else if (name == "vector")
|
||||
return LBC_TYPE_VECTOR;
|
||||
else if (name == "any" || name == "unknown")
|
||||
return LBC_TYPE_ANY;
|
||||
|
@ -747,6 +746,7 @@ struct TypeMapVisitor : AstVisitor
|
|||
case LBF_BUFFER_READF64:
|
||||
case LBF_VECTOR_MAGNITUDE:
|
||||
case LBF_VECTOR_DOT:
|
||||
case LBF_MATH_LERP:
|
||||
recordResolvedType(node, &builtinTypes.numberType);
|
||||
break;
|
||||
|
||||
|
|
|
@ -305,7 +305,8 @@ static Error parseJson(const std::string& contents, Action action)
|
|||
arrayTop = (lexer.current().type == '[');
|
||||
next(lexer);
|
||||
}
|
||||
else if (lexer.current().type == Lexeme::QuotedString || lexer.current().type == Lexeme::ReservedTrue || lexer.current().type == Lexeme::ReservedFalse)
|
||||
else if (lexer.current().type == Lexeme::QuotedString || lexer.current().type == Lexeme::ReservedTrue ||
|
||||
lexer.current().type == Lexeme::ReservedFalse)
|
||||
{
|
||||
std::string value = lexer.current().type == Lexeme::QuotedString
|
||||
? std::string(lexer.current().data, lexer.current().getLength())
|
||||
|
|
|
@ -198,7 +198,13 @@ private:
|
|||
{
|
||||
// An e-node 𝑛 is canonical iff 𝑛 = canonicalize(𝑛), where
|
||||
// canonicalize(𝑓(𝑎1, 𝑎2, ...)) = 𝑓(find(𝑎1), find(𝑎2), ...).
|
||||
Luau::EqSat::canonicalize(enode, [&](Id id) { return find(id); });
|
||||
Luau::EqSat::canonicalize(
|
||||
enode,
|
||||
[&](Id id)
|
||||
{
|
||||
return find(id);
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
bool isCanonical(const L& enode) const
|
||||
|
|
|
@ -244,7 +244,7 @@ private:
|
|||
template<typename Phantom, typename T>
|
||||
struct NodeSet
|
||||
{
|
||||
template <typename P_, typename T_, typename Find>
|
||||
template<typename P_, typename T_, typename Find>
|
||||
friend void canonicalize(NodeSet<P_, T_>& node, Find&& find);
|
||||
|
||||
template<typename... Args>
|
||||
|
@ -302,7 +302,7 @@ struct Language final
|
|||
template<typename T>
|
||||
using WithinDomain = std::disjunction<std::is_same<std::decay_t<T>, Ts>...>;
|
||||
|
||||
template <typename Find, typename... Vs>
|
||||
template<typename Find, typename... Vs>
|
||||
friend void canonicalize(Language<Vs...>& enode, Find&& find);
|
||||
|
||||
template<typename T>
|
||||
|
@ -388,7 +388,7 @@ private:
|
|||
VariantTy v;
|
||||
};
|
||||
|
||||
template <typename Node, typename Find>
|
||||
template<typename Node, typename Find>
|
||||
void canonicalize(Node& node, Find&& find)
|
||||
{
|
||||
// An e-node 𝑛 is canonical iff 𝑛 = canonicalize(𝑛), where
|
||||
|
@ -398,7 +398,7 @@ void canonicalize(Node& node, Find&& find)
|
|||
}
|
||||
|
||||
// Canonicalizing the Ids in a NodeSet may result in the set decreasing in size.
|
||||
template <typename Phantom, typename T, typename Find>
|
||||
template<typename Phantom, typename T, typename Find>
|
||||
void canonicalize(NodeSet<Phantom, T>& node, Find&& find)
|
||||
{
|
||||
for (Id& id : node.vector)
|
||||
|
@ -409,7 +409,7 @@ void canonicalize(NodeSet<Phantom, T>& node, Find&& find)
|
|||
node.vector.erase(endIt, end(node.vector));
|
||||
}
|
||||
|
||||
template <typename Find, typename... Vs>
|
||||
template<typename Find, typename... Vs>
|
||||
void canonicalize(Language<Vs...>& enode, Find&& find)
|
||||
{
|
||||
visit(
|
||||
|
|
20
Makefile
20
Makefile
|
@ -42,23 +42,23 @@ ISOCLINE_SOURCES=extern/isocline/src/isocline.c
|
|||
ISOCLINE_OBJECTS=$(ISOCLINE_SOURCES:%=$(BUILD)/%.o)
|
||||
ISOCLINE_TARGET=$(BUILD)/libisocline.a
|
||||
|
||||
TESTS_SOURCES=$(wildcard tests/*.cpp) CLI/FileUtils.cpp CLI/Flags.cpp CLI/Profiler.cpp CLI/Coverage.cpp CLI/Repl.cpp CLI/Require.cpp
|
||||
TESTS_SOURCES=$(wildcard tests/*.cpp) CLI/src/FileUtils.cpp CLI/src/Flags.cpp CLI/src/Profiler.cpp CLI/src/Coverage.cpp CLI/src/Repl.cpp CLI/src/Require.cpp
|
||||
TESTS_OBJECTS=$(TESTS_SOURCES:%=$(BUILD)/%.o)
|
||||
TESTS_TARGET=$(BUILD)/luau-tests
|
||||
|
||||
REPL_CLI_SOURCES=CLI/FileUtils.cpp CLI/Flags.cpp CLI/Profiler.cpp CLI/Coverage.cpp CLI/Repl.cpp CLI/ReplEntry.cpp CLI/Require.cpp
|
||||
REPL_CLI_SOURCES=CLI/src/FileUtils.cpp CLI/src/Flags.cpp CLI/src/Profiler.cpp CLI/src/Coverage.cpp CLI/src/Repl.cpp CLI/src/ReplEntry.cpp CLI/src/Require.cpp
|
||||
REPL_CLI_OBJECTS=$(REPL_CLI_SOURCES:%=$(BUILD)/%.o)
|
||||
REPL_CLI_TARGET=$(BUILD)/luau
|
||||
|
||||
ANALYZE_CLI_SOURCES=CLI/FileUtils.cpp CLI/Flags.cpp CLI/Require.cpp CLI/Analyze.cpp
|
||||
ANALYZE_CLI_SOURCES=CLI/src/FileUtils.cpp CLI/src/Flags.cpp CLI/src/Require.cpp CLI/src/Analyze.cpp
|
||||
ANALYZE_CLI_OBJECTS=$(ANALYZE_CLI_SOURCES:%=$(BUILD)/%.o)
|
||||
ANALYZE_CLI_TARGET=$(BUILD)/luau-analyze
|
||||
|
||||
COMPILE_CLI_SOURCES=CLI/FileUtils.cpp CLI/Flags.cpp CLI/Compile.cpp
|
||||
COMPILE_CLI_SOURCES=CLI/src/FileUtils.cpp CLI/src/Flags.cpp CLI/src/Compile.cpp
|
||||
COMPILE_CLI_OBJECTS=$(COMPILE_CLI_SOURCES:%=$(BUILD)/%.o)
|
||||
COMPILE_CLI_TARGET=$(BUILD)/luau-compile
|
||||
|
||||
BYTECODE_CLI_SOURCES=CLI/FileUtils.cpp CLI/Flags.cpp CLI/Bytecode.cpp
|
||||
BYTECODE_CLI_SOURCES=CLI/src/FileUtils.cpp CLI/src/Flags.cpp CLI/src/Bytecode.cpp
|
||||
BYTECODE_CLI_OBJECTS=$(BYTECODE_CLI_SOURCES:%=$(BUILD)/%.o)
|
||||
BYTECODE_CLI_TARGET=$(BUILD)/luau-bytecode
|
||||
|
||||
|
@ -149,11 +149,11 @@ $(EQSAT_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IEqSat/include
|
|||
$(CODEGEN_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -ICodeGen/include -IVM/include -IVM/src # Code generation needs VM internals
|
||||
$(VM_OBJECTS): CXXFLAGS+=-std=c++11 -ICommon/include -IVM/include
|
||||
$(ISOCLINE_OBJECTS): CXXFLAGS+=-Wno-unused-function -Iextern/isocline/include
|
||||
$(TESTS_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IConfig/include -IAnalysis/include -IEqSat/include -ICodeGen/include -IVM/include -ICLI -Iextern -DDOCTEST_CONFIG_DOUBLE_STRINGIFY
|
||||
$(REPL_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IVM/include -ICodeGen/include -Iextern -Iextern/isocline/include
|
||||
$(ANALYZE_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -IAnalysis/include -IEqSat/include -IConfig/include -Iextern
|
||||
$(COMPILE_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IVM/include -ICodeGen/include
|
||||
$(BYTECODE_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IVM/include -ICodeGen/include
|
||||
$(TESTS_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IConfig/include -IAnalysis/include -IEqSat/include -ICodeGen/include -IVM/include -ICLI/include -Iextern -DDOCTEST_CONFIG_DOUBLE_STRINGIFY
|
||||
$(REPL_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IVM/include -ICodeGen/include -Iextern -Iextern/isocline/include -ICLI/include
|
||||
$(ANALYZE_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -IAnalysis/include -IEqSat/include -IConfig/include -Iextern -ICLI/include
|
||||
$(COMPILE_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IVM/include -ICodeGen/include -ICLI/include
|
||||
$(BYTECODE_CLI_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IVM/include -ICodeGen/include -ICLI/include
|
||||
$(FUZZ_OBJECTS): CXXFLAGS+=-std=c++17 -ICommon/include -IAst/include -ICompiler/include -IAnalysis/include -IEqSat/include -IVM/include -ICodeGen/include -IConfig/include
|
||||
|
||||
$(TESTS_TARGET): LDFLAGS+=-lpthread
|
||||
|
|
|
@ -389,36 +389,39 @@ target_sources(isocline PRIVATE
|
|||
|
||||
# Common sources shared between all CLI apps
|
||||
target_sources(Luau.CLI.lib PRIVATE
|
||||
CLI/FileUtils.cpp
|
||||
CLI/Flags.cpp
|
||||
CLI/Flags.h
|
||||
CLI/FileUtils.h
|
||||
CLI/include/Luau/FileUtils.h
|
||||
CLI/include/Luau/Flags.h
|
||||
CLI/include/Luau/Require.h
|
||||
|
||||
CLI/src/FileUtils.cpp
|
||||
CLI/src/Flags.cpp
|
||||
CLI/src/Require.cpp
|
||||
)
|
||||
|
||||
if(TARGET Luau.Repl.CLI)
|
||||
# Luau.Repl.CLI Sources
|
||||
target_sources(Luau.Repl.CLI PRIVATE
|
||||
CLI/Coverage.h
|
||||
CLI/Coverage.cpp
|
||||
CLI/Profiler.h
|
||||
CLI/Profiler.cpp
|
||||
CLI/Repl.cpp
|
||||
CLI/ReplEntry.cpp
|
||||
CLI/Require.cpp)
|
||||
CLI/include/Luau/Coverage.h
|
||||
CLI/include/Luau/Profiler.h
|
||||
|
||||
CLI/src/Coverage.cpp
|
||||
CLI/src/Profiler.cpp
|
||||
CLI/src/Repl.cpp
|
||||
CLI/src/ReplEntry.cpp
|
||||
)
|
||||
endif()
|
||||
|
||||
if(TARGET Luau.Analyze.CLI)
|
||||
# Luau.Analyze.CLI Sources
|
||||
target_sources(Luau.Analyze.CLI PRIVATE
|
||||
CLI/Analyze.cpp
|
||||
CLI/Require.cpp
|
||||
CLI/src/Analyze.cpp
|
||||
)
|
||||
endif()
|
||||
|
||||
if(TARGET Luau.Ast.CLI)
|
||||
# Luau.Ast.CLI Sources
|
||||
target_sources(Luau.Ast.CLI PRIVATE
|
||||
CLI/Ast.cpp
|
||||
CLI/src/Ast.cpp
|
||||
)
|
||||
endif()
|
||||
|
||||
|
@ -543,12 +546,12 @@ endif()
|
|||
if(TARGET Luau.CLI.Test)
|
||||
# Luau.CLI.Test Sources
|
||||
target_sources(Luau.CLI.Test PRIVATE
|
||||
CLI/Coverage.h
|
||||
CLI/Coverage.cpp
|
||||
CLI/Profiler.h
|
||||
CLI/Profiler.cpp
|
||||
CLI/Repl.cpp
|
||||
CLI/Require.cpp
|
||||
CLI/include/Luau/Coverage.h
|
||||
CLI/include/Luau/Profiler.h
|
||||
|
||||
CLI/src/Coverage.cpp
|
||||
CLI/src/Profiler.cpp
|
||||
CLI/src/Repl.cpp
|
||||
|
||||
tests/RegisterCallbacks.h
|
||||
tests/RegisterCallbacks.cpp
|
||||
|
@ -560,24 +563,24 @@ endif()
|
|||
if(TARGET Luau.Web)
|
||||
# Luau.Web Sources
|
||||
target_sources(Luau.Web PRIVATE
|
||||
CLI/Web.cpp)
|
||||
CLI/src/Web.cpp)
|
||||
endif()
|
||||
|
||||
if(TARGET Luau.Reduce.CLI)
|
||||
# Luau.Reduce.CLI Sources
|
||||
target_sources(Luau.Reduce.CLI PRIVATE
|
||||
CLI/Reduce.cpp
|
||||
CLI/src/Reduce.cpp
|
||||
)
|
||||
endif()
|
||||
|
||||
if(TARGET Luau.Compile.CLI)
|
||||
# Luau.Compile.CLI Sources
|
||||
target_sources(Luau.Compile.CLI PRIVATE
|
||||
CLI/Compile.cpp)
|
||||
CLI/src/Compile.cpp)
|
||||
endif()
|
||||
|
||||
if(TARGET Luau.Bytecode.CLI)
|
||||
# Luau.Bytecode.CLI Sources
|
||||
target_sources(Luau.Bytecode.CLI PRIVATE
|
||||
CLI/Bytecode.cpp)
|
||||
CLI/src/Bytecode.cpp)
|
||||
endif()
|
||||
|
|
|
@ -10,6 +10,8 @@
|
|||
|
||||
#include <string.h>
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauBufferBitMethods)
|
||||
|
||||
// while C API returns 'size_t' for binary compatibility in case of future extensions,
|
||||
// in the current implementation, length and offset are limited to 31 bits
|
||||
// because offset is limited to an integer, a single 64bit comparison can be used and will not overflow
|
||||
|
@ -247,7 +249,68 @@ static int buffer_fill(lua_State* L)
|
|||
return 0;
|
||||
}
|
||||
|
||||
static const luaL_Reg bufferlib[] = {
|
||||
static int buffer_readbits(lua_State* L)
|
||||
{
|
||||
size_t len = 0;
|
||||
void* buf = luaL_checkbuffer(L, 1, &len);
|
||||
int64_t bitoffset = (int64_t)luaL_checknumber(L, 2);
|
||||
int bitcount = luaL_checkinteger(L, 3);
|
||||
|
||||
if (bitoffset < 0)
|
||||
luaL_error(L, "buffer access out of bounds");
|
||||
|
||||
if (unsigned(bitcount) > 32)
|
||||
luaL_error(L, "bit count is out of range of [0; 32]");
|
||||
|
||||
if (uint64_t(bitoffset + bitcount) > uint64_t(len) * 8)
|
||||
luaL_error(L, "buffer access out of bounds");
|
||||
|
||||
unsigned startbyte = unsigned(bitoffset / 8);
|
||||
unsigned endbyte = unsigned((bitoffset + bitcount + 7) / 8);
|
||||
|
||||
uint64_t data = 0;
|
||||
memcpy(&data, (char*)buf + startbyte, endbyte - startbyte);
|
||||
|
||||
uint64_t subbyteoffset = bitoffset & 0x7;
|
||||
uint64_t mask = (1ull << bitcount) - 1;
|
||||
|
||||
lua_pushunsigned(L, unsigned((data >> subbyteoffset) & mask));
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int buffer_writebits(lua_State* L)
|
||||
{
|
||||
size_t len = 0;
|
||||
void* buf = luaL_checkbuffer(L, 1, &len);
|
||||
int64_t bitoffset = (int64_t)luaL_checknumber(L, 2);
|
||||
int bitcount = luaL_checkinteger(L, 3);
|
||||
unsigned value = luaL_checkunsigned(L, 4);
|
||||
|
||||
if (bitoffset < 0)
|
||||
luaL_error(L, "buffer access out of bounds");
|
||||
|
||||
if (unsigned(bitcount) > 32)
|
||||
luaL_error(L, "bit count is out of range of [0; 32]");
|
||||
|
||||
if (uint64_t(bitoffset + bitcount) > uint64_t(len) * 8)
|
||||
luaL_error(L, "buffer access out of bounds");
|
||||
|
||||
unsigned startbyte = unsigned(bitoffset / 8);
|
||||
unsigned endbyte = unsigned((bitoffset + bitcount + 7) / 8);
|
||||
|
||||
uint64_t data = 0;
|
||||
memcpy(&data, (char*)buf + startbyte, endbyte - startbyte);
|
||||
|
||||
uint64_t subbyteoffset = bitoffset & 0x7;
|
||||
uint64_t mask = ((1ull << bitcount) - 1) << subbyteoffset;
|
||||
|
||||
data = (data & ~mask) | ((uint64_t(value) << subbyteoffset) & mask);
|
||||
|
||||
memcpy((char*)buf + startbyte, &data, endbyte - startbyte);
|
||||
return 0;
|
||||
}
|
||||
|
||||
static const luaL_Reg bufferlib_DEPRECATED[] = {
|
||||
{"create", buffer_create},
|
||||
{"fromstring", buffer_fromstring},
|
||||
{"tostring", buffer_tostring},
|
||||
|
@ -275,9 +338,39 @@ static const luaL_Reg bufferlib[] = {
|
|||
{NULL, NULL},
|
||||
};
|
||||
|
||||
static const luaL_Reg bufferlib[] = {
|
||||
{"create", buffer_create},
|
||||
{"fromstring", buffer_fromstring},
|
||||
{"tostring", buffer_tostring},
|
||||
{"readi8", buffer_readinteger<int8_t>},
|
||||
{"readu8", buffer_readinteger<uint8_t>},
|
||||
{"readi16", buffer_readinteger<int16_t>},
|
||||
{"readu16", buffer_readinteger<uint16_t>},
|
||||
{"readi32", buffer_readinteger<int32_t>},
|
||||
{"readu32", buffer_readinteger<uint32_t>},
|
||||
{"readf32", buffer_readfp<float, uint32_t>},
|
||||
{"readf64", buffer_readfp<double, uint64_t>},
|
||||
{"writei8", buffer_writeinteger<int8_t>},
|
||||
{"writeu8", buffer_writeinteger<uint8_t>},
|
||||
{"writei16", buffer_writeinteger<int16_t>},
|
||||
{"writeu16", buffer_writeinteger<uint16_t>},
|
||||
{"writei32", buffer_writeinteger<int32_t>},
|
||||
{"writeu32", buffer_writeinteger<uint32_t>},
|
||||
{"writef32", buffer_writefp<float, uint32_t>},
|
||||
{"writef64", buffer_writefp<double, uint64_t>},
|
||||
{"readstring", buffer_readstring},
|
||||
{"writestring", buffer_writestring},
|
||||
{"len", buffer_len},
|
||||
{"copy", buffer_copy},
|
||||
{"fill", buffer_fill},
|
||||
{"readbits", buffer_readbits},
|
||||
{"writebits", buffer_writebits},
|
||||
{NULL, NULL},
|
||||
};
|
||||
|
||||
int luaopen_buffer(lua_State* L)
|
||||
{
|
||||
luaL_register(L, LUA_BUFFERLIBNAME, bufferlib);
|
||||
luaL_register(L, LUA_BUFFERLIBNAME, FFlag::LuauBufferBitMethods ? bufferlib : bufferlib_DEPRECATED);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
|
|
@ -1730,6 +1730,23 @@ static int luauF_vectormax(lua_State* L, StkId res, TValue* arg0, int nresults,
|
|||
return -1;
|
||||
}
|
||||
|
||||
static int luauF_lerp(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||||
{
|
||||
if (nparams >= 3 && nresults <= 1 && ttisnumber(arg0) && ttisnumber(args) && ttisnumber(args + 1))
|
||||
{
|
||||
double a = nvalue(arg0);
|
||||
double b = nvalue(args);
|
||||
double t = nvalue(args + 1);
|
||||
|
||||
double r = (t == 1.0) ? b : a + (b - a) * t;
|
||||
|
||||
setnvalue(res, r);
|
||||
return 1;
|
||||
}
|
||||
|
||||
return -1;
|
||||
}
|
||||
|
||||
static int luauF_missing(lua_State* L, StkId res, TValue* arg0, int nresults, StkId args, int nparams)
|
||||
{
|
||||
return -1;
|
||||
|
@ -1925,6 +1942,8 @@ const luau_FastFunction luauF_table[256] = {
|
|||
luauF_vectormin,
|
||||
luauF_vectormax,
|
||||
|
||||
luauF_lerp,
|
||||
|
||||
// When adding builtins, add them above this line; what follows is 64 "dummy" entries with luauF_missing fallback.
|
||||
// This is important so that older versions of the runtime that don't support newer builtins automatically fall back via luauF_missing.
|
||||
// Given the builtin addition velocity this should always provide a larger compatibility window than bytecode versions suggest.
|
||||
|
|
|
@ -18,6 +18,7 @@
|
|||
#include <string.h>
|
||||
|
||||
LUAU_DYNAMIC_FASTFLAGVARIABLE(LuauStackLimit, false)
|
||||
LUAU_DYNAMIC_FASTFLAGVARIABLE(LuauPopIncompleteCi, false)
|
||||
|
||||
// keep max stack allocation request under 1GB
|
||||
#define MAX_STACK_SIZE (int(1024 / sizeof(TValue)) * 1024 * 1024)
|
||||
|
@ -179,11 +180,23 @@ static void correctstack(lua_State* L, TValue* oldstack)
|
|||
L->base = (L->base - oldstack) + L->stack;
|
||||
}
|
||||
|
||||
void luaD_reallocstack(lua_State* L, int newsize)
|
||||
void luaD_reallocstack(lua_State* L, int newsize, int fornewci)
|
||||
{
|
||||
// throw 'out of memory' error because space for a custom error message cannot be guaranteed here
|
||||
if (DFFlag::LuauStackLimit && newsize > MAX_STACK_SIZE)
|
||||
{
|
||||
// reallocation was performaed to setup a new CallInfo frame, which we have to remove
|
||||
if (DFFlag::LuauPopIncompleteCi && fornewci)
|
||||
{
|
||||
CallInfo* cip = L->ci - 1;
|
||||
|
||||
L->ci = cip;
|
||||
L->base = cip->base;
|
||||
L->top = cip->top;
|
||||
}
|
||||
|
||||
luaD_throw(L, LUA_ERRMEM);
|
||||
}
|
||||
|
||||
TValue* oldstack = L->stack;
|
||||
int realsize = newsize + EXTRA_STACK;
|
||||
|
@ -208,10 +221,17 @@ void luaD_reallocCI(lua_State* L, int newsize)
|
|||
|
||||
void luaD_growstack(lua_State* L, int n)
|
||||
{
|
||||
if (n <= L->stacksize) // double size is enough?
|
||||
luaD_reallocstack(L, 2 * L->stacksize);
|
||||
if (DFFlag::LuauPopIncompleteCi)
|
||||
{
|
||||
luaD_reallocstack(L, getgrownstacksize(L, n), 0);
|
||||
}
|
||||
else
|
||||
luaD_reallocstack(L, L->stacksize + n);
|
||||
{
|
||||
if (n <= L->stacksize) // double size is enough?
|
||||
luaD_reallocstack(L, 2 * L->stacksize, 0);
|
||||
else
|
||||
luaD_reallocstack(L, L->stacksize + n, 0);
|
||||
}
|
||||
}
|
||||
|
||||
CallInfo* luaD_growCI(lua_State* L)
|
||||
|
|
14
VM/src/ldo.h
14
VM/src/ldo.h
|
@ -7,11 +7,21 @@
|
|||
#include "luaconf.h"
|
||||
#include "ldebug.h"
|
||||
|
||||
// returns target stack for 'n' extra elements to reallocate
|
||||
// if possible, stack size growth factor is 2x
|
||||
#define getgrownstacksize(L, n) ((n) <= L->stacksize ? 2 * L->stacksize : L->stacksize + (n))
|
||||
|
||||
#define luaD_checkstackfornewci(L, n) \
|
||||
if ((char*)L->stack_last - (char*)L->top <= (n) * (int)sizeof(TValue)) \
|
||||
luaD_reallocstack(L, getgrownstacksize(L, (n)), 1); \
|
||||
else \
|
||||
condhardstacktests(luaD_reallocstack(L, L->stacksize - EXTRA_STACK, 1));
|
||||
|
||||
#define luaD_checkstack(L, n) \
|
||||
if ((char*)L->stack_last - (char*)L->top <= (n) * (int)sizeof(TValue)) \
|
||||
luaD_growstack(L, n); \
|
||||
else \
|
||||
condhardstacktests(luaD_reallocstack(L, L->stacksize - EXTRA_STACK));
|
||||
condhardstacktests(luaD_reallocstack(L, L->stacksize - EXTRA_STACK, 0));
|
||||
|
||||
#define incr_top(L) \
|
||||
{ \
|
||||
|
@ -47,7 +57,7 @@ LUAI_FUNC CallInfo* luaD_growCI(lua_State* L);
|
|||
LUAI_FUNC void luaD_call(lua_State* L, StkId func, int nresults);
|
||||
LUAI_FUNC int luaD_pcall(lua_State* L, Pfunc func, void* u, ptrdiff_t oldtop, ptrdiff_t ef);
|
||||
LUAI_FUNC void luaD_reallocCI(lua_State* L, int newsize);
|
||||
LUAI_FUNC void luaD_reallocstack(lua_State* L, int newsize);
|
||||
LUAI_FUNC void luaD_reallocstack(lua_State* L, int newsize, int fornewci);
|
||||
LUAI_FUNC void luaD_growstack(lua_State* L, int n);
|
||||
LUAI_FUNC void luaD_checkCstack(lua_State* L);
|
||||
|
||||
|
|
|
@ -442,7 +442,7 @@ static void shrinkstack(lua_State* L)
|
|||
condhardstacktests(luaD_reallocCI(L, ci_used + 1));
|
||||
|
||||
if (3 * size_t(s_used) < size_t(L->stacksize) && 2 * (BASIC_STACK_SIZE + EXTRA_STACK) < L->stacksize)
|
||||
luaD_reallocstack(L, L->stacksize / 2); // still big enough...
|
||||
luaD_reallocstack(L, L->stacksize / 2, 0); // still big enough...
|
||||
condhardstacktests(luaD_reallocstack(L, s_used));
|
||||
}
|
||||
|
||||
|
|
|
@ -8,6 +8,7 @@
|
|||
#include <time.h>
|
||||
|
||||
LUAU_FASTFLAGVARIABLE(LuauMathMap)
|
||||
LUAU_FASTFLAGVARIABLE(LuauMathLerp)
|
||||
|
||||
#undef PI
|
||||
#define PI (3.14159265358979323846)
|
||||
|
@ -418,6 +419,17 @@ static int math_map(lua_State* L)
|
|||
return 1;
|
||||
}
|
||||
|
||||
static int math_lerp(lua_State* L)
|
||||
{
|
||||
double a = luaL_checknumber(L, 1);
|
||||
double b = luaL_checknumber(L, 2);
|
||||
double t = luaL_checknumber(L, 3);
|
||||
|
||||
double r = (t == 1.0) ? b : a + (b - a) * t;
|
||||
lua_pushnumber(L, r);
|
||||
return 1;
|
||||
}
|
||||
|
||||
static const luaL_Reg mathlib[] = {
|
||||
{"abs", math_abs},
|
||||
{"acos", math_acos},
|
||||
|
@ -451,6 +463,7 @@ static const luaL_Reg mathlib[] = {
|
|||
{"clamp", math_clamp},
|
||||
{"sign", math_sign},
|
||||
{"round", math_round},
|
||||
{"map", math_map},
|
||||
{NULL, NULL},
|
||||
};
|
||||
|
||||
|
@ -471,10 +484,16 @@ int luaopen_math(lua_State* L)
|
|||
lua_pushnumber(L, HUGE_VAL);
|
||||
lua_setfield(L, -2, "huge");
|
||||
|
||||
if (FFlag::LuauMathMap)
|
||||
if (FFlag::LuauMathLerp)
|
||||
{
|
||||
lua_pushcfunction(L, math_map, "map");
|
||||
lua_setfield(L, -2, "map");
|
||||
lua_pushcfunction(L, math_lerp, "lerp");
|
||||
lua_setfield(L, -2, "lerp");
|
||||
}
|
||||
|
||||
if (FFlag::LuauMathLerp)
|
||||
{
|
||||
lua_pushcfunction(L, math_lerp, "lerp");
|
||||
lua_setfield(L, -2, "lerp");
|
||||
}
|
||||
|
||||
return 1;
|
||||
|
|
|
@ -192,7 +192,7 @@ struct SizeClassConfig
|
|||
const SizeClassConfig kSizeClassConfig;
|
||||
|
||||
// size class for a block of size sz; returns -1 for size=0 because empty allocations take no space
|
||||
#define sizeclass(sz) (size_t((sz)-1) < kMaxSmallSizeUsed ? kSizeClassConfig.classForSize[sz] : -1)
|
||||
#define sizeclass(sz) (size_t((sz) - 1) < kMaxSmallSizeUsed ? kSizeClassConfig.classForSize[sz] : -1)
|
||||
|
||||
// metadata for a block is stored in the first pointer of the block
|
||||
#define metadata(block) (*(void**)(block))
|
||||
|
|
|
@ -149,7 +149,7 @@ void lua_resetthread(lua_State* L)
|
|||
L->nCcalls = L->baseCcalls = 0;
|
||||
// clear thread stack
|
||||
if (L->stacksize != BASIC_STACK_SIZE + EXTRA_STACK)
|
||||
luaD_reallocstack(L, BASIC_STACK_SIZE);
|
||||
luaD_reallocstack(L, BASIC_STACK_SIZE, 0);
|
||||
for (int i = 0; i < L->stacksize; i++)
|
||||
setnilvalue(L->stack + i);
|
||||
}
|
||||
|
|
|
@ -16,6 +16,8 @@
|
|||
|
||||
#include <string.h>
|
||||
|
||||
LUAU_DYNAMIC_FASTFLAG(LuauPopIncompleteCi)
|
||||
|
||||
// Disable c99-designator to avoid the warning in CGOTO dispatch table
|
||||
#ifdef __clang__
|
||||
#if __has_warning("-Wc99-designator")
|
||||
|
@ -935,7 +937,14 @@ reentry:
|
|||
// note: this reallocs stack, but we don't need to VM_PROTECT this
|
||||
// this is because we're going to modify base/savedpc manually anyhow
|
||||
// crucially, we can't use ra/argtop after this line
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
if (DFFlag::LuauPopIncompleteCi)
|
||||
{
|
||||
luaD_checkstackfornewci(L, ccl->stacksize);
|
||||
}
|
||||
else
|
||||
{
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
}
|
||||
|
||||
LUAU_ASSERT(ci->top <= L->stack_last);
|
||||
|
||||
|
@ -3071,7 +3080,14 @@ int luau_precall(lua_State* L, StkId func, int nresults)
|
|||
L->base = ci->base;
|
||||
// Note: L->top is assigned externally
|
||||
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
if (DFFlag::LuauPopIncompleteCi)
|
||||
{
|
||||
luaD_checkstackfornewci(L, ccl->stacksize);
|
||||
}
|
||||
else
|
||||
{
|
||||
luaD_checkstack(L, ccl->stacksize);
|
||||
}
|
||||
LUAU_ASSERT(ci->top <= L->stack_last);
|
||||
|
||||
if (!ccl->isC)
|
||||
|
|
|
@ -18,6 +18,7 @@ LUAU_FASTFLAG(LuauSolverV2)
|
|||
LUAU_FASTFLAG(DebugLuauFreezeArena)
|
||||
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
||||
LUAU_FASTFLAG(StudioReportLuauAny2)
|
||||
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope)
|
||||
|
||||
|
||||
struct ATSFixture : BuiltinsFixture
|
||||
|
@ -97,7 +98,10 @@ end
|
|||
|
||||
LUAU_ASSERT(module->ats.typeInfo.size() == 3);
|
||||
LUAU_ASSERT(module->ats.typeInfo[1].code == Pattern::TypePk);
|
||||
LUAU_ASSERT(module->ats.typeInfo[0].node == "local function fallible(t: number): ...any\n if t > 0 then\n return true, t\n end\n return false, 'must be positive'\nend");
|
||||
LUAU_ASSERT(
|
||||
module->ats.typeInfo[0].node ==
|
||||
"local function fallible(t: number): ...any\n if t > 0 then\n return true, t\n end\n return false, 'must be positive'\nend"
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ATSFixture, "typepacks_no_ret")
|
||||
|
@ -581,6 +585,9 @@ TEST_CASE_FIXTURE(ATSFixture, "racing_spawning_1")
|
|||
ScopedFastFlag sff[] = {
|
||||
{FFlag::LuauSolverV2, true},
|
||||
{FFlag::StudioReportLuauAny2, true},
|
||||
// Previously we'd report an error because number <: 'a is not a
|
||||
// supertype.
|
||||
{FFlag::LuauTrackInteriorFreeTypesOnScope, true}
|
||||
};
|
||||
|
||||
fileResolver.source["game/Gui/Modules/A"] = R"(
|
||||
|
@ -632,7 +639,7 @@ initialize()
|
|||
)";
|
||||
|
||||
CheckResult result1 = frontend.check("game/Gui/Modules/A");
|
||||
LUAU_REQUIRE_ERROR_COUNT(5, result1);
|
||||
LUAU_REQUIRE_ERROR_COUNT(4, result1);
|
||||
|
||||
ModulePtr module = frontend.moduleResolver.getModule("game/Gui/Modules/A");
|
||||
|
||||
|
|
|
@ -506,6 +506,7 @@ TEST_CASE_FIXTURE(AssemblyBuilderX64Fixture, "AVXBinaryInstructionForms")
|
|||
SINGLE_COMPARE(vmaxsd(xmm8, xmm10, xmm14), 0xc4, 0x41, 0x2b, 0x5f, 0xc6);
|
||||
SINGLE_COMPARE(vminsd(xmm8, xmm10, xmm14), 0xc4, 0x41, 0x2b, 0x5d, 0xc6);
|
||||
|
||||
SINGLE_COMPARE(vcmpeqsd(xmm8, xmm10, xmm14), 0xc4, 0x41, 0x2b, 0xc2, 0xc6, 0x00);
|
||||
SINGLE_COMPARE(vcmpltsd(xmm8, xmm10, xmm14), 0xc4, 0x41, 0x2b, 0xc2, 0xc6, 0x01);
|
||||
}
|
||||
|
||||
|
|
|
@ -2232,7 +2232,7 @@ local ec = e(f@5)
|
|||
TEST_CASE_FIXTURE(ACFixture, "type_correct_suggestion_for_overloads")
|
||||
{
|
||||
if (FFlag::LuauSolverV2) // CLI-116814 Autocomplete needs to populate expected types for function arguments correctly
|
||||
// (overloads and singletons)
|
||||
// (overloads and singletons)
|
||||
return;
|
||||
check(R"(
|
||||
local target: ((number) -> string) & ((string) -> number))
|
||||
|
@ -2582,7 +2582,7 @@ end
|
|||
TEST_CASE_FIXTURE(ACFixture, "suggest_table_keys")
|
||||
{
|
||||
if (FFlag::LuauSolverV2) // CLI-116812 AutocompleteTest.suggest_table_keys needs to populate expected types for nested
|
||||
// tables without an annotation
|
||||
// tables without an annotation
|
||||
return;
|
||||
|
||||
check(R"(
|
||||
|
@ -3069,7 +3069,7 @@ TEST_CASE_FIXTURE(ACBuiltinsFixture, "autocomplete_on_string_singletons")
|
|||
TEST_CASE_FIXTURE(ACFixture, "autocomplete_string_singletons")
|
||||
{
|
||||
if (FFlag::LuauSolverV2) // CLI-116814 Autocomplete needs to populate expected types for function arguments correctly
|
||||
// (overloads and singletons)
|
||||
// (overloads and singletons)
|
||||
return;
|
||||
|
||||
check(R"(
|
||||
|
@ -4293,8 +4293,7 @@ end
|
|||
foo(@1)
|
||||
)");
|
||||
|
||||
const std::optional<std::string> EXPECTED_INSERT =
|
||||
FFlag::LuauSolverV2 ? "function(...: number): number end" : "function(...): number end";
|
||||
const std::optional<std::string> EXPECTED_INSERT = FFlag::LuauSolverV2 ? "function(...: number): number end" : "function(...): number end";
|
||||
|
||||
auto ac = autocomplete('1');
|
||||
|
||||
|
|
|
@ -23,10 +23,8 @@ LUAU_FASTINT(LuauCompileInlineThresholdMaxBoost)
|
|||
LUAU_FASTINT(LuauCompileLoopUnrollThreshold)
|
||||
LUAU_FASTINT(LuauCompileLoopUnrollThresholdMaxBoost)
|
||||
LUAU_FASTINT(LuauRecursionLimit)
|
||||
LUAU_FASTFLAG(LuauCompileVectorTypeInfo)
|
||||
LUAU_FASTFLAG(LuauCompileOptimizeRevArith)
|
||||
LUAU_FASTFLAG(LuauCompileLibraryConstants)
|
||||
LUAU_FASTFLAG(LuauVectorBuiltins)
|
||||
LUAU_FASTFLAG(LuauVectorFolding)
|
||||
LUAU_FASTFLAG(LuauVector2Constants)
|
||||
LUAU_FASTFLAG(LuauCompileDisabledBuiltins)
|
||||
|
@ -1493,7 +1491,6 @@ RETURN R0 1
|
|||
|
||||
TEST_CASE("ConstantFoldVectorArith")
|
||||
{
|
||||
ScopedFastFlag luauVectorBuiltins{FFlag::LuauVectorBuiltins, true};
|
||||
ScopedFastFlag luauVectorFolding{FFlag::LuauVectorFolding, true};
|
||||
|
||||
CHECK_EQ("\n" + compileFunction("local n = 2; local a, b = vector.create(1, 2, 3), vector.create(2, 4, 8); return a + b", 0, 2), R"(
|
||||
|
@ -1561,7 +1558,6 @@ RETURN R0 1
|
|||
|
||||
TEST_CASE("ConstantFoldVectorArith4Wide")
|
||||
{
|
||||
ScopedFastFlag luauVectorBuiltins{FFlag::LuauVectorBuiltins, true};
|
||||
ScopedFastFlag luauVectorFolding{FFlag::LuauVectorFolding, true};
|
||||
|
||||
CHECK_EQ("\n" + compileFunction("local n = 2; local a, b = vector.create(1, 2, 3, 4), vector.create(2, 4, 8, 1); return a + b", 0, 2), R"(
|
||||
|
@ -5152,7 +5148,6 @@ RETURN R0 1
|
|||
|
||||
TEST_CASE("VectorConstantFields")
|
||||
{
|
||||
ScopedFastFlag luauVectorBuiltins{FFlag::LuauVectorBuiltins, true};
|
||||
ScopedFastFlag luauCompileLibraryConstants{FFlag::LuauCompileLibraryConstants, true};
|
||||
|
||||
CHECK_EQ("\n" + compileFunction("return vector.one, vector.zero", 0, 2), R"(
|
||||
|
@ -8681,8 +8676,6 @@ end
|
|||
|
||||
TEST_CASE("BuiltinTypeVector")
|
||||
{
|
||||
ScopedFastFlag luauCompileVectorTypeInfo{FFlag::LuauCompileVectorTypeInfo, true};
|
||||
|
||||
CHECK_EQ(
|
||||
"\n" + compileTypeTable(R"(
|
||||
function myfunc(test: Instance, pos: vector)
|
||||
|
|
|
@ -31,17 +31,17 @@ extern int optimizationLevel;
|
|||
void luaC_fullgc(lua_State* L);
|
||||
void luaC_validate(lua_State* L);
|
||||
|
||||
LUAU_FASTFLAG(LuauMathMap)
|
||||
LUAU_FASTFLAG(LuauMathLerp)
|
||||
LUAU_FASTFLAG(DebugLuauAbortingChecks)
|
||||
LUAU_FASTINT(CodegenHeuristicsInstructionLimit)
|
||||
LUAU_DYNAMIC_FASTFLAG(LuauStackLimit)
|
||||
LUAU_FASTFLAG(LuauVectorDefinitions)
|
||||
LUAU_DYNAMIC_FASTFLAG(LuauDebugInfoInvArgLeftovers)
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeCodegen)
|
||||
LUAU_FASTFLAG(LuauVectorLibNativeDot)
|
||||
LUAU_FASTFLAG(LuauVectorBuiltins)
|
||||
LUAU_FASTFLAG(LuauVectorMetatable)
|
||||
LUAU_FASTFLAG(LuauVector2Constructor)
|
||||
LUAU_FASTFLAG(LuauBufferBitMethods)
|
||||
LUAU_FASTFLAG(LuauCodeGenLimitLiveSlotReuse)
|
||||
|
||||
static lua_CompileOptions defaultOptions()
|
||||
{
|
||||
|
@ -655,12 +655,14 @@ TEST_CASE("Basic")
|
|||
|
||||
TEST_CASE("Buffers")
|
||||
{
|
||||
ScopedFastFlag luauBufferBitMethods{FFlag::LuauBufferBitMethods, true};
|
||||
|
||||
runConformance("buffers.lua");
|
||||
}
|
||||
|
||||
TEST_CASE("Math")
|
||||
{
|
||||
ScopedFastFlag LuauMathMap{FFlag::LuauMathMap, true};
|
||||
ScopedFastFlag LuauMathLerp{FFlag::LuauMathLerp, true};
|
||||
|
||||
runConformance("math.lua");
|
||||
}
|
||||
|
@ -892,7 +894,6 @@ TEST_CASE("Vector")
|
|||
|
||||
TEST_CASE("VectorLibrary")
|
||||
{
|
||||
ScopedFastFlag luauVectorBuiltins{FFlag::LuauVectorBuiltins, true};
|
||||
ScopedFastFlag luauVectorLibNativeCodegen{FFlag::LuauVectorLibNativeCodegen, true};
|
||||
ScopedFastFlag luauVectorLibNativeDot{FFlag::LuauVectorLibNativeDot, true};
|
||||
ScopedFastFlag luauVectorMetatable{FFlag::LuauVectorMetatable, true};
|
||||
|
@ -913,9 +914,7 @@ TEST_CASE("VectorLibrary")
|
|||
copts.optimizationLevel = 2;
|
||||
}
|
||||
|
||||
runConformance(
|
||||
"vector_library.lua", [](lua_State* L) {}, nullptr, nullptr, &copts
|
||||
);
|
||||
runConformance("vector_library.lua", [](lua_State* L) {}, nullptr, nullptr, &copts);
|
||||
}
|
||||
|
||||
static void populateRTTI(lua_State* L, Luau::TypeId type)
|
||||
|
@ -991,6 +990,7 @@ TEST_CASE("Types")
|
|||
{
|
||||
ScopedFastFlag luauVectorDefinitions{FFlag::LuauVectorDefinitions, true};
|
||||
ScopedFastFlag luauVector2Constructor{FFlag::LuauVector2Constructor, true};
|
||||
ScopedFastFlag luauMathLerp{FFlag::LuauMathLerp, false}; // waiting for math.lerp to be added to embedded type definitions
|
||||
|
||||
runConformance(
|
||||
"types.lua",
|
||||
|
@ -2581,6 +2581,8 @@ TEST_CASE("SafeEnv")
|
|||
|
||||
TEST_CASE("Native")
|
||||
{
|
||||
ScopedFastFlag luauCodeGenLimitLiveSlotReuse{FFlag::LuauCodeGenLimitLiveSlotReuse, true};
|
||||
|
||||
// This tests requires code to run natively, otherwise all 'is_native' checks will fail
|
||||
if (!codegen || !luau_codegen_supported())
|
||||
return;
|
||||
|
|
|
@ -23,15 +23,11 @@ struct ESFixture : Fixture
|
|||
TypeId genericT = arena_.addType(GenericType{"T"});
|
||||
TypeId genericU = arena_.addType(GenericType{"U"});
|
||||
|
||||
TypeId numberToString = arena_.addType(FunctionType{
|
||||
arena_.addTypePack({builtinTypes->numberType}),
|
||||
arena_.addTypePack({builtinTypes->stringType})
|
||||
});
|
||||
TypeId numberToString =
|
||||
arena_.addType(FunctionType{arena_.addTypePack({builtinTypes->numberType}), arena_.addTypePack({builtinTypes->stringType})});
|
||||
|
||||
TypeId stringToNumber = arena_.addType(FunctionType{
|
||||
arena_.addTypePack({builtinTypes->stringType}),
|
||||
arena_.addTypePack({builtinTypes->numberType})
|
||||
});
|
||||
TypeId stringToNumber =
|
||||
arena_.addType(FunctionType{arena_.addTypePack({builtinTypes->stringType}), arena_.addTypePack({builtinTypes->numberType})});
|
||||
|
||||
ESFixture()
|
||||
: simplifier(newSimplifier(arena, builtinTypes))
|
||||
|
@ -163,10 +159,11 @@ TEST_CASE_FIXTURE(ESFixture, "never & string")
|
|||
|
||||
TEST_CASE_FIXTURE(ESFixture, "string & (unknown | never)")
|
||||
{
|
||||
CHECK("string" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->stringType,
|
||||
arena->addType(UnionType{{builtinTypes->unknownType, builtinTypes->neverType}})
|
||||
}})));
|
||||
CHECK(
|
||||
"string" == simplifyStr(arena->addType(
|
||||
IntersectionType{{builtinTypes->stringType, arena->addType(UnionType{{builtinTypes->unknownType, builtinTypes->neverType}})}}
|
||||
))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "true | false")
|
||||
|
@ -211,112 +208,97 @@ TEST_CASE_FIXTURE(ESFixture, "error | unknown")
|
|||
|
||||
TEST_CASE_FIXTURE(ESFixture, "\"hello\" | string")
|
||||
{
|
||||
CHECK("string" == simplifyStr(arena->addType(UnionType{{
|
||||
arena->addType(SingletonType{StringSingleton{"hello"}}), builtinTypes->stringType
|
||||
}})));
|
||||
CHECK("string" == simplifyStr(arena->addType(UnionType{{arena->addType(SingletonType{StringSingleton{"hello"}}), builtinTypes->stringType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "\"hello\" | \"world\" | \"hello\"")
|
||||
{
|
||||
CHECK("\"hello\" | \"world\"" == simplifyStr(arena->addType(UnionType{{
|
||||
arena->addType(SingletonType{StringSingleton{"hello"}}),
|
||||
arena->addType(SingletonType{StringSingleton{"world"}}),
|
||||
arena->addType(SingletonType{StringSingleton{"hello"}}),
|
||||
}})));
|
||||
CHECK(
|
||||
"\"hello\" | \"world\"" == simplifyStr(arena->addType(UnionType{{
|
||||
arena->addType(SingletonType{StringSingleton{"hello"}}),
|
||||
arena->addType(SingletonType{StringSingleton{"world"}}),
|
||||
arena->addType(SingletonType{StringSingleton{"hello"}}),
|
||||
}}))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "nil | boolean | number | string | thread | function | table | class | buffer")
|
||||
{
|
||||
CHECK("unknown" == simplifyStr(arena->addType(UnionType{{
|
||||
builtinTypes->nilType,
|
||||
builtinTypes->booleanType,
|
||||
builtinTypes->numberType,
|
||||
builtinTypes->stringType,
|
||||
builtinTypes->threadType,
|
||||
builtinTypes->functionType,
|
||||
builtinTypes->tableType,
|
||||
builtinTypes->classType,
|
||||
builtinTypes->bufferType,
|
||||
}})));
|
||||
CHECK(
|
||||
"unknown" == simplifyStr(arena->addType(UnionType{{
|
||||
builtinTypes->nilType,
|
||||
builtinTypes->booleanType,
|
||||
builtinTypes->numberType,
|
||||
builtinTypes->stringType,
|
||||
builtinTypes->threadType,
|
||||
builtinTypes->functionType,
|
||||
builtinTypes->tableType,
|
||||
builtinTypes->classType,
|
||||
builtinTypes->bufferType,
|
||||
}}))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "Parent & number")
|
||||
{
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{
|
||||
parentClass, builtinTypes->numberType
|
||||
}})));
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{parentClass, builtinTypes->numberType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "Child & Parent")
|
||||
{
|
||||
CHECK("Child" == simplifyStr(arena->addType(IntersectionType{{
|
||||
childClass, parentClass
|
||||
}})));
|
||||
CHECK("Child" == simplifyStr(arena->addType(IntersectionType{{childClass, parentClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "Child & Unrelated")
|
||||
{
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{
|
||||
childClass, unrelatedClass
|
||||
}})));
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{childClass, unrelatedClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "Child | Parent")
|
||||
{
|
||||
CHECK("Parent" == simplifyStr(arena->addType(UnionType{{
|
||||
childClass, parentClass
|
||||
}})));
|
||||
CHECK("Parent" == simplifyStr(arena->addType(UnionType{{childClass, parentClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "class | Child")
|
||||
{
|
||||
CHECK("class" == simplifyStr(arena->addType(UnionType{{
|
||||
builtinTypes->classType, childClass
|
||||
}})));
|
||||
CHECK("class" == simplifyStr(arena->addType(UnionType{{builtinTypes->classType, childClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "Parent | class | Child")
|
||||
{
|
||||
CHECK("class" == simplifyStr(arena->addType(UnionType{{
|
||||
parentClass, builtinTypes->classType, childClass
|
||||
}})));
|
||||
CHECK("class" == simplifyStr(arena->addType(UnionType{{parentClass, builtinTypes->classType, childClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "Parent | Unrelated")
|
||||
{
|
||||
CHECK("Parent | Unrelated" == simplifyStr(arena->addType(UnionType{{
|
||||
parentClass, unrelatedClass
|
||||
}})));
|
||||
CHECK("Parent | Unrelated" == simplifyStr(arena->addType(UnionType{{parentClass, unrelatedClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "never | Parent | Unrelated")
|
||||
{
|
||||
CHECK("Parent | Unrelated" == simplifyStr(arena->addType(UnionType{{
|
||||
builtinTypes->neverType, parentClass, unrelatedClass
|
||||
}})));
|
||||
CHECK("Parent | Unrelated" == simplifyStr(arena->addType(UnionType{{builtinTypes->neverType, parentClass, unrelatedClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "never | Parent | (number & string) | Unrelated")
|
||||
{
|
||||
CHECK("Parent | Unrelated" == simplifyStr(arena->addType(UnionType{{
|
||||
builtinTypes->neverType, parentClass,
|
||||
arena->addType(IntersectionType{{builtinTypes->numberType, builtinTypes->stringType}}),
|
||||
unrelatedClass
|
||||
}})));
|
||||
CHECK(
|
||||
"Parent | Unrelated" == simplifyStr(arena->addType(UnionType{
|
||||
{builtinTypes->neverType,
|
||||
parentClass,
|
||||
arena->addType(IntersectionType{{builtinTypes->numberType, builtinTypes->stringType}}),
|
||||
unrelatedClass}
|
||||
}))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "T & U")
|
||||
{
|
||||
CHECK("T & U" == simplifyStr(arena->addType(IntersectionType{{
|
||||
genericT, genericU
|
||||
}})));
|
||||
CHECK("T & U" == simplifyStr(arena->addType(IntersectionType{{genericT, genericU}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "boolean & true")
|
||||
{
|
||||
CHECK("true" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->booleanType, builtinTypes->trueType
|
||||
}})));
|
||||
CHECK("true" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->booleanType, builtinTypes->trueType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "boolean & (true | number | string | thread | function | table | class | buffer)")
|
||||
|
@ -332,23 +314,17 @@ TEST_CASE_FIXTURE(ESFixture, "boolean & (true | number | string | thread | funct
|
|||
builtinTypes->bufferType,
|
||||
}});
|
||||
|
||||
CHECK("true" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->booleanType, truthy
|
||||
}})));
|
||||
CHECK("true" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->booleanType, truthy}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "boolean & ~(false?)")
|
||||
{
|
||||
CHECK("true" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->booleanType, builtinTypes->truthyType
|
||||
}})));
|
||||
CHECK("true" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->booleanType, builtinTypes->truthyType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "false & ~(false?)")
|
||||
{
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->falseType, builtinTypes->truthyType
|
||||
}})));
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->falseType, builtinTypes->truthyType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "(number) -> string & (number) -> string")
|
||||
|
@ -399,28 +375,25 @@ TEST_CASE_FIXTURE(ESFixture, "(number) -> string | (string) -> number")
|
|||
|
||||
TEST_CASE_FIXTURE(ESFixture, "add<number, number>")
|
||||
{
|
||||
CHECK("number" == simplifyStr(arena->addType(
|
||||
TypeFunctionInstanceType{builtinTypeFunctions().addFunc, {
|
||||
builtinTypes->numberType, builtinTypes->numberType
|
||||
}}
|
||||
)));
|
||||
CHECK(
|
||||
"number" ==
|
||||
simplifyStr(arena->addType(TypeFunctionInstanceType{builtinTypeFunctions().addFunc, {builtinTypes->numberType, builtinTypes->numberType}}))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "union<number, number>")
|
||||
{
|
||||
CHECK("number" == simplifyStr(arena->addType(
|
||||
TypeFunctionInstanceType{builtinTypeFunctions().unionFunc, {
|
||||
builtinTypes->numberType, builtinTypes->numberType
|
||||
}}
|
||||
)));
|
||||
CHECK(
|
||||
"number" ==
|
||||
simplifyStr(arena->addType(TypeFunctionInstanceType{builtinTypeFunctions().unionFunc, {builtinTypes->numberType, builtinTypes->numberType}}))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "never & ~string")
|
||||
{
|
||||
CHECK("never" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->neverType,
|
||||
arena->addType(NegationType{builtinTypes->stringType})
|
||||
}})));
|
||||
CHECK(
|
||||
"never" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->neverType, arena->addType(NegationType{builtinTypes->stringType})}}))
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "blocked & never")
|
||||
|
@ -444,7 +417,9 @@ TEST_CASE_FIXTURE(ESFixture, "blocked & ~number & function")
|
|||
|
||||
TEST_CASE_FIXTURE(ESFixture, "(number | boolean | string | nil | table) & (false | nil)")
|
||||
{
|
||||
const TypeId t1 = arena->addType(UnionType{{builtinTypes->numberType, builtinTypes->booleanType, builtinTypes->stringType, builtinTypes->nilType, builtinTypes->tableType}});
|
||||
const TypeId t1 = arena->addType(
|
||||
UnionType{{builtinTypes->numberType, builtinTypes->booleanType, builtinTypes->stringType, builtinTypes->nilType, builtinTypes->tableType}}
|
||||
);
|
||||
|
||||
CHECK("false?" == simplifyStr(arena->addType(IntersectionType{{t1, builtinTypes->falsyType}})));
|
||||
}
|
||||
|
@ -493,26 +468,17 @@ TEST_CASE_FIXTURE(ESFixture, "(blocked & number) | (blocked & number)")
|
|||
|
||||
TEST_CASE_FIXTURE(ESFixture, "{} & unknown")
|
||||
{
|
||||
CHECK("{ }" == simplifyStr(arena->addType(IntersectionType{{
|
||||
tbl({}),
|
||||
builtinTypes->unknownType
|
||||
}})));
|
||||
CHECK("{ }" == simplifyStr(arena->addType(IntersectionType{{tbl({}), builtinTypes->unknownType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "{} & table")
|
||||
{
|
||||
CHECK("{ }" == simplifyStr(arena->addType(IntersectionType{{
|
||||
tbl({}),
|
||||
builtinTypes->tableType
|
||||
}})));
|
||||
CHECK("{ }" == simplifyStr(arena->addType(IntersectionType{{tbl({}), builtinTypes->tableType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "{} & ~(false?)")
|
||||
{
|
||||
CHECK("{ }" == simplifyStr(arena->addType(IntersectionType{{
|
||||
tbl({}),
|
||||
builtinTypes->truthyType
|
||||
}})));
|
||||
CHECK("{ }" == simplifyStr(arena->addType(IntersectionType{{tbl({}), builtinTypes->truthyType}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "{x: number?} & {x: number}")
|
||||
|
@ -606,10 +572,7 @@ TEST_CASE_FIXTURE(ESFixture, "{ x: number } & ~boolean")
|
|||
{
|
||||
const TypeId tblTy = tbl(TableType::Props{{"x", builtinTypes->numberType}});
|
||||
|
||||
const TypeId ty = arena->addType(IntersectionType{{
|
||||
tblTy,
|
||||
arena->addType(NegationType{builtinTypes->booleanType})
|
||||
}});
|
||||
const TypeId ty = arena->addType(IntersectionType{{tblTy, arena->addType(NegationType{builtinTypes->booleanType})}});
|
||||
|
||||
CHECK("{ x: number }" == simplifyStr(ty));
|
||||
}
|
||||
|
@ -634,10 +597,7 @@ TEST_CASE_FIXTURE(ESFixture, "string & (\"hi\" | \"bye\")")
|
|||
const TypeId hi = arena->addType(SingletonType{StringSingleton{"hi"}});
|
||||
const TypeId bye = arena->addType(SingletonType{StringSingleton{"bye"}});
|
||||
|
||||
CHECK("\"bye\" | \"hi\"" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->stringType,
|
||||
arena->addType(UnionType{{hi, bye}})
|
||||
}})));
|
||||
CHECK("\"bye\" | \"hi\"" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->stringType, arena->addType(UnionType{{hi, bye}})}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "(\"err\" | \"ok\") & ~\"ok\"")
|
||||
|
@ -646,46 +606,32 @@ TEST_CASE_FIXTURE(ESFixture, "(\"err\" | \"ok\") & ~\"ok\"")
|
|||
TypeId ok1 = arena->addType(SingletonType{StringSingleton{"ok"}});
|
||||
TypeId ok2 = arena->addType(SingletonType{StringSingleton{"ok"}});
|
||||
|
||||
TypeId ty = arena->addType(IntersectionType{{
|
||||
arena->addType(UnionType{{err, ok1}}),
|
||||
arena->addType(NegationType{ok2})
|
||||
}});
|
||||
TypeId ty = arena->addType(IntersectionType{{arena->addType(UnionType{{err, ok1}}), arena->addType(NegationType{ok2})}});
|
||||
|
||||
CHECK("\"err\"" == simplifyStr(ty));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "(Child | Unrelated) & ~Child")
|
||||
{
|
||||
const TypeId ty = arena->addType(IntersectionType{{
|
||||
arena->addType(UnionType{{childClass, unrelatedClass}}),
|
||||
arena->addType(NegationType{childClass})
|
||||
}});
|
||||
const TypeId ty =
|
||||
arena->addType(IntersectionType{{arena->addType(UnionType{{childClass, unrelatedClass}}), arena->addType(NegationType{childClass})}});
|
||||
|
||||
CHECK("Unrelated" == simplifyStr(ty));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "string & ~Child")
|
||||
{
|
||||
CHECK("string" == simplifyStr(arena->addType(IntersectionType{{
|
||||
builtinTypes->stringType,
|
||||
arena->addType(NegationType{childClass})
|
||||
}})));
|
||||
CHECK("string" == simplifyStr(arena->addType(IntersectionType{{builtinTypes->stringType, arena->addType(NegationType{childClass})}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "(Child | Unrelated) & Child")
|
||||
{
|
||||
CHECK("Child" == simplifyStr(arena->addType(IntersectionType{{
|
||||
arena->addType(UnionType{{childClass, unrelatedClass}}),
|
||||
childClass
|
||||
}})));
|
||||
CHECK("Child" == simplifyStr(arena->addType(IntersectionType{{arena->addType(UnionType{{childClass, unrelatedClass}}), childClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "(Child | AnotherChild) & ~Child")
|
||||
{
|
||||
CHECK("Child" == simplifyStr(arena->addType(IntersectionType{{
|
||||
arena->addType(UnionType{{childClass, anotherChild}}),
|
||||
childClass
|
||||
}})));
|
||||
CHECK("Child" == simplifyStr(arena->addType(IntersectionType{{arena->addType(UnionType{{childClass, anotherChild}}), childClass}})));
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(ESFixture, "{ tag: \"Part\", x: never }")
|
||||
|
@ -706,11 +652,7 @@ TEST_CASE_FIXTURE(ESFixture, "{ tag: \"Part\", x: number? } & { x: string }")
|
|||
TEST_CASE_FIXTURE(ESFixture, "Child & add<Child | AnotherChild | string, Parent>")
|
||||
{
|
||||
const TypeId u = arena->addType(UnionType{{childClass, anotherChild, builtinTypes->stringType}});
|
||||
const TypeId intersectTf = arena->addType(TypeFunctionInstanceType{
|
||||
builtinTypeFunctions().addFunc,
|
||||
{u, parentClass},
|
||||
{}
|
||||
});
|
||||
const TypeId intersectTf = arena->addType(TypeFunctionInstanceType{builtinTypeFunctions().addFunc, {u, parentClass}, {}});
|
||||
|
||||
const TypeId intersection = arena->addType(IntersectionType{{childClass, intersectTf}});
|
||||
|
||||
|
@ -720,11 +662,7 @@ TEST_CASE_FIXTURE(ESFixture, "Child & add<Child | AnotherChild | string, Parent>
|
|||
TEST_CASE_FIXTURE(ESFixture, "Child & intersect<Child | AnotherChild | string, Parent>")
|
||||
{
|
||||
const TypeId u = arena->addType(UnionType{{childClass, anotherChild, builtinTypes->stringType}});
|
||||
const TypeId intersectTf = arena->addType(TypeFunctionInstanceType{
|
||||
builtinTypeFunctions().intersectFunc,
|
||||
{u, parentClass},
|
||||
{}
|
||||
});
|
||||
const TypeId intersectTf = arena->addType(TypeFunctionInstanceType{builtinTypeFunctions().intersectFunc, {u, parentClass}, {}});
|
||||
|
||||
const TypeId intersection = arena->addType(IntersectionType{{childClass, intersectTf}});
|
||||
|
||||
|
@ -740,7 +678,8 @@ TEST_CASE_FIXTURE(ESFixture, "lt<number, _> == boolean")
|
|||
{arena->addType(BlockedType{}), builtinTypes->stringType},
|
||||
};
|
||||
|
||||
for (const auto& [lhs, rhs] : cases) {
|
||||
for (const auto& [lhs, rhs] : cases)
|
||||
{
|
||||
const TypeId tfun = arena->addType(TypeFunctionInstanceType{builtinTypeFunctions().ltFunc, {lhs, rhs}});
|
||||
CHECK("boolean" == simplifyStr(tfun));
|
||||
}
|
||||
|
|
|
@ -29,8 +29,7 @@ LUAU_FASTFLAG(DebugLuauFreezeArena)
|
|||
LUAU_FASTFLAG(DebugLuauForceAllNewSolverTests)
|
||||
LUAU_FASTFLAG(LuauVectorDefinitionsExtra)
|
||||
|
||||
#define DOES_NOT_PASS_NEW_SOLVER_GUARD_IMPL(line) \
|
||||
ScopedFastFlag sff_##line{FFlag::LuauSolverV2, FFlag::DebugLuauForceAllNewSolverTests};
|
||||
#define DOES_NOT_PASS_NEW_SOLVER_GUARD_IMPL(line) ScopedFastFlag sff_##line{FFlag::LuauSolverV2, FFlag::DebugLuauForceAllNewSolverTests};
|
||||
|
||||
#define DOES_NOT_PASS_NEW_SOLVER_GUARD() DOES_NOT_PASS_NEW_SOLVER_GUARD_IMPL(__LINE__)
|
||||
|
||||
|
|
|
@ -25,6 +25,7 @@ LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
|||
LUAU_FASTFLAG(LuauSymbolEquality);
|
||||
LUAU_FASTFLAG(LuauStoreSolverTypeOnModule);
|
||||
LUAU_FASTFLAG(LexerResumesFromPosition2)
|
||||
LUAU_FASTFLAG(LuauIncrementalAutocompleteCommentDetection)
|
||||
|
||||
static std::optional<AutocompleteEntryMap> nullCallback(std::string tag, std::optional<const ClassType*> ptr, std::optional<std::string> contents)
|
||||
{
|
||||
|
@ -91,7 +92,8 @@ struct FragmentAutocompleteFixtureImpl : BaseType
|
|||
std::optional<Position> fragmentEndPosition = std::nullopt
|
||||
)
|
||||
{
|
||||
return Luau::typecheckFragment(this->frontend, "MainModule", cursorPos, getOptions(), document, fragmentEndPosition);
|
||||
auto [_, result] = Luau::typecheckFragment(this->frontend, "MainModule", cursorPos, getOptions(), document, fragmentEndPosition);
|
||||
return result;
|
||||
}
|
||||
|
||||
FragmentAutocompleteResult autocompleteFragment(
|
||||
|
@ -1383,4 +1385,177 @@ t
|
|||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments_simple")
|
||||
{
|
||||
const std::string source = R"(
|
||||
-- sel
|
||||
-- retur
|
||||
-- fo
|
||||
-- if
|
||||
-- end
|
||||
-- the
|
||||
)";
|
||||
ScopedFastFlag sff{FFlag::LuauIncrementalAutocompleteCommentDetection, true};
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{4, 6},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments_blocks")
|
||||
{
|
||||
const std::string source = R"(
|
||||
--[[
|
||||
comment 1
|
||||
]] local
|
||||
-- [[ comment 2]]
|
||||
--
|
||||
-- sdfsdfsdf
|
||||
--[[comment 3]]
|
||||
--[[
|
||||
foo
|
||||
bar
|
||||
baz
|
||||
]]
|
||||
)";
|
||||
ScopedFastFlag sff{FFlag::LuauIncrementalAutocompleteCommentDetection, true};
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{3, 0},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{3, 2},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(!result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{8, 6},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{10, 0},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments")
|
||||
{
|
||||
const std::string source = R"(
|
||||
-- sel
|
||||
-- retur
|
||||
-- fo
|
||||
--[[ sel ]]
|
||||
local -- hello
|
||||
)";
|
||||
ScopedFastFlag sff{FFlag::LuauIncrementalAutocompleteCommentDetection, true};
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{1, 7},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{2, 9},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{3, 6},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{4, 9},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{5, 6},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(!result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
source,
|
||||
Position{5, 14},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
TEST_CASE_FIXTURE(FragmentAutocompleteFixture, "no_recs_for_comments_in_incremental_fragment")
|
||||
{
|
||||
const std::string source = R"(
|
||||
local x = 5
|
||||
if x == 5
|
||||
)";
|
||||
const std::string updated = R"(
|
||||
local x = 5
|
||||
if x == 5 then -- a comment
|
||||
)";
|
||||
ScopedFastFlag sff{FFlag::LuauIncrementalAutocompleteCommentDetection, true};
|
||||
autocompleteFragmentInBothSolvers(
|
||||
source,
|
||||
updated,
|
||||
Position{2, 28},
|
||||
[](FragmentAutocompleteResult& result)
|
||||
{
|
||||
CHECK(result.acResults.entryMap.empty());
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
TEST_SUITE_END();
|
||||
|
|
|
@ -520,9 +520,9 @@ bb_bytecode_0:
|
|||
JUMP bb_2
|
||||
bb_2:
|
||||
CHECK_SAFE_ENV exit(3)
|
||||
JUMP_EQ_TAG K1, tnil, bb_fallback_4, bb_3
|
||||
JUMP_EQ_TAG K1 (nil), tnil, bb_fallback_4, bb_3
|
||||
bb_3:
|
||||
%9 = LOAD_TVALUE K1
|
||||
%9 = LOAD_TVALUE K1 (nil)
|
||||
STORE_TVALUE R1, %9
|
||||
JUMP bb_5
|
||||
bb_5:
|
||||
|
@ -575,7 +575,7 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
%4 = LOAD_TVALUE K0, 0i, tvector
|
||||
%4 = LOAD_TVALUE K0 (1, 2, 3), 0i, tvector
|
||||
%11 = LOAD_TVALUE R0
|
||||
%12 = ADD_VEC %4, %11
|
||||
%13 = TAG_VECTOR %12
|
||||
|
@ -602,7 +602,7 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
FALLBACK_NAMECALL 0u, R1, R0, K0
|
||||
FALLBACK_NAMECALL 0u, R1, R0, K0 ('Abs')
|
||||
INTERRUPT 2u
|
||||
SET_SAVEDPC 3u
|
||||
CALL R1, 1i, -1i
|
||||
|
@ -628,8 +628,8 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
FALLBACK_GETTABLEKS 0u, R3, R0, K0
|
||||
FALLBACK_GETTABLEKS 2u, R4, R0, K1
|
||||
FALLBACK_GETTABLEKS 0u, R3, R0, K0 ('XX')
|
||||
FALLBACK_GETTABLEKS 2u, R4, R0, K1 ('YY')
|
||||
CHECK_TAG R3, tnumber, bb_fallback_3
|
||||
CHECK_TAG R4, tnumber, bb_fallback_3
|
||||
%14 = LOAD_DOUBLE R3
|
||||
|
@ -639,7 +639,7 @@ bb_bytecode_1:
|
|||
JUMP bb_4
|
||||
bb_4:
|
||||
CHECK_TAG R0, tvector, exit(5)
|
||||
FALLBACK_GETTABLEKS 5u, R3, R0, K2
|
||||
FALLBACK_GETTABLEKS 5u, R3, R0, K2 ('ZZ')
|
||||
CHECK_TAG R2, tnumber, bb_fallback_5
|
||||
CHECK_TAG R3, tnumber, bb_fallback_5
|
||||
%30 = LOAD_DOUBLE R2
|
||||
|
@ -857,8 +857,8 @@ bb_2:
|
|||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
%8 = LOAD_POINTER R0
|
||||
%9 = GET_SLOT_NODE_ADDR %8, 0u, K1
|
||||
CHECK_SLOT_MATCH %9, K1, bb_fallback_3
|
||||
%9 = GET_SLOT_NODE_ADDR %8, 0u, K1 ('n')
|
||||
CHECK_SLOT_MATCH %9, K1 ('n'), bb_fallback_3
|
||||
%11 = LOAD_TVALUE %9, 0i
|
||||
STORE_TVALUE R3, %11
|
||||
JUMP bb_4
|
||||
|
@ -885,8 +885,8 @@ bb_4:
|
|||
STORE_VECTOR R3, %30, %33, %36
|
||||
CHECK_TAG R0, ttable, exit(6)
|
||||
%41 = LOAD_POINTER R0
|
||||
%42 = GET_SLOT_NODE_ADDR %41, 6u, K3
|
||||
CHECK_SLOT_MATCH %42, K3, bb_fallback_5
|
||||
%42 = GET_SLOT_NODE_ADDR %41, 6u, K3 ('b')
|
||||
CHECK_SLOT_MATCH %42, K3 ('b'), bb_fallback_5
|
||||
%44 = LOAD_TVALUE %42, 0i
|
||||
STORE_TVALUE R5, %44
|
||||
JUMP bb_6
|
||||
|
@ -929,8 +929,8 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
FALLBACK_GETTABLEKS 0u, R2, R0, K0
|
||||
FALLBACK_GETTABLEKS 2u, R3, R0, K1
|
||||
FALLBACK_GETTABLEKS 0u, R2, R0, K0 ('x')
|
||||
FALLBACK_GETTABLEKS 2u, R3, R0, K1 ('y')
|
||||
CHECK_TAG R2, tnumber, bb_fallback_3
|
||||
CHECK_TAG R3, tnumber, bb_fallback_3
|
||||
%14 = LOAD_DOUBLE R2
|
||||
|
@ -964,9 +964,9 @@ bb_2:
|
|||
bb_bytecode_1:
|
||||
STORE_DOUBLE R1, 3
|
||||
STORE_TAG R1, tnumber
|
||||
FALLBACK_SETTABLEKS 1u, R1, R0, K0
|
||||
FALLBACK_SETTABLEKS 1u, R1, R0, K0 ('x')
|
||||
STORE_DOUBLE R1, 4
|
||||
FALLBACK_SETTABLEKS 4u, R1, R0, K1
|
||||
FALLBACK_SETTABLEKS 4u, R1, R0, K1 ('y')
|
||||
INTERRUPT 6u
|
||||
RETURN R0, 0i
|
||||
)"
|
||||
|
@ -989,11 +989,11 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
FALLBACK_NAMECALL 0u, R2, R0, K0
|
||||
FALLBACK_NAMECALL 0u, R2, R0, K0 ('GetX')
|
||||
INTERRUPT 2u
|
||||
SET_SAVEDPC 3u
|
||||
CALL R2, 1i, 1i
|
||||
FALLBACK_NAMECALL 3u, R3, R0, K1
|
||||
FALLBACK_NAMECALL 3u, R3, R0, K1 ('GetY')
|
||||
INTERRUPT 5u
|
||||
SET_SAVEDPC 6u
|
||||
CALL R3, 1i, 1i
|
||||
|
@ -1367,8 +1367,8 @@ bb_bytecode_1:
|
|||
bb_4:
|
||||
CHECK_TAG R2, ttable, exit(1)
|
||||
%23 = LOAD_POINTER R2
|
||||
%24 = GET_SLOT_NODE_ADDR %23, 1u, K0
|
||||
CHECK_SLOT_MATCH %24, K0, bb_fallback_5
|
||||
%24 = GET_SLOT_NODE_ADDR %23, 1u, K0 ('pos')
|
||||
CHECK_SLOT_MATCH %24, K0 ('pos'), bb_fallback_5
|
||||
%26 = LOAD_TVALUE %24, 0i
|
||||
STORE_TVALUE R4, %26
|
||||
JUMP bb_6
|
||||
|
@ -1476,13 +1476,13 @@ bb_bytecode_1:
|
|||
bb_4:
|
||||
CHECK_TAG R3, ttable, bb_fallback_5
|
||||
%23 = LOAD_POINTER R3
|
||||
%24 = GET_SLOT_NODE_ADDR %23, 1u, K0
|
||||
CHECK_SLOT_MATCH %24, K0, bb_fallback_5
|
||||
%24 = GET_SLOT_NODE_ADDR %23, 1u, K0 ('normal')
|
||||
CHECK_SLOT_MATCH %24, K0 ('normal'), bb_fallback_5
|
||||
%26 = LOAD_TVALUE %24, 0i
|
||||
STORE_TVALUE R2, %26
|
||||
JUMP bb_6
|
||||
bb_6:
|
||||
%31 = LOAD_TVALUE K1, 0i, tvector
|
||||
%31 = LOAD_TVALUE K1 (0.707000017, 0, 0.707000017), 0i, tvector
|
||||
STORE_TVALUE R4, %31
|
||||
CHECK_TAG R2, tvector, exit(4)
|
||||
%37 = LOAD_FLOAT R2, 0i
|
||||
|
@ -1603,9 +1603,9 @@ bb_bytecode_1:
|
|||
STORE_DOUBLE R1, 0
|
||||
STORE_TAG R1, tnumber
|
||||
CHECK_SAFE_ENV exit(1)
|
||||
JUMP_EQ_TAG K1, tnil, bb_fallback_6, bb_5
|
||||
JUMP_EQ_TAG K1 (nil), tnil, bb_fallback_6, bb_5
|
||||
bb_5:
|
||||
%9 = LOAD_TVALUE K1
|
||||
%9 = LOAD_TVALUE K1 (nil)
|
||||
STORE_TVALUE R2, %9
|
||||
JUMP bb_7
|
||||
bb_7:
|
||||
|
@ -1627,8 +1627,8 @@ bb_9:
|
|||
bb_bytecode_2:
|
||||
CHECK_TAG R6, ttable, exit(6)
|
||||
%35 = LOAD_POINTER R6
|
||||
%36 = GET_SLOT_NODE_ADDR %35, 6u, K2
|
||||
CHECK_SLOT_MATCH %36, K2, bb_fallback_10
|
||||
%36 = GET_SLOT_NODE_ADDR %35, 6u, K2 ('pos')
|
||||
CHECK_SLOT_MATCH %36, K2 ('pos'), bb_fallback_10
|
||||
%38 = LOAD_TVALUE %36, 0i
|
||||
STORE_TVALUE R8, %38
|
||||
JUMP bb_11
|
||||
|
@ -1829,8 +1829,8 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
FALLBACK_GETTABLEKS 0u, R2, R0, K0
|
||||
FALLBACK_GETTABLEKS 2u, R3, R0, K1
|
||||
FALLBACK_GETTABLEKS 0u, R2, R0, K0 ('Row1')
|
||||
FALLBACK_GETTABLEKS 2u, R3, R0, K1 ('Row2')
|
||||
CHECK_TAG R2, tvector, exit(4)
|
||||
CHECK_TAG R3, tvector, exit(4)
|
||||
%14 = LOAD_TVALUE R2
|
||||
|
@ -2138,10 +2138,10 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
%4 = LOAD_TVALUE K0, 0i, tvector
|
||||
%4 = LOAD_TVALUE K0 (1, 0, 0), 0i, tvector
|
||||
%11 = LOAD_TVALUE R0
|
||||
%12 = MUL_VEC %4, %11
|
||||
%15 = LOAD_TVALUE K1, 0i, tvector
|
||||
%15 = LOAD_TVALUE K1 (0, 1, 0), 0i, tvector
|
||||
%23 = ADD_VEC %12, %15
|
||||
%24 = TAG_VECTOR %23
|
||||
STORE_TVALUE R1, %24
|
||||
|
@ -2176,7 +2176,7 @@ bb_0:
|
|||
bb_2:
|
||||
JUMP bb_bytecode_1
|
||||
bb_bytecode_1:
|
||||
%4 = LOAD_TVALUE K0, 0i, tvector
|
||||
%4 = LOAD_TVALUE K0 (0, 0, 0), 0i, tvector
|
||||
%11 = LOAD_TVALUE R0
|
||||
%12 = ADD_VEC %4, %11
|
||||
%13 = TAG_VECTOR %12
|
||||
|
@ -2208,9 +2208,9 @@ bb_bytecode_0:
|
|||
STORE_TAG R1, tboolean
|
||||
STORE_DOUBLE R2, 4.75
|
||||
STORE_TAG R2, tnumber
|
||||
%5 = LOAD_TVALUE K1, 0i, tvector
|
||||
%5 = LOAD_TVALUE K1 (1, 2, 4), 0i, tvector
|
||||
STORE_TVALUE R3, %5
|
||||
%7 = LOAD_TVALUE K2, 0i, tstring
|
||||
%7 = LOAD_TVALUE K2 ('test'), 0i, tstring
|
||||
STORE_TVALUE R4, %7
|
||||
INTERRUPT 5u
|
||||
RETURN R0, 5i
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue