mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 18:30:54 +01:00
Merge remote-tracking branch 'remotes/upstream/master' into udtf-typeof-name
This commit is contained in:
commit
178121e1dc
225 changed files with 11810 additions and 3527 deletions
|
@ -65,10 +65,7 @@ TypeId makeFunction( // Polymorphic
|
||||||
bool checked = false
|
bool checked = false
|
||||||
);
|
);
|
||||||
|
|
||||||
void attachMagicFunction(TypeId ty, MagicFunction fn);
|
void attachMagicFunction(TypeId ty, std::shared_ptr<MagicFunction> fn);
|
||||||
void attachDcrMagicFunction(TypeId ty, DcrMagicFunction fn);
|
|
||||||
void attachDcrMagicRefinement(TypeId ty, DcrMagicRefinement fn);
|
|
||||||
void attachDcrMagicFunctionTypeCheck(TypeId ty, DcrMagicFunctionTypeCheck fn);
|
|
||||||
Property makeProperty(TypeId ty, std::optional<std::string> documentationSymbol = std::nullopt);
|
Property makeProperty(TypeId ty, std::optional<std::string> documentationSymbol = std::nullopt);
|
||||||
void assignPropDocumentationSymbols(TableType::Props& props, const std::string& baseName);
|
void assignPropDocumentationSymbols(TableType::Props& props, const std::string& baseName);
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
#include <Luau/NotNull.h>
|
#include <Luau/NotNull.h>
|
||||||
#include "Luau/TypeArena.h"
|
#include "Luau/TypeArena.h"
|
||||||
#include "Luau/Type.h"
|
#include "Luau/Type.h"
|
||||||
|
#include "Luau/Scope.h"
|
||||||
|
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
|
|
||||||
|
@ -26,13 +27,17 @@ struct CloneState
|
||||||
* while `clone` will make a deep copy of the entire type and its every component.
|
* while `clone` will make a deep copy of the entire type and its every component.
|
||||||
*
|
*
|
||||||
* Be mindful about which behavior you actually _want_.
|
* Be mindful about which behavior you actually _want_.
|
||||||
|
*
|
||||||
|
* Persistent types are not cloned as an optimization.
|
||||||
|
* If a type is cloned in order to mutate it, 'ignorePersistent' has to be set
|
||||||
*/
|
*/
|
||||||
|
|
||||||
TypePackId shallowClone(TypePackId tp, TypeArena& dest, CloneState& cloneState);
|
TypePackId shallowClone(TypePackId tp, TypeArena& dest, CloneState& cloneState, bool ignorePersistent = false);
|
||||||
TypeId shallowClone(TypeId typeId, TypeArena& dest, CloneState& cloneState);
|
TypeId shallowClone(TypeId typeId, TypeArena& dest, CloneState& cloneState, bool ignorePersistent = false);
|
||||||
|
|
||||||
TypePackId clone(TypePackId tp, TypeArena& dest, CloneState& cloneState);
|
TypePackId clone(TypePackId tp, TypeArena& dest, CloneState& cloneState);
|
||||||
TypeId clone(TypeId tp, TypeArena& dest, CloneState& cloneState);
|
TypeId clone(TypeId tp, TypeArena& dest, CloneState& cloneState);
|
||||||
TypeFun clone(const TypeFun& typeFun, TypeArena& dest, CloneState& cloneState);
|
TypeFun clone(const TypeFun& typeFun, TypeArena& dest, CloneState& cloneState);
|
||||||
|
Binding clone(const Binding& binding, TypeArena& dest, CloneState& cloneState);
|
||||||
|
|
||||||
} // namespace Luau
|
} // namespace Luau
|
||||||
|
|
|
@ -392,7 +392,7 @@ private:
|
||||||
**/
|
**/
|
||||||
std::vector<std::pair<Name, GenericTypeDefinition>> createGenerics(
|
std::vector<std::pair<Name, GenericTypeDefinition>> createGenerics(
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
AstArray<AstGenericType> generics,
|
AstArray<AstGenericType*> generics,
|
||||||
bool useCache = false,
|
bool useCache = false,
|
||||||
bool addTypes = true
|
bool addTypes = true
|
||||||
);
|
);
|
||||||
|
@ -409,7 +409,7 @@ private:
|
||||||
**/
|
**/
|
||||||
std::vector<std::pair<Name, GenericTypePackDefinition>> createGenericPacks(
|
std::vector<std::pair<Name, GenericTypePackDefinition>> createGenericPacks(
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
AstArray<AstGenericTypePack> packs,
|
AstArray<AstGenericTypePack*> packs,
|
||||||
bool useCache = false,
|
bool useCache = false,
|
||||||
bool addTypes = true
|
bool addTypes = true
|
||||||
);
|
);
|
||||||
|
|
|
@ -166,7 +166,7 @@ struct ConstraintSolver
|
||||||
**/
|
**/
|
||||||
void finalizeTypeFunctions();
|
void finalizeTypeFunctions();
|
||||||
|
|
||||||
bool isDone();
|
bool isDone() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
/**
|
/**
|
||||||
|
@ -298,10 +298,10 @@ public:
|
||||||
// FIXME: This use of a boolean for the return result is an appalling
|
// FIXME: This use of a boolean for the return result is an appalling
|
||||||
// interface.
|
// interface.
|
||||||
bool blockOnPendingTypes(TypeId target, NotNull<const Constraint> constraint);
|
bool blockOnPendingTypes(TypeId target, NotNull<const Constraint> constraint);
|
||||||
bool blockOnPendingTypes(TypePackId target, NotNull<const Constraint> constraint);
|
bool blockOnPendingTypes(TypePackId targetPack, NotNull<const Constraint> constraint);
|
||||||
|
|
||||||
void unblock(NotNull<const Constraint> progressed);
|
void unblock(NotNull<const Constraint> progressed);
|
||||||
void unblock(TypeId progressed, Location location);
|
void unblock(TypeId ty, Location location);
|
||||||
void unblock(TypePackId progressed, Location location);
|
void unblock(TypePackId progressed, Location location);
|
||||||
void unblock(const std::vector<TypeId>& types, Location location);
|
void unblock(const std::vector<TypeId>& types, Location location);
|
||||||
void unblock(const std::vector<TypePackId>& packs, Location location);
|
void unblock(const std::vector<TypePackId>& packs, Location location);
|
||||||
|
@ -336,7 +336,7 @@ public:
|
||||||
* @param location the location where the require is taking place; used for
|
* @param location the location where the require is taking place; used for
|
||||||
* error locations.
|
* error locations.
|
||||||
**/
|
**/
|
||||||
TypeId resolveModule(const ModuleInfo& module, const Location& location);
|
TypeId resolveModule(const ModuleInfo& info, const Location& location);
|
||||||
|
|
||||||
void reportError(TypeErrorData&& data, const Location& location);
|
void reportError(TypeErrorData&& data, const Location& location);
|
||||||
void reportError(TypeError e);
|
void reportError(TypeError e);
|
||||||
|
@ -420,6 +420,11 @@ public:
|
||||||
void throwUserCancelError() const;
|
void throwUserCancelError() const;
|
||||||
|
|
||||||
ToStringOptions opts;
|
ToStringOptions opts;
|
||||||
|
|
||||||
|
void fillInDiscriminantTypes(
|
||||||
|
NotNull<const Constraint> constraint,
|
||||||
|
const std::vector<std::optional<TypeId>>& discriminantTypes
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
void dump(NotNull<Scope> rootScope, struct ToStringOptions& opts);
|
void dump(NotNull<Scope> rootScope, struct ToStringOptions& opts);
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
#include "Luau/ControlFlow.h"
|
#include "Luau/ControlFlow.h"
|
||||||
#include "Luau/DenseHash.h"
|
#include "Luau/DenseHash.h"
|
||||||
#include "Luau/Def.h"
|
#include "Luau/Def.h"
|
||||||
|
#include "Luau/NotNull.h"
|
||||||
#include "Luau/Symbol.h"
|
#include "Luau/Symbol.h"
|
||||||
#include "Luau/TypedAllocator.h"
|
#include "Luau/TypedAllocator.h"
|
||||||
|
|
||||||
|
@ -48,13 +49,13 @@ struct DataFlowGraph
|
||||||
const RefinementKey* getRefinementKey(const AstExpr* expr) const;
|
const RefinementKey* getRefinementKey(const AstExpr* expr) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DataFlowGraph() = default;
|
DataFlowGraph(NotNull<DefArena> defArena, NotNull<RefinementKeyArena> keyArena);
|
||||||
|
|
||||||
DataFlowGraph(const DataFlowGraph&) = delete;
|
DataFlowGraph(const DataFlowGraph&) = delete;
|
||||||
DataFlowGraph& operator=(const DataFlowGraph&) = delete;
|
DataFlowGraph& operator=(const DataFlowGraph&) = delete;
|
||||||
|
|
||||||
DefArena defArena;
|
NotNull<DefArena> defArena;
|
||||||
RefinementKeyArena keyArena;
|
NotNull<RefinementKeyArena> keyArena;
|
||||||
|
|
||||||
DenseHashMap<const AstExpr*, const Def*> astDefs{nullptr};
|
DenseHashMap<const AstExpr*, const Def*> astDefs{nullptr};
|
||||||
|
|
||||||
|
@ -110,30 +111,22 @@ using ScopeStack = std::vector<DfgScope*>;
|
||||||
|
|
||||||
struct DataFlowGraphBuilder
|
struct DataFlowGraphBuilder
|
||||||
{
|
{
|
||||||
static DataFlowGraph build(AstStatBlock* root, NotNull<struct InternalErrorReporter> handle);
|
static DataFlowGraph build(
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is identical to the build method above, but returns a pair of dfg, scopes as the data flow graph
|
|
||||||
* here is intended to live on the module between runs of typechecking. Before, the DFG only needed to live as
|
|
||||||
* long as the typecheck, but in a world with incremental typechecking, we need the information on the dfg to incrementally
|
|
||||||
* typecheck small fragments of code.
|
|
||||||
* @param block - pointer to the ast to build the dfg for
|
|
||||||
* @param handle - for raising internal errors while building the dfg
|
|
||||||
*/
|
|
||||||
static std::pair<std::shared_ptr<DataFlowGraph>, std::vector<std::unique_ptr<DfgScope>>> buildShared(
|
|
||||||
AstStatBlock* block,
|
AstStatBlock* block,
|
||||||
NotNull<InternalErrorReporter> handle
|
NotNull<DefArena> defArena,
|
||||||
|
NotNull<RefinementKeyArena> keyArena,
|
||||||
|
NotNull<struct InternalErrorReporter> handle
|
||||||
);
|
);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
DataFlowGraphBuilder() = default;
|
DataFlowGraphBuilder(NotNull<DefArena> defArena, NotNull<RefinementKeyArena> keyArena);
|
||||||
|
|
||||||
DataFlowGraphBuilder(const DataFlowGraphBuilder&) = delete;
|
DataFlowGraphBuilder(const DataFlowGraphBuilder&) = delete;
|
||||||
DataFlowGraphBuilder& operator=(const DataFlowGraphBuilder&) = delete;
|
DataFlowGraphBuilder& operator=(const DataFlowGraphBuilder&) = delete;
|
||||||
|
|
||||||
DataFlowGraph graph;
|
DataFlowGraph graph;
|
||||||
NotNull<DefArena> defArena{&graph.defArena};
|
NotNull<DefArena> defArena;
|
||||||
NotNull<RefinementKeyArena> keyArena{&graph.keyArena};
|
NotNull<RefinementKeyArena> keyArena;
|
||||||
|
|
||||||
struct InternalErrorReporter* handle = nullptr;
|
struct InternalErrorReporter* handle = nullptr;
|
||||||
|
|
||||||
|
@ -228,8 +221,8 @@ private:
|
||||||
|
|
||||||
void visitTypeList(AstTypeList l);
|
void visitTypeList(AstTypeList l);
|
||||||
|
|
||||||
void visitGenerics(AstArray<AstGenericType> g);
|
void visitGenerics(AstArray<AstGenericType*> g);
|
||||||
void visitGenericPacks(AstArray<AstGenericTypePack> g);
|
void visitGenericPacks(AstArray<AstGenericTypePack*> g);
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Luau
|
} // namespace Luau
|
||||||
|
|
|
@ -105,6 +105,9 @@ private:
|
||||||
std::vector<Id> storage;
|
std::vector<Id> storage;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template <typename L>
|
||||||
|
using Node = EqSat::Node<L>;
|
||||||
|
|
||||||
using EType = EqSat::Language<
|
using EType = EqSat::Language<
|
||||||
TNil,
|
TNil,
|
||||||
TBoolean,
|
TBoolean,
|
||||||
|
@ -146,7 +149,7 @@ using EType = EqSat::Language<
|
||||||
struct StringCache
|
struct StringCache
|
||||||
{
|
{
|
||||||
Allocator allocator;
|
Allocator allocator;
|
||||||
DenseHashMap<size_t, StringId> strings{{}};
|
DenseHashMap<std::string_view, StringId> strings{{}};
|
||||||
std::vector<std::string_view> views;
|
std::vector<std::string_view> views;
|
||||||
|
|
||||||
StringId add(std::string_view s);
|
StringId add(std::string_view s);
|
||||||
|
@ -171,6 +174,9 @@ struct Subst
|
||||||
Id eclass;
|
Id eclass;
|
||||||
Id newClass;
|
Id newClass;
|
||||||
|
|
||||||
|
// The node into eclass which is boring, if any
|
||||||
|
std::optional<size_t> boringIndex;
|
||||||
|
|
||||||
std::string desc;
|
std::string desc;
|
||||||
|
|
||||||
Subst(Id eclass, Id newClass, std::string desc = "");
|
Subst(Id eclass, Id newClass, std::string desc = "");
|
||||||
|
@ -211,6 +217,7 @@ struct Simplifier
|
||||||
void subst(Id from, Id to);
|
void subst(Id from, Id to);
|
||||||
void subst(Id from, Id to, const std::string& ruleName);
|
void subst(Id from, Id to, const std::string& ruleName);
|
||||||
void subst(Id from, Id to, const std::string& ruleName, const std::unordered_map<Id, size_t>& forceNodes);
|
void subst(Id from, Id to, const std::string& ruleName, const std::unordered_map<Id, size_t>& forceNodes);
|
||||||
|
void subst(Id from, size_t boringIndex, Id to, const std::string& ruleName, const std::unordered_map<Id, size_t>& forceNodes);
|
||||||
|
|
||||||
void unionClasses(std::vector<Id>& hereParts, Id there);
|
void unionClasses(std::vector<Id>& hereParts, Id there);
|
||||||
|
|
||||||
|
@ -295,13 +302,13 @@ QueryIterator<Tag>::QueryIterator(EGraph* egraph_, Id eclass)
|
||||||
|
|
||||||
for (const auto& enode : ecl.nodes)
|
for (const auto& enode : ecl.nodes)
|
||||||
{
|
{
|
||||||
if (enode.index() < idx)
|
if (enode.node.index() < idx)
|
||||||
++index;
|
++index;
|
||||||
else
|
else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (index >= ecl.nodes.size() || ecl.nodes[index].index() != idx)
|
if (index >= ecl.nodes.size() || ecl.nodes[index].node.index() != idx)
|
||||||
{
|
{
|
||||||
egraph = nullptr;
|
egraph = nullptr;
|
||||||
index = 0;
|
index = 0;
|
||||||
|
@ -331,7 +338,7 @@ std::pair<const Tag*, size_t> QueryIterator<Tag>::operator*() const
|
||||||
EGraph::EClassT& ecl = (*egraph)[eclass];
|
EGraph::EClassT& ecl = (*egraph)[eclass];
|
||||||
|
|
||||||
LUAU_ASSERT(index < ecl.nodes.size());
|
LUAU_ASSERT(index < ecl.nodes.size());
|
||||||
auto& enode = ecl.nodes[index];
|
auto& enode = ecl.nodes[index].node;
|
||||||
Tag* result = enode.template get<Tag>();
|
Tag* result = enode.template get<Tag>();
|
||||||
LUAU_ASSERT(result);
|
LUAU_ASSERT(result);
|
||||||
return {result, index};
|
return {result, index};
|
||||||
|
@ -343,12 +350,16 @@ QueryIterator<Tag>& QueryIterator<Tag>::operator++()
|
||||||
{
|
{
|
||||||
const auto& ecl = (*egraph)[eclass];
|
const auto& ecl = (*egraph)[eclass];
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
++index;
|
++index;
|
||||||
if (index >= ecl.nodes.size() || ecl.nodes[index].index() != EType::VariantTy::getTypeId<Tag>())
|
if (index >= ecl.nodes.size() || ecl.nodes[index].node.index() != EType::VariantTy::getTypeId<Tag>())
|
||||||
{
|
{
|
||||||
egraph = nullptr;
|
egraph = nullptr;
|
||||||
index = 0;
|
index = 0;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
} while (ecl.nodes[index].boring);
|
||||||
|
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,12 @@ namespace Luau
|
||||||
{
|
{
|
||||||
struct FrontendOptions;
|
struct FrontendOptions;
|
||||||
|
|
||||||
|
enum class FragmentTypeCheckStatus
|
||||||
|
{
|
||||||
|
SkipAutocomplete,
|
||||||
|
Success,
|
||||||
|
};
|
||||||
|
|
||||||
struct FragmentAutocompleteAncestryResult
|
struct FragmentAutocompleteAncestryResult
|
||||||
{
|
{
|
||||||
DenseHashMap<AstName, AstLocal*> localMap{AstName()};
|
DenseHashMap<AstName, AstLocal*> localMap{AstName()};
|
||||||
|
@ -29,6 +35,7 @@ struct FragmentParseResult
|
||||||
AstStatBlock* root = nullptr;
|
AstStatBlock* root = nullptr;
|
||||||
std::vector<AstNode*> ancestry;
|
std::vector<AstNode*> ancestry;
|
||||||
AstStat* nearestStatement = nullptr;
|
AstStat* nearestStatement = nullptr;
|
||||||
|
std::vector<Comment> commentLocations;
|
||||||
std::unique_ptr<Allocator> alloc = std::make_unique<Allocator>();
|
std::unique_ptr<Allocator> alloc = std::make_unique<Allocator>();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -49,14 +56,14 @@ struct FragmentAutocompleteResult
|
||||||
|
|
||||||
FragmentAutocompleteAncestryResult findAncestryForFragmentParse(AstStatBlock* root, const Position& cursorPos);
|
FragmentAutocompleteAncestryResult findAncestryForFragmentParse(AstStatBlock* root, const Position& cursorPos);
|
||||||
|
|
||||||
FragmentParseResult parseFragment(
|
std::optional<FragmentParseResult> parseFragment(
|
||||||
const SourceModule& srcModule,
|
const SourceModule& srcModule,
|
||||||
std::string_view src,
|
std::string_view src,
|
||||||
const Position& cursorPos,
|
const Position& cursorPos,
|
||||||
std::optional<Position> fragmentEndPosition
|
std::optional<Position> fragmentEndPosition
|
||||||
);
|
);
|
||||||
|
|
||||||
FragmentTypeCheckResult typecheckFragment(
|
std::pair<FragmentTypeCheckStatus, FragmentTypeCheckResult> typecheckFragment(
|
||||||
Frontend& frontend,
|
Frontend& frontend,
|
||||||
const ModuleName& moduleName,
|
const ModuleName& moduleName,
|
||||||
const Position& cursorPos,
|
const Position& cursorPos,
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
#include "Luau/ModuleResolver.h"
|
#include "Luau/ModuleResolver.h"
|
||||||
#include "Luau/RequireTracer.h"
|
#include "Luau/RequireTracer.h"
|
||||||
#include "Luau/Scope.h"
|
#include "Luau/Scope.h"
|
||||||
|
#include "Luau/Set.h"
|
||||||
#include "Luau/TypeCheckLimits.h"
|
#include "Luau/TypeCheckLimits.h"
|
||||||
#include "Luau/Variant.h"
|
#include "Luau/Variant.h"
|
||||||
#include "Luau/AnyTypeSummary.h"
|
#include "Luau/AnyTypeSummary.h"
|
||||||
|
@ -56,13 +57,32 @@ struct SourceNode
|
||||||
return forAutocomplete ? dirtyModuleForAutocomplete : dirtyModule;
|
return forAutocomplete ? dirtyModuleForAutocomplete : dirtyModule;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool hasInvalidModuleDependency(bool forAutocomplete) const
|
||||||
|
{
|
||||||
|
return forAutocomplete ? invalidModuleDependencyForAutocomplete : invalidModuleDependency;
|
||||||
|
}
|
||||||
|
|
||||||
|
void setInvalidModuleDependency(bool value, bool forAutocomplete)
|
||||||
|
{
|
||||||
|
if (forAutocomplete)
|
||||||
|
invalidModuleDependencyForAutocomplete = value;
|
||||||
|
else
|
||||||
|
invalidModuleDependency = value;
|
||||||
|
}
|
||||||
|
|
||||||
ModuleName name;
|
ModuleName name;
|
||||||
std::string humanReadableName;
|
std::string humanReadableName;
|
||||||
DenseHashSet<ModuleName> requireSet{{}};
|
DenseHashSet<ModuleName> requireSet{{}};
|
||||||
std::vector<std::pair<ModuleName, Location>> requireLocations;
|
std::vector<std::pair<ModuleName, Location>> requireLocations;
|
||||||
|
Set<ModuleName> dependents{{}};
|
||||||
|
|
||||||
bool dirtySourceModule = true;
|
bool dirtySourceModule = true;
|
||||||
bool dirtyModule = true;
|
bool dirtyModule = true;
|
||||||
bool dirtyModuleForAutocomplete = true;
|
bool dirtyModuleForAutocomplete = true;
|
||||||
|
|
||||||
|
bool invalidModuleDependency = true;
|
||||||
|
bool invalidModuleDependencyForAutocomplete = true;
|
||||||
|
|
||||||
double autocompleteLimitsMult = 1.0;
|
double autocompleteLimitsMult = 1.0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -117,7 +137,7 @@ struct FrontendModuleResolver : ModuleResolver
|
||||||
std::optional<ModuleInfo> resolveModuleInfo(const ModuleName& currentModuleName, const AstExpr& pathExpr) override;
|
std::optional<ModuleInfo> resolveModuleInfo(const ModuleName& currentModuleName, const AstExpr& pathExpr) override;
|
||||||
std::string getHumanReadableModuleName(const ModuleName& moduleName) const override;
|
std::string getHumanReadableModuleName(const ModuleName& moduleName) const override;
|
||||||
|
|
||||||
void setModule(const ModuleName& moduleName, ModulePtr module);
|
bool setModule(const ModuleName& moduleName, ModulePtr module);
|
||||||
void clearModules();
|
void clearModules();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -151,9 +171,13 @@ struct Frontend
|
||||||
// Parse and typecheck module graph
|
// Parse and typecheck module graph
|
||||||
CheckResult check(const ModuleName& name, std::optional<FrontendOptions> optionOverride = {}); // new shininess
|
CheckResult check(const ModuleName& name, std::optional<FrontendOptions> optionOverride = {}); // new shininess
|
||||||
|
|
||||||
|
bool allModuleDependenciesValid(const ModuleName& name, bool forAutocomplete = false) const;
|
||||||
|
|
||||||
bool isDirty(const ModuleName& name, bool forAutocomplete = false) const;
|
bool isDirty(const ModuleName& name, bool forAutocomplete = false) const;
|
||||||
void markDirty(const ModuleName& name, std::vector<ModuleName>* markedDirty = nullptr);
|
void markDirty(const ModuleName& name, std::vector<ModuleName>* markedDirty = nullptr);
|
||||||
|
|
||||||
|
void traverseDependents(const ModuleName& name, std::function<bool(SourceNode&)> processSubtree);
|
||||||
|
|
||||||
/** Borrow a pointer into the SourceModule cache.
|
/** Borrow a pointer into the SourceModule cache.
|
||||||
*
|
*
|
||||||
* Returns nullptr if we don't have it. This could mean that the script
|
* Returns nullptr if we don't have it. This could mean that the script
|
||||||
|
|
|
@ -16,9 +16,17 @@
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
|
||||||
|
LUAU_FASTFLAG(LuauIncrementalAutocompleteCommentDetection)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
using LogLuauProc = void (*)(std::string_view);
|
||||||
|
extern LogLuauProc logLuau;
|
||||||
|
|
||||||
|
void setLogLuau(LogLuauProc ll);
|
||||||
|
void resetLogLuauProc();
|
||||||
|
|
||||||
struct Module;
|
struct Module;
|
||||||
struct AnyTypeSummary;
|
struct AnyTypeSummary;
|
||||||
|
|
||||||
|
@ -55,6 +63,7 @@ struct SourceModule
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
bool isWithinComment(const std::vector<Comment>& commentLocations, Position pos);
|
||||||
bool isWithinComment(const SourceModule& sourceModule, Position pos);
|
bool isWithinComment(const SourceModule& sourceModule, Position pos);
|
||||||
bool isWithinComment(const ParseResult& result, Position pos);
|
bool isWithinComment(const ParseResult& result, Position pos);
|
||||||
|
|
||||||
|
@ -136,6 +145,11 @@ struct Module
|
||||||
TypePackId returnType = nullptr;
|
TypePackId returnType = nullptr;
|
||||||
std::unordered_map<Name, TypeFun> exportedTypeBindings;
|
std::unordered_map<Name, TypeFun> exportedTypeBindings;
|
||||||
|
|
||||||
|
// Arenas related to the DFG must persist after the DFG no longer exists, as
|
||||||
|
// Module objects maintain raw pointers to objects in these arenas.
|
||||||
|
DefArena defArena;
|
||||||
|
RefinementKeyArena keyArena;
|
||||||
|
|
||||||
bool hasModuleScope() const;
|
bool hasModuleScope() const;
|
||||||
ScopePtr getModuleScope() const;
|
ScopePtr getModuleScope() const;
|
||||||
|
|
||||||
|
|
|
@ -95,6 +95,8 @@ struct Scope
|
||||||
// we need that the generic type `T` in both cases is the same, so we use a cache.
|
// 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, TypeId> typeAliasTypeParameters;
|
||||||
std::unordered_map<Name, TypePackId> typeAliasTypePackParameters;
|
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
|
// Returns true iff the left scope encloses the right scope. A Scope* equal to
|
||||||
|
|
|
@ -19,10 +19,10 @@ struct SimplifyResult
|
||||||
DenseHashSet<TypeId> blockedTypes;
|
DenseHashSet<TypeId> blockedTypes;
|
||||||
};
|
};
|
||||||
|
|
||||||
SimplifyResult simplifyIntersection(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, TypeId ty, TypeId discriminant);
|
SimplifyResult simplifyIntersection(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, TypeId left, TypeId right);
|
||||||
SimplifyResult simplifyIntersection(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, std::set<TypeId> parts);
|
SimplifyResult simplifyIntersection(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, std::set<TypeId> parts);
|
||||||
|
|
||||||
SimplifyResult simplifyUnion(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, TypeId ty, TypeId discriminant);
|
SimplifyResult simplifyUnion(NotNull<BuiltinTypes> builtinTypes, NotNull<TypeArena> arena, TypeId left, TypeId right);
|
||||||
|
|
||||||
enum class Relation
|
enum class Relation
|
||||||
{
|
{
|
||||||
|
|
|
@ -69,12 +69,16 @@ using Name = std::string;
|
||||||
// A free type is one whose exact shape has yet to be fully determined.
|
// A free type is one whose exact shape has yet to be fully determined.
|
||||||
struct FreeType
|
struct FreeType
|
||||||
{
|
{
|
||||||
|
// New constructors
|
||||||
|
explicit FreeType(TypeLevel level, TypeId lowerBound, TypeId upperBound);
|
||||||
|
// This one got promoted to explicit
|
||||||
|
explicit FreeType(Scope* scope, TypeId lowerBound, TypeId upperBound);
|
||||||
|
explicit FreeType(Scope* scope, TypeLevel level, TypeId lowerBound, TypeId upperBound);
|
||||||
|
// Old constructors
|
||||||
explicit FreeType(TypeLevel level);
|
explicit FreeType(TypeLevel level);
|
||||||
explicit FreeType(Scope* scope);
|
explicit FreeType(Scope* scope);
|
||||||
FreeType(Scope* scope, TypeLevel level);
|
FreeType(Scope* scope, TypeLevel level);
|
||||||
|
|
||||||
FreeType(Scope* scope, TypeId lowerBound, TypeId upperBound);
|
|
||||||
|
|
||||||
int index;
|
int index;
|
||||||
TypeLevel level;
|
TypeLevel level;
|
||||||
Scope* scope = nullptr;
|
Scope* scope = nullptr;
|
||||||
|
@ -131,14 +135,14 @@ struct BlockedType
|
||||||
BlockedType();
|
BlockedType();
|
||||||
int index;
|
int index;
|
||||||
|
|
||||||
Constraint* getOwner() const;
|
const Constraint* getOwner() const;
|
||||||
void setOwner(Constraint* newOwner);
|
void setOwner(const Constraint* newOwner);
|
||||||
void replaceOwner(Constraint* newOwner);
|
void replaceOwner(const Constraint* newOwner);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// The constraint that is intended to unblock this type. Other constraints
|
// The constraint that is intended to unblock this type. Other constraints
|
||||||
// should block on this constraint if present.
|
// should block on this constraint if present.
|
||||||
Constraint* owner = nullptr;
|
const Constraint* owner = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct PrimitiveType
|
struct PrimitiveType
|
||||||
|
@ -279,9 +283,6 @@ struct WithPredicate
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
using MagicFunction = std::function<std::optional<
|
|
||||||
WithPredicate<TypePackId>>(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>)>;
|
|
||||||
|
|
||||||
struct MagicFunctionCallContext
|
struct MagicFunctionCallContext
|
||||||
{
|
{
|
||||||
NotNull<struct ConstraintSolver> solver;
|
NotNull<struct ConstraintSolver> solver;
|
||||||
|
@ -291,7 +292,6 @@ struct MagicFunctionCallContext
|
||||||
TypePackId result;
|
TypePackId result;
|
||||||
};
|
};
|
||||||
|
|
||||||
using DcrMagicFunction = std::function<bool(MagicFunctionCallContext)>;
|
|
||||||
struct MagicRefinementContext
|
struct MagicRefinementContext
|
||||||
{
|
{
|
||||||
NotNull<Scope> scope;
|
NotNull<Scope> scope;
|
||||||
|
@ -308,8 +308,29 @@ struct MagicFunctionTypeCheckContext
|
||||||
NotNull<Scope> checkScope;
|
NotNull<Scope> checkScope;
|
||||||
};
|
};
|
||||||
|
|
||||||
using DcrMagicRefinement = void (*)(const MagicRefinementContext&);
|
struct MagicFunction
|
||||||
using DcrMagicFunctionTypeCheck = std::function<void(const MagicFunctionTypeCheckContext&)>;
|
{
|
||||||
|
virtual std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) = 0;
|
||||||
|
|
||||||
|
// Callback to allow custom typechecking of builtin function calls whose argument types
|
||||||
|
// will only be resolved after constraint solving. For example, the arguments to string.format
|
||||||
|
// have types that can only be decided after parsing the format string and unifying
|
||||||
|
// with the passed in values, but the correctness of the call can only be decided after
|
||||||
|
// all the types have been finalized.
|
||||||
|
virtual bool infer(const MagicFunctionCallContext&) = 0;
|
||||||
|
virtual void refine(const MagicRefinementContext&) {}
|
||||||
|
|
||||||
|
// If a magic function needs to do its own special typechecking, do it here.
|
||||||
|
// Returns true if magic typechecking was performed. Return false if the
|
||||||
|
// default typechecking logic should run.
|
||||||
|
virtual bool typeCheck(const MagicFunctionTypeCheckContext&)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~MagicFunction() {}
|
||||||
|
};
|
||||||
|
|
||||||
struct FunctionType
|
struct FunctionType
|
||||||
{
|
{
|
||||||
// Global monomorphic function
|
// Global monomorphic function
|
||||||
|
@ -367,16 +388,7 @@ struct FunctionType
|
||||||
Scope* scope = nullptr;
|
Scope* scope = nullptr;
|
||||||
TypePackId argTypes;
|
TypePackId argTypes;
|
||||||
TypePackId retTypes;
|
TypePackId retTypes;
|
||||||
MagicFunction magicFunction = nullptr;
|
std::shared_ptr<MagicFunction> magic = nullptr;
|
||||||
DcrMagicFunction dcrMagicFunction = nullptr;
|
|
||||||
DcrMagicRefinement dcrMagicRefinement = nullptr;
|
|
||||||
|
|
||||||
// Callback to allow custom typechecking of builtin function calls whose argument types
|
|
||||||
// will only be resolved after constraint solving. For example, the arguments to string.format
|
|
||||||
// have types that can only be decided after parsing the format string and unifying
|
|
||||||
// with the passed in values, but the correctness of the call can only be decided after
|
|
||||||
// all the types have been finalized.
|
|
||||||
DcrMagicFunctionTypeCheck dcrMagicTypeCheck = nullptr;
|
|
||||||
|
|
||||||
bool hasSelf;
|
bool hasSelf;
|
||||||
// `hasNoFreeOrGenericTypes` should be true if and only if the type does not have any free or generic types present inside it.
|
// `hasNoFreeOrGenericTypes` should be true if and only if the type does not have any free or generic types present inside it.
|
||||||
|
|
|
@ -32,9 +32,13 @@ struct TypeArena
|
||||||
|
|
||||||
TypeId addTV(Type&& tv);
|
TypeId addTV(Type&& tv);
|
||||||
|
|
||||||
TypeId freshType(TypeLevel level);
|
TypeId freshType(NotNull<BuiltinTypes> builtins, TypeLevel level);
|
||||||
TypeId freshType(Scope* scope);
|
TypeId freshType(NotNull<BuiltinTypes> builtins, Scope* scope);
|
||||||
TypeId freshType(Scope* scope, TypeLevel level);
|
TypeId freshType(NotNull<BuiltinTypes> builtins, Scope* scope, TypeLevel level);
|
||||||
|
|
||||||
|
TypeId freshType_DEPRECATED(TypeLevel level);
|
||||||
|
TypeId freshType_DEPRECATED(Scope* scope);
|
||||||
|
TypeId freshType_DEPRECATED(Scope* scope, TypeLevel level);
|
||||||
|
|
||||||
TypePackId freshTypePack(Scope* scope);
|
TypePackId freshTypePack(Scope* scope);
|
||||||
|
|
||||||
|
|
|
@ -63,7 +63,7 @@ void check(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<Simplifier> simplifier,
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<UnifierSharedState> sharedState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
NotNull<TypeCheckLimits> limits,
|
NotNull<TypeCheckLimits> limits,
|
||||||
DcrLogger* logger,
|
DcrLogger* logger,
|
||||||
const SourceModule& sourceModule,
|
const SourceModule& sourceModule,
|
||||||
|
@ -116,14 +116,14 @@ private:
|
||||||
std::optional<StackPusher> pushStack(AstNode* node);
|
std::optional<StackPusher> pushStack(AstNode* node);
|
||||||
void checkForInternalTypeFunction(TypeId ty, Location location);
|
void checkForInternalTypeFunction(TypeId ty, Location location);
|
||||||
TypeId checkForTypeFunctionInhabitance(TypeId instance, Location location);
|
TypeId checkForTypeFunctionInhabitance(TypeId instance, Location location);
|
||||||
TypePackId lookupPack(AstExpr* expr);
|
TypePackId lookupPack(AstExpr* expr) const;
|
||||||
TypeId lookupType(AstExpr* expr);
|
TypeId lookupType(AstExpr* expr);
|
||||||
TypeId lookupAnnotation(AstType* annotation);
|
TypeId lookupAnnotation(AstType* annotation);
|
||||||
std::optional<TypePackId> lookupPackAnnotation(AstTypePack* annotation);
|
std::optional<TypePackId> lookupPackAnnotation(AstTypePack* annotation) const;
|
||||||
TypeId lookupExpectedType(AstExpr* expr);
|
TypeId lookupExpectedType(AstExpr* expr) const;
|
||||||
TypePackId lookupExpectedPack(AstExpr* expr, TypeArena& arena);
|
TypePackId lookupExpectedPack(AstExpr* expr, TypeArena& arena) const;
|
||||||
TypePackId reconstructPack(AstArray<AstExpr*> exprs, TypeArena& arena);
|
TypePackId reconstructPack(AstArray<AstExpr*> exprs, TypeArena& arena);
|
||||||
Scope* findInnermostScope(Location location);
|
Scope* findInnermostScope(Location location) const;
|
||||||
void visit(AstStat* stat);
|
void visit(AstStat* stat);
|
||||||
void visit(AstStatIf* ifStatement);
|
void visit(AstStatIf* ifStatement);
|
||||||
void visit(AstStatWhile* whileStatement);
|
void visit(AstStatWhile* whileStatement);
|
||||||
|
@ -160,7 +160,7 @@ private:
|
||||||
void visit(AstExprVarargs* expr);
|
void visit(AstExprVarargs* expr);
|
||||||
void visitCall(AstExprCall* call);
|
void visitCall(AstExprCall* call);
|
||||||
void visit(AstExprCall* call);
|
void visit(AstExprCall* call);
|
||||||
std::optional<TypeId> tryStripUnionFromNil(TypeId ty);
|
std::optional<TypeId> tryStripUnionFromNil(TypeId ty) const;
|
||||||
TypeId stripFromNilAndReport(TypeId ty, const Location& location);
|
TypeId stripFromNilAndReport(TypeId ty, const Location& location);
|
||||||
void visitExprName(AstExpr* expr, Location location, const std::string& propName, ValueContext context, TypeId astIndexExprTy);
|
void visitExprName(AstExpr* expr, Location location, const std::string& propName, ValueContext context, TypeId astIndexExprTy);
|
||||||
void visit(AstExprIndexName* indexName, ValueContext context);
|
void visit(AstExprIndexName* indexName, ValueContext context);
|
||||||
|
@ -175,7 +175,7 @@ private:
|
||||||
void visit(AstExprInterpString* interpString);
|
void visit(AstExprInterpString* interpString);
|
||||||
void visit(AstExprError* expr);
|
void visit(AstExprError* expr);
|
||||||
TypeId flattenPack(TypePackId pack);
|
TypeId flattenPack(TypePackId pack);
|
||||||
void visitGenerics(AstArray<AstGenericType> generics, AstArray<AstGenericTypePack> genericPacks);
|
void visitGenerics(AstArray<AstGenericType*> generics, AstArray<AstGenericTypePack*> genericPacks);
|
||||||
void visit(AstType* ty);
|
void visit(AstType* ty);
|
||||||
void visit(AstTypeReference* ty);
|
void visit(AstTypeReference* ty);
|
||||||
void visit(AstTypeTable* table);
|
void visit(AstTypeTable* table);
|
||||||
|
|
|
@ -241,6 +241,9 @@ struct BuiltinTypeFunctions
|
||||||
TypeFunction indexFunc;
|
TypeFunction indexFunc;
|
||||||
TypeFunction rawgetFunc;
|
TypeFunction rawgetFunc;
|
||||||
|
|
||||||
|
TypeFunction setmetatableFunc;
|
||||||
|
TypeFunction getmetatableFunc;
|
||||||
|
|
||||||
void addToScope(NotNull<TypeArena> arena, NotNull<Scope> scope) const;
|
void addToScope(NotNull<TypeArena> arena, NotNull<Scope> scope) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -119,7 +119,14 @@ struct TypeFunctionVariadicTypePack
|
||||||
TypeFunctionTypeId type;
|
TypeFunctionTypeId type;
|
||||||
};
|
};
|
||||||
|
|
||||||
using TypeFunctionTypePackVariant = Variant<TypeFunctionTypePack, TypeFunctionVariadicTypePack>;
|
struct TypeFunctionGenericTypePack
|
||||||
|
{
|
||||||
|
bool isNamed = false;
|
||||||
|
|
||||||
|
std::string name;
|
||||||
|
};
|
||||||
|
|
||||||
|
using TypeFunctionTypePackVariant = Variant<TypeFunctionTypePack, TypeFunctionVariadicTypePack, TypeFunctionGenericTypePack>;
|
||||||
|
|
||||||
struct TypeFunctionTypePackVar
|
struct TypeFunctionTypePackVar
|
||||||
{
|
{
|
||||||
|
@ -135,6 +142,9 @@ struct TypeFunctionTypePackVar
|
||||||
|
|
||||||
struct TypeFunctionFunctionType
|
struct TypeFunctionFunctionType
|
||||||
{
|
{
|
||||||
|
std::vector<TypeFunctionTypeId> generics;
|
||||||
|
std::vector<TypeFunctionTypePackId> genericPacks;
|
||||||
|
|
||||||
TypeFunctionTypePackId argTypes;
|
TypeFunctionTypePackId argTypes;
|
||||||
TypeFunctionTypePackId retTypes;
|
TypeFunctionTypePackId retTypes;
|
||||||
};
|
};
|
||||||
|
@ -210,6 +220,14 @@ struct TypeFunctionClassType
|
||||||
std::string name;
|
std::string name;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct TypeFunctionGenericType
|
||||||
|
{
|
||||||
|
bool isNamed = false;
|
||||||
|
bool isPack = false;
|
||||||
|
|
||||||
|
std::string name;
|
||||||
|
};
|
||||||
|
|
||||||
using TypeFunctionTypeVariant = Luau::Variant<
|
using TypeFunctionTypeVariant = Luau::Variant<
|
||||||
TypeFunctionPrimitiveType,
|
TypeFunctionPrimitiveType,
|
||||||
TypeFunctionAnyType,
|
TypeFunctionAnyType,
|
||||||
|
@ -221,7 +239,8 @@ using TypeFunctionTypeVariant = Luau::Variant<
|
||||||
TypeFunctionNegationType,
|
TypeFunctionNegationType,
|
||||||
TypeFunctionFunctionType,
|
TypeFunctionFunctionType,
|
||||||
TypeFunctionTableType,
|
TypeFunctionTableType,
|
||||||
TypeFunctionClassType>;
|
TypeFunctionClassType,
|
||||||
|
TypeFunctionGenericType>;
|
||||||
|
|
||||||
struct TypeFunctionType
|
struct TypeFunctionType
|
||||||
{
|
{
|
||||||
|
|
|
@ -399,8 +399,8 @@ private:
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
std::optional<TypeLevel> levelOpt,
|
std::optional<TypeLevel> levelOpt,
|
||||||
const AstNode& node,
|
const AstNode& node,
|
||||||
const AstArray<AstGenericType>& genericNames,
|
const AstArray<AstGenericType*>& genericNames,
|
||||||
const AstArray<AstGenericTypePack>& genericPackNames,
|
const AstArray<AstGenericTypePack*>& genericPackNames,
|
||||||
bool useCache = false
|
bool useCache = false
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,7 @@ struct InConditionalContext
|
||||||
TypeContext* typeContext;
|
TypeContext* typeContext;
|
||||||
TypeContext oldValue;
|
TypeContext oldValue;
|
||||||
|
|
||||||
InConditionalContext(TypeContext* c)
|
explicit InConditionalContext(TypeContext* c)
|
||||||
: typeContext(c)
|
: typeContext(c)
|
||||||
, oldValue(*c)
|
, oldValue(*c)
|
||||||
{
|
{
|
||||||
|
@ -280,4 +280,13 @@ std::vector<TypeId> findBlockedTypesIn(AstExprTable* expr, NotNull<DenseHashMap<
|
||||||
*/
|
*/
|
||||||
std::vector<TypeId> findBlockedArgTypesIn(AstExprCall* expr, NotNull<DenseHashMap<const AstExpr*, TypeId>> astTypes);
|
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
|
} // namespace Luau
|
||||||
|
|
|
@ -49,6 +49,27 @@ struct UnifierSharedState
|
||||||
DenseHashSet<TypePackId> tempSeenTp{nullptr};
|
DenseHashSet<TypePackId> tempSeenTp{nullptr};
|
||||||
|
|
||||||
UnifierCounters counters;
|
UnifierCounters counters;
|
||||||
|
|
||||||
|
bool reentrantTypeReduction = false;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
struct TypeReductionRentrancyGuard final
|
||||||
|
{
|
||||||
|
explicit TypeReductionRentrancyGuard(NotNull<UnifierSharedState> sharedState)
|
||||||
|
: sharedState{sharedState}
|
||||||
|
{
|
||||||
|
sharedState->reentrantTypeReduction = true;
|
||||||
|
}
|
||||||
|
~TypeReductionRentrancyGuard()
|
||||||
|
{
|
||||||
|
sharedState->reentrantTypeReduction = false;
|
||||||
|
}
|
||||||
|
TypeReductionRentrancyGuard(const TypeReductionRentrancyGuard&) = delete;
|
||||||
|
TypeReductionRentrancyGuard(TypeReductionRentrancyGuard&&) = delete;
|
||||||
|
|
||||||
|
private:
|
||||||
|
NotNull<UnifierSharedState> sharedState;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Luau
|
} // namespace Luau
|
||||||
|
|
|
@ -85,6 +85,8 @@ struct GenericTypeVisitor
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~GenericTypeVisitor() {}
|
||||||
|
|
||||||
virtual void cycle(TypeId) {}
|
virtual void cycle(TypeId) {}
|
||||||
virtual void cycle(TypePackId) {}
|
virtual void cycle(TypePackId) {}
|
||||||
|
|
||||||
|
|
|
@ -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)
|
void AnyTypeSummary::visit(const Scope* scope, AstStatLocal* local, const Module* module, NotNull<BuiltinTypes> builtinTypes)
|
||||||
|
|
|
@ -1161,6 +1161,19 @@ struct AstJsonEncoder : public AstVisitor
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool visit(class AstTypeGroup* node) override
|
||||||
|
{
|
||||||
|
writeNode(
|
||||||
|
node,
|
||||||
|
"AstTypeGroup",
|
||||||
|
[&]()
|
||||||
|
{
|
||||||
|
write("type", node->type);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool visit(class AstTypeSingletonBool* node) override
|
bool visit(class AstTypeSingletonBool* node) override
|
||||||
{
|
{
|
||||||
writeNode(
|
writeNode(
|
||||||
|
|
|
@ -13,7 +13,7 @@
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauDocumentationAtPosition)
|
LUAU_FASTFLAG(LuauExtendStatEndPosWithSemicolon)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -42,12 +42,27 @@ struct AutocompleteNodeFinder : public AstVisitor
|
||||||
}
|
}
|
||||||
|
|
||||||
bool visit(AstStat* stat) override
|
bool visit(AstStat* stat) override
|
||||||
|
{
|
||||||
|
if (FFlag::LuauExtendStatEndPosWithSemicolon)
|
||||||
|
{
|
||||||
|
// Consider 'local myLocal = 4;|' and 'local myLocal = 4', where '|' is the cursor position. In both cases, the cursor position is equal
|
||||||
|
// to `AstStatLocal.location.end`. However, in the first case (semicolon), we are starting a new statement, whilst in the second case
|
||||||
|
// (no semicolon) we are still part of the AstStatLocal, hence the different comparison check.
|
||||||
|
if (stat->location.begin < pos && (stat->hasSemicolon ? pos < stat->location.end : pos <= stat->location.end))
|
||||||
|
{
|
||||||
|
ancestry.push_back(stat);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
if (stat->location.begin < pos && pos <= stat->location.end)
|
if (stat->location.begin < pos && pos <= stat->location.end)
|
||||||
{
|
{
|
||||||
ancestry.push_back(stat);
|
ancestry.push_back(stat);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -518,7 +533,6 @@ static std::optional<DocumentationSymbol> getMetatableDocumentation(
|
||||||
const AstName& index
|
const AstName& index
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
LUAU_ASSERT(FFlag::LuauDocumentationAtPosition);
|
|
||||||
auto indexIt = mtable->props.find("__index");
|
auto indexIt = mtable->props.find("__index");
|
||||||
if (indexIt == mtable->props.end())
|
if (indexIt == mtable->props.end())
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
|
@ -574,8 +588,6 @@ std::optional<DocumentationSymbol> getDocumentationSymbolAtPosition(const Source
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (const ClassType* ctv = get<ClassType>(parentTy))
|
else if (const ClassType* ctv = get<ClassType>(parentTy))
|
||||||
{
|
|
||||||
if (FFlag::LuauDocumentationAtPosition)
|
|
||||||
{
|
{
|
||||||
while (ctv)
|
while (ctv)
|
||||||
{
|
{
|
||||||
|
@ -594,25 +606,7 @@ std::optional<DocumentationSymbol> getDocumentationSymbolAtPosition(const Source
|
||||||
ctv = ctv->parent ? Luau::get<Luau::ClassType>(*ctv->parent) : nullptr;
|
ctv = ctv->parent ? Luau::get<Luau::ClassType>(*ctv->parent) : nullptr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else if (const PrimitiveType* ptv = get<PrimitiveType>(parentTy); ptv && ptv->metatable)
|
||||||
{
|
|
||||||
if (auto propIt = ctv->props.find(indexName->index.value); propIt != ctv->props.end())
|
|
||||||
{
|
|
||||||
if (FFlag::LuauSolverV2)
|
|
||||||
{
|
|
||||||
if (auto ty = propIt->second.readTy)
|
|
||||||
return checkOverloadedDocumentationSymbol(module, *ty, parentExpr, propIt->second.documentationSymbol);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
return checkOverloadedDocumentationSymbol(
|
|
||||||
module, propIt->second.type(), parentExpr, propIt->second.documentationSymbol
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (FFlag::LuauDocumentationAtPosition)
|
|
||||||
{
|
|
||||||
if (const PrimitiveType* ptv = get<PrimitiveType>(parentTy); ptv && ptv->metatable)
|
|
||||||
{
|
{
|
||||||
if (auto mtable = get<TableType>(*ptv->metatable))
|
if (auto mtable = get<TableType>(*ptv->metatable))
|
||||||
{
|
{
|
||||||
|
@ -622,7 +616,6 @@ std::optional<DocumentationSymbol> getDocumentationSymbolAtPosition(const Source
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else if (AstExprFunction* fn = targetExpr->as<AstExprFunction>())
|
else if (AstExprFunction* fn = targetExpr->as<AstExprFunction>())
|
||||||
{
|
{
|
||||||
// Handle event connection-like structures where we have
|
// Handle event connection-like structures where we have
|
||||||
|
|
|
@ -25,6 +25,7 @@ LUAU_FASTINT(LuauTypeInferIterationLimit)
|
||||||
LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
LUAU_FASTFLAGVARIABLE(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauAutocompleteUseLimits)
|
||||||
|
|
||||||
static const std::unordered_set<std::string> kStatementStartingKeywords =
|
static const std::unordered_set<std::string> kStatementStartingKeywords =
|
||||||
{"while", "if", "local", "repeat", "function", "do", "for", "return", "break", "continue", "type", "export"};
|
{"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.normalize = false;
|
||||||
unifier.checkInhabited = false;
|
unifier.checkInhabited = false;
|
||||||
|
|
||||||
|
if (FFlag::LuauAutocompleteUseLimits)
|
||||||
|
{
|
||||||
|
unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit;
|
||||||
|
unifierState.counters.iterationLimit = FInt::LuauTypeInferIterationLimit;
|
||||||
|
}
|
||||||
|
|
||||||
return unifier.canUnify(subTy, superTy).empty();
|
return unifier.canUnify(subTy, superTy).empty();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,50 +29,82 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauTypestateBuiltins2)
|
LUAU_FASTFLAGVARIABLE(LuauStringFormatErrorSuppression)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauStringFormatArityFix)
|
|
||||||
LUAU_FASTFLAG(AutocompleteRequirePathSuggestions2)
|
LUAU_FASTFLAG(AutocompleteRequirePathSuggestions2)
|
||||||
LUAU_FASTFLAG(LuauVectorDefinitionsExtra)
|
LUAU_FASTFLAGVARIABLE(LuauTableCloneClonesType3)
|
||||||
|
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauFreezeIgnorePersistent)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauFollowTableFreeze)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionSelect(
|
struct MagicSelect final : MagicFunction
|
||||||
TypeChecker& typechecker,
|
{
|
||||||
const ScopePtr& scope,
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
const AstExprCall& expr,
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
WithPredicate<TypePackId> withPredicate
|
};
|
||||||
);
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionSetMetaTable(
|
|
||||||
TypeChecker& typechecker,
|
|
||||||
const ScopePtr& scope,
|
|
||||||
const AstExprCall& expr,
|
|
||||||
WithPredicate<TypePackId> withPredicate
|
|
||||||
);
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionAssert(
|
|
||||||
TypeChecker& typechecker,
|
|
||||||
const ScopePtr& scope,
|
|
||||||
const AstExprCall& expr,
|
|
||||||
WithPredicate<TypePackId> withPredicate
|
|
||||||
);
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionPack(
|
|
||||||
TypeChecker& typechecker,
|
|
||||||
const ScopePtr& scope,
|
|
||||||
const AstExprCall& expr,
|
|
||||||
WithPredicate<TypePackId> withPredicate
|
|
||||||
);
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionRequire(
|
|
||||||
TypeChecker& typechecker,
|
|
||||||
const ScopePtr& scope,
|
|
||||||
const AstExprCall& expr,
|
|
||||||
WithPredicate<TypePackId> withPredicate
|
|
||||||
);
|
|
||||||
|
|
||||||
|
struct MagicSetMetatable final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
static bool dcrMagicFunctionSelect(MagicFunctionCallContext context);
|
struct MagicAssert final : MagicFunction
|
||||||
static bool dcrMagicFunctionRequire(MagicFunctionCallContext context);
|
{
|
||||||
static bool dcrMagicFunctionPack(MagicFunctionCallContext context);
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
static bool dcrMagicFunctionFreeze(MagicFunctionCallContext context);
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicPack final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicRequire final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicClone final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicFreeze final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicFormat final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
bool typeCheck(const MagicFunctionTypeCheckContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicMatch final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicGmatch final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct MagicFind final : MagicFunction
|
||||||
|
{
|
||||||
|
std::optional<WithPredicate<TypePackId>> handleOldSolver(struct TypeChecker&, const std::shared_ptr<struct Scope>&, const class AstExprCall&, WithPredicate<TypePackId>) override;
|
||||||
|
bool infer(const MagicFunctionCallContext& ctx) override;
|
||||||
|
};
|
||||||
|
|
||||||
TypeId makeUnion(TypeArena& arena, std::vector<TypeId>&& types)
|
TypeId makeUnion(TypeArena& arena, std::vector<TypeId>&& types)
|
||||||
{
|
{
|
||||||
|
@ -167,34 +199,10 @@ TypeId makeFunction(
|
||||||
return arena.addType(std::move(ftv));
|
return arena.addType(std::move(ftv));
|
||||||
}
|
}
|
||||||
|
|
||||||
void attachMagicFunction(TypeId ty, MagicFunction fn)
|
void attachMagicFunction(TypeId ty, std::shared_ptr<MagicFunction> magic)
|
||||||
{
|
{
|
||||||
if (auto ftv = getMutable<FunctionType>(ty))
|
if (auto ftv = getMutable<FunctionType>(ty))
|
||||||
ftv->magicFunction = fn;
|
ftv->magic = std::move(magic);
|
||||||
else
|
|
||||||
LUAU_ASSERT(!"Got a non functional type");
|
|
||||||
}
|
|
||||||
|
|
||||||
void attachDcrMagicFunction(TypeId ty, DcrMagicFunction fn)
|
|
||||||
{
|
|
||||||
if (auto ftv = getMutable<FunctionType>(ty))
|
|
||||||
ftv->dcrMagicFunction = fn;
|
|
||||||
else
|
|
||||||
LUAU_ASSERT(!"Got a non functional type");
|
|
||||||
}
|
|
||||||
|
|
||||||
void attachDcrMagicRefinement(TypeId ty, DcrMagicRefinement fn)
|
|
||||||
{
|
|
||||||
if (auto ftv = getMutable<FunctionType>(ty))
|
|
||||||
ftv->dcrMagicRefinement = fn;
|
|
||||||
else
|
|
||||||
LUAU_ASSERT(!"Got a non functional type");
|
|
||||||
}
|
|
||||||
|
|
||||||
void attachDcrMagicFunctionTypeCheck(TypeId ty, DcrMagicFunctionTypeCheck fn)
|
|
||||||
{
|
|
||||||
if (auto ftv = getMutable<FunctionType>(ty))
|
|
||||||
ftv->dcrMagicTypeCheck = fn;
|
|
||||||
else
|
else
|
||||||
LUAU_ASSERT(!"Got a non functional type");
|
LUAU_ASSERT(!"Got a non functional type");
|
||||||
}
|
}
|
||||||
|
@ -301,8 +309,6 @@ void registerBuiltinGlobals(Frontend& frontend, GlobalTypes& globals, bool typeC
|
||||||
addGlobalBinding(globals, "string", it->second.type(), "@luau");
|
addGlobalBinding(globals, "string", it->second.type(), "@luau");
|
||||||
|
|
||||||
// Setup 'vector' metatable
|
// Setup 'vector' metatable
|
||||||
if (FFlag::LuauVectorDefinitionsExtra)
|
|
||||||
{
|
|
||||||
if (auto it = globals.globalScope->exportedTypeBindings.find("vector"); it != globals.globalScope->exportedTypeBindings.end())
|
if (auto it = globals.globalScope->exportedTypeBindings.find("vector"); it != globals.globalScope->exportedTypeBindings.end())
|
||||||
{
|
{
|
||||||
TypeId vectorTy = it->second.type;
|
TypeId vectorTy = it->second.type;
|
||||||
|
@ -323,7 +329,6 @@ void registerBuiltinGlobals(Frontend& frontend, GlobalTypes& globals, bool typeC
|
||||||
metatableTy->props["__div"] = {makeIntersection(arena, mulOverloads)};
|
metatableTy->props["__div"] = {makeIntersection(arena, mulOverloads)};
|
||||||
metatableTy->props["__idiv"] = {makeIntersection(arena, mulOverloads)};
|
metatableTy->props["__idiv"] = {makeIntersection(arena, mulOverloads)};
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// next<K, V>(t: Table<K, V>, i: K?) -> (K?, V)
|
// next<K, V>(t: Table<K, V>, i: K?) -> (K?, V)
|
||||||
TypePackId nextArgsTypePack = arena.addTypePack(TypePack{{mapOfKtoV, makeOption(builtinTypes, arena, genericK)}});
|
TypePackId nextArgsTypePack = arena.addTypePack(TypePack{{mapOfKtoV, makeOption(builtinTypes, arena, genericK)}});
|
||||||
|
@ -395,7 +400,7 @@ void registerBuiltinGlobals(Frontend& frontend, GlobalTypes& globals, bool typeC
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
attachMagicFunction(getGlobalBinding(globals, "assert"), magicFunctionAssert);
|
attachMagicFunction(getGlobalBinding(globals, "assert"), std::make_shared<MagicAssert>());
|
||||||
|
|
||||||
if (FFlag::LuauSolverV2)
|
if (FFlag::LuauSolverV2)
|
||||||
{
|
{
|
||||||
|
@ -411,9 +416,8 @@ void registerBuiltinGlobals(Frontend& frontend, GlobalTypes& globals, bool typeC
|
||||||
addGlobalBinding(globals, "assert", assertTy, "@luau");
|
addGlobalBinding(globals, "assert", assertTy, "@luau");
|
||||||
}
|
}
|
||||||
|
|
||||||
attachMagicFunction(getGlobalBinding(globals, "setmetatable"), magicFunctionSetMetaTable);
|
attachMagicFunction(getGlobalBinding(globals, "setmetatable"), std::make_shared<MagicSetMetatable>());
|
||||||
attachMagicFunction(getGlobalBinding(globals, "select"), magicFunctionSelect);
|
attachMagicFunction(getGlobalBinding(globals, "select"), std::make_shared<MagicSelect>());
|
||||||
attachDcrMagicFunction(getGlobalBinding(globals, "select"), dcrMagicFunctionSelect);
|
|
||||||
|
|
||||||
if (TableType* ttv = getMutable<TableType>(getGlobalBinding(globals, "table")))
|
if (TableType* ttv = getMutable<TableType>(getGlobalBinding(globals, "table")))
|
||||||
{
|
{
|
||||||
|
@ -444,23 +448,21 @@ void registerBuiltinGlobals(Frontend& frontend, GlobalTypes& globals, bool typeC
|
||||||
ttv->props["foreach"].deprecated = true;
|
ttv->props["foreach"].deprecated = true;
|
||||||
ttv->props["foreachi"].deprecated = true;
|
ttv->props["foreachi"].deprecated = true;
|
||||||
|
|
||||||
attachMagicFunction(ttv->props["pack"].type(), magicFunctionPack);
|
attachMagicFunction(ttv->props["pack"].type(), std::make_shared<MagicPack>());
|
||||||
attachDcrMagicFunction(ttv->props["pack"].type(), dcrMagicFunctionPack);
|
if (FFlag::LuauTableCloneClonesType3)
|
||||||
if (FFlag::LuauTypestateBuiltins2)
|
attachMagicFunction(ttv->props["clone"].type(), std::make_shared<MagicClone>());
|
||||||
attachDcrMagicFunction(ttv->props["freeze"].type(), dcrMagicFunctionFreeze);
|
attachMagicFunction(ttv->props["freeze"].type(), std::make_shared<MagicFreeze>());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::AutocompleteRequirePathSuggestions2)
|
if (FFlag::AutocompleteRequirePathSuggestions2)
|
||||||
{
|
{
|
||||||
TypeId requireTy = getGlobalBinding(globals, "require");
|
TypeId requireTy = getGlobalBinding(globals, "require");
|
||||||
attachTag(requireTy, kRequireTagName);
|
attachTag(requireTy, kRequireTagName);
|
||||||
attachMagicFunction(requireTy, magicFunctionRequire);
|
attachMagicFunction(requireTy, std::make_shared<MagicRequire>());
|
||||||
attachDcrMagicFunction(requireTy, dcrMagicFunctionRequire);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
attachMagicFunction(getGlobalBinding(globals, "require"), magicFunctionRequire);
|
attachMagicFunction(getGlobalBinding(globals, "require"), std::make_shared<MagicRequire>());
|
||||||
attachDcrMagicFunction(getGlobalBinding(globals, "require"), dcrMagicFunctionRequire);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -500,7 +502,7 @@ static std::vector<TypeId> parseFormatString(NotNull<BuiltinTypes> builtinTypes,
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<WithPredicate<TypePackId>> magicFunctionFormat(
|
std::optional<WithPredicate<TypePackId>> MagicFormat::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -550,7 +552,7 @@ std::optional<WithPredicate<TypePackId>> magicFunctionFormat(
|
||||||
return WithPredicate<TypePackId>{arena.addTypePack({typechecker.stringType})};
|
return WithPredicate<TypePackId>{arena.addTypePack({typechecker.stringType})};
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionFormat(MagicFunctionCallContext context)
|
bool MagicFormat::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
TypeArena* arena = context.solver->arena;
|
TypeArena* arena = context.solver->arena;
|
||||||
|
|
||||||
|
@ -594,7 +596,7 @@ static bool dcrMagicFunctionFormat(MagicFunctionCallContext context)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void dcrMagicFunctionTypeCheckFormat(MagicFunctionTypeCheckContext context)
|
bool MagicFormat::typeCheck(const MagicFunctionTypeCheckContext& context)
|
||||||
{
|
{
|
||||||
AstExprConstantString* fmt = nullptr;
|
AstExprConstantString* fmt = nullptr;
|
||||||
if (auto index = context.callSite->func->as<AstExprIndexName>(); index && context.callSite->self)
|
if (auto index = context.callSite->func->as<AstExprIndexName>(); index && context.callSite->self)
|
||||||
|
@ -610,9 +612,8 @@ static void dcrMagicFunctionTypeCheckFormat(MagicFunctionTypeCheckContext contex
|
||||||
|
|
||||||
if (!fmt)
|
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;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<TypeId> expected = parseFormatString(context.builtinTypes, fmt->value.data, fmt->value.size);
|
std::vector<TypeId> expected = parseFormatString(context.builtinTypes, fmt->value.data, fmt->value.size);
|
||||||
|
@ -629,12 +630,33 @@ static void dcrMagicFunctionTypeCheckFormat(MagicFunctionTypeCheckContext contex
|
||||||
Location location = context.callSite->args.data[i + (calledWithSelf ? 0 : paramOffset)]->location;
|
Location location = context.callSite->args.data[i + (calledWithSelf ? 0 : paramOffset)]->location;
|
||||||
// use subtyping instead here
|
// use subtyping instead here
|
||||||
SubtypingResult result = context.typechecker->subtyping->isSubtype(actualTy, expectedTy, context.checkScope);
|
SubtypingResult result = context.typechecker->subtyping->isSubtype(actualTy, expectedTy, context.checkScope);
|
||||||
|
|
||||||
if (!result.isSubtype)
|
if (!result.isSubtype)
|
||||||
|
{
|
||||||
|
if (FFlag::LuauStringFormatErrorSuppression)
|
||||||
|
{
|
||||||
|
switch (shouldSuppressErrors(NotNull{&context.typechecker->normalizer}, actualTy))
|
||||||
|
{
|
||||||
|
case ErrorSuppression::Suppress:
|
||||||
|
break;
|
||||||
|
case ErrorSuppression::NormalizationFailed:
|
||||||
|
break;
|
||||||
|
case ErrorSuppression::DoNotSuppress:
|
||||||
|
Reasonings reasonings = context.typechecker->explainReasonings(actualTy, expectedTy, location, result);
|
||||||
|
|
||||||
|
if (!reasonings.suppressed)
|
||||||
|
context.typechecker->reportError(TypeMismatch{expectedTy, actualTy, reasonings.toString()}, location);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
Reasonings reasonings = context.typechecker->explainReasonings(actualTy, expectedTy, location, result);
|
Reasonings reasonings = context.typechecker->explainReasonings(actualTy, expectedTy, location, result);
|
||||||
context.typechecker->reportError(TypeMismatch{expectedTy, actualTy, reasonings.toString()}, location);
|
context.typechecker->reportError(TypeMismatch{expectedTy, actualTy, reasonings.toString()}, location);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::vector<TypeId> parsePatternString(NotNull<BuiltinTypes> builtinTypes, const char* data, size_t size)
|
static std::vector<TypeId> parsePatternString(NotNull<BuiltinTypes> builtinTypes, const char* data, size_t size)
|
||||||
|
@ -697,7 +719,7 @@ static std::vector<TypeId> parsePatternString(NotNull<BuiltinTypes> builtinTypes
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionGmatch(
|
std::optional<WithPredicate<TypePackId>> MagicGmatch::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -733,7 +755,7 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionGmatch(
|
||||||
return WithPredicate<TypePackId>{arena.addTypePack({iteratorType})};
|
return WithPredicate<TypePackId>{arena.addTypePack({iteratorType})};
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionGmatch(MagicFunctionCallContext context)
|
bool MagicGmatch::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
const auto& [params, tail] = flatten(context.arguments);
|
const auto& [params, tail] = flatten(context.arguments);
|
||||||
|
|
||||||
|
@ -766,7 +788,7 @@ static bool dcrMagicFunctionGmatch(MagicFunctionCallContext context)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionMatch(
|
std::optional<WithPredicate<TypePackId>> MagicMatch::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -806,7 +828,7 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionMatch(
|
||||||
return WithPredicate<TypePackId>{returnList};
|
return WithPredicate<TypePackId>{returnList};
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionMatch(MagicFunctionCallContext context)
|
bool MagicMatch::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
const auto& [params, tail] = flatten(context.arguments);
|
const auto& [params, tail] = flatten(context.arguments);
|
||||||
|
|
||||||
|
@ -842,7 +864,7 @@ static bool dcrMagicFunctionMatch(MagicFunctionCallContext context)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionFind(
|
std::optional<WithPredicate<TypePackId>> MagicFind::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -900,7 +922,7 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionFind(
|
||||||
return WithPredicate<TypePackId>{returnList};
|
return WithPredicate<TypePackId>{returnList};
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionFind(MagicFunctionCallContext context)
|
bool MagicFind::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
const auto& [params, tail] = flatten(context.arguments);
|
const auto& [params, tail] = flatten(context.arguments);
|
||||||
|
|
||||||
|
@ -977,11 +999,9 @@ TypeId makeStringMetatable(NotNull<BuiltinTypes> builtinTypes)
|
||||||
|
|
||||||
|
|
||||||
FunctionType formatFTV{arena->addTypePack(TypePack{{stringType}, variadicTailPack}), oneStringPack};
|
FunctionType formatFTV{arena->addTypePack(TypePack{{stringType}, variadicTailPack}), oneStringPack};
|
||||||
formatFTV.magicFunction = &magicFunctionFormat;
|
|
||||||
formatFTV.isCheckedFunction = true;
|
formatFTV.isCheckedFunction = true;
|
||||||
const TypeId formatFn = arena->addType(formatFTV);
|
const TypeId formatFn = arena->addType(formatFTV);
|
||||||
attachDcrMagicFunction(formatFn, dcrMagicFunctionFormat);
|
attachMagicFunction(formatFn, std::make_shared<MagicFormat>());
|
||||||
attachDcrMagicFunctionTypeCheck(formatFn, dcrMagicFunctionTypeCheckFormat);
|
|
||||||
|
|
||||||
|
|
||||||
const TypeId stringToStringType = makeFunction(*arena, std::nullopt, {}, {}, {stringType}, {}, {stringType}, /* checked */ true);
|
const TypeId stringToStringType = makeFunction(*arena, std::nullopt, {}, {}, {stringType}, {}, {stringType}, /* checked */ true);
|
||||||
|
@ -995,16 +1015,14 @@ TypeId makeStringMetatable(NotNull<BuiltinTypes> builtinTypes)
|
||||||
makeFunction(*arena, stringType, {}, {}, {stringType, replArgType, optionalNumber}, {}, {stringType, numberType}, /* checked */ false);
|
makeFunction(*arena, stringType, {}, {}, {stringType, replArgType, optionalNumber}, {}, {stringType, numberType}, /* checked */ false);
|
||||||
const TypeId gmatchFunc =
|
const TypeId gmatchFunc =
|
||||||
makeFunction(*arena, stringType, {}, {}, {stringType}, {}, {arena->addType(FunctionType{emptyPack, stringVariadicList})}, /* checked */ true);
|
makeFunction(*arena, stringType, {}, {}, {stringType}, {}, {arena->addType(FunctionType{emptyPack, stringVariadicList})}, /* checked */ true);
|
||||||
attachMagicFunction(gmatchFunc, magicFunctionGmatch);
|
attachMagicFunction(gmatchFunc, std::make_shared<MagicGmatch>());
|
||||||
attachDcrMagicFunction(gmatchFunc, dcrMagicFunctionGmatch);
|
|
||||||
|
|
||||||
FunctionType matchFuncTy{
|
FunctionType matchFuncTy{
|
||||||
arena->addTypePack({stringType, stringType, optionalNumber}), arena->addTypePack(TypePackVar{VariadicTypePack{stringType}})
|
arena->addTypePack({stringType, stringType, optionalNumber}), arena->addTypePack(TypePackVar{VariadicTypePack{stringType}})
|
||||||
};
|
};
|
||||||
matchFuncTy.isCheckedFunction = true;
|
matchFuncTy.isCheckedFunction = true;
|
||||||
const TypeId matchFunc = arena->addType(matchFuncTy);
|
const TypeId matchFunc = arena->addType(matchFuncTy);
|
||||||
attachMagicFunction(matchFunc, magicFunctionMatch);
|
attachMagicFunction(matchFunc, std::make_shared<MagicMatch>());
|
||||||
attachDcrMagicFunction(matchFunc, dcrMagicFunctionMatch);
|
|
||||||
|
|
||||||
FunctionType findFuncTy{
|
FunctionType findFuncTy{
|
||||||
arena->addTypePack({stringType, stringType, optionalNumber, optionalBoolean}),
|
arena->addTypePack({stringType, stringType, optionalNumber, optionalBoolean}),
|
||||||
|
@ -1012,8 +1030,7 @@ TypeId makeStringMetatable(NotNull<BuiltinTypes> builtinTypes)
|
||||||
};
|
};
|
||||||
findFuncTy.isCheckedFunction = true;
|
findFuncTy.isCheckedFunction = true;
|
||||||
const TypeId findFunc = arena->addType(findFuncTy);
|
const TypeId findFunc = arena->addType(findFuncTy);
|
||||||
attachMagicFunction(findFunc, magicFunctionFind);
|
attachMagicFunction(findFunc, std::make_shared<MagicFind>());
|
||||||
attachDcrMagicFunction(findFunc, dcrMagicFunctionFind);
|
|
||||||
|
|
||||||
// string.byte : string -> number? -> number? -> ...number
|
// string.byte : string -> number? -> number? -> ...number
|
||||||
FunctionType stringDotByte{arena->addTypePack({stringType, optionalNumber, optionalNumber}), numberVariadicList};
|
FunctionType stringDotByte{arena->addTypePack({stringType, optionalNumber, optionalNumber}), numberVariadicList};
|
||||||
|
@ -1074,7 +1091,7 @@ TypeId makeStringMetatable(NotNull<BuiltinTypes> builtinTypes)
|
||||||
return arena->addType(TableType{{{{"__index", {tableType}}}}, std::nullopt, TypeLevel{}, TableState::Sealed});
|
return arena->addType(TableType{{{{"__index", {tableType}}}}, std::nullopt, TypeLevel{}, TableState::Sealed});
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionSelect(
|
std::optional<WithPredicate<TypePackId>> MagicSelect::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -1119,7 +1136,7 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionSelect(
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionSelect(MagicFunctionCallContext context)
|
bool MagicSelect::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
if (context.callSite->args.size <= 0)
|
if (context.callSite->args.size <= 0)
|
||||||
{
|
{
|
||||||
|
@ -1164,7 +1181,7 @@ static bool dcrMagicFunctionSelect(MagicFunctionCallContext context)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionSetMetaTable(
|
std::optional<WithPredicate<TypePackId>> MagicSetMetatable::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -1246,7 +1263,12 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionSetMetaTable(
|
||||||
return WithPredicate<TypePackId>{arena.addTypePack({target})};
|
return WithPredicate<TypePackId>{arena.addTypePack({target})};
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionAssert(
|
bool MagicSetMetatable::infer(const MagicFunctionCallContext&)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WithPredicate<TypePackId>> MagicAssert::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -1280,7 +1302,12 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionAssert(
|
||||||
return WithPredicate<TypePackId>{arena.addTypePack(TypePack{std::move(head), tail})};
|
return WithPredicate<TypePackId>{arena.addTypePack(TypePack{std::move(head), tail})};
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionPack(
|
bool MagicAssert::infer(const MagicFunctionCallContext&)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<WithPredicate<TypePackId>> MagicPack::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -1323,7 +1350,7 @@ static std::optional<WithPredicate<TypePackId>> magicFunctionPack(
|
||||||
return WithPredicate<TypePackId>{arena.addTypePack({packedTable})};
|
return WithPredicate<TypePackId>{arena.addTypePack({packedTable})};
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionPack(MagicFunctionCallContext context)
|
bool MagicPack::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
|
|
||||||
TypeArena* arena = context.solver->arena;
|
TypeArena* arena = context.solver->arena;
|
||||||
|
@ -1363,10 +1390,78 @@ static bool dcrMagicFunctionPack(MagicFunctionCallContext context)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<TypeId> freezeTable(TypeId inputType, MagicFunctionCallContext& context)
|
std::optional<WithPredicate<TypePackId>> MagicClone::handleOldSolver(
|
||||||
|
TypeChecker& typechecker,
|
||||||
|
const ScopePtr& scope,
|
||||||
|
const AstExprCall& expr,
|
||||||
|
WithPredicate<TypePackId> withPredicate
|
||||||
|
)
|
||||||
{
|
{
|
||||||
|
LUAU_ASSERT(FFlag::LuauTableCloneClonesType3);
|
||||||
|
|
||||||
|
auto [paramPack, _predicates] = withPredicate;
|
||||||
|
|
||||||
|
TypeArena& arena = typechecker.currentModule->internalTypes;
|
||||||
|
|
||||||
|
const auto& [paramTypes, paramTail] = flatten(paramPack);
|
||||||
|
if (paramTypes.empty() || expr.args.size == 0)
|
||||||
|
{
|
||||||
|
typechecker.reportError(expr.argLocation, CountMismatch{1, std::nullopt, 0});
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId inputType = follow(paramTypes[0]);
|
||||||
|
|
||||||
|
if (!get<TableType>(inputType))
|
||||||
|
return std::nullopt;
|
||||||
|
|
||||||
|
CloneState cloneState{typechecker.builtinTypes};
|
||||||
|
TypeId resultType = shallowClone(inputType, arena, cloneState);
|
||||||
|
|
||||||
|
TypePackId clonedTypePack = arena.addTypePack({resultType});
|
||||||
|
return WithPredicate<TypePackId>{clonedTypePack};
|
||||||
|
}
|
||||||
|
|
||||||
|
bool MagicClone::infer(const MagicFunctionCallContext& context)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(FFlag::LuauTableCloneClonesType3);
|
||||||
|
|
||||||
TypeArena* arena = context.solver->arena;
|
TypeArena* arena = context.solver->arena;
|
||||||
|
|
||||||
|
const auto& [paramTypes, paramTail] = flatten(context.arguments);
|
||||||
|
if (paramTypes.empty() || context.callSite->args.size == 0)
|
||||||
|
{
|
||||||
|
context.solver->reportError(CountMismatch{1, std::nullopt, 0}, context.callSite->argLocation);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId inputType = follow(paramTypes[0]);
|
||||||
|
|
||||||
|
if (!get<TableType>(inputType))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
CloneState cloneState{context.solver->builtinTypes};
|
||||||
|
TypeId resultType = shallowClone(inputType, *arena, cloneState, /* ignorePersistent */ FFlag::LuauFreezeIgnorePersistent);
|
||||||
|
|
||||||
|
if (auto tableType = getMutable<TableType>(resultType))
|
||||||
|
{
|
||||||
|
tableType->scope = context.constraint->scope.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
trackInteriorFreeType(context.constraint->scope.get(), resultType);
|
||||||
|
|
||||||
|
TypePackId clonedTypePack = arena->addTypePack({resultType});
|
||||||
|
asMutable(context.result)->ty.emplace<BoundTypePack>(clonedTypePack);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static std::optional<TypeId> freezeTable(TypeId inputType, const MagicFunctionCallContext& context)
|
||||||
|
{
|
||||||
|
TypeArena* arena = context.solver->arena;
|
||||||
|
if (FFlag::LuauFollowTableFreeze)
|
||||||
|
inputType = follow(inputType);
|
||||||
if (auto mt = get<MetatableType>(inputType))
|
if (auto mt = get<MetatableType>(inputType))
|
||||||
{
|
{
|
||||||
std::optional<TypeId> frozenTable = freezeTable(mt->table, context);
|
std::optional<TypeId> frozenTable = freezeTable(mt->table, context);
|
||||||
|
@ -1383,7 +1478,7 @@ static std::optional<TypeId> freezeTable(TypeId inputType, MagicFunctionCallCont
|
||||||
{
|
{
|
||||||
// Clone the input type, this will become our final result type after we mutate it.
|
// Clone the input type, this will become our final result type after we mutate it.
|
||||||
CloneState cloneState{context.solver->builtinTypes};
|
CloneState cloneState{context.solver->builtinTypes};
|
||||||
TypeId resultType = shallowClone(inputType, *arena, cloneState);
|
TypeId resultType = shallowClone(inputType, *arena, cloneState, /* ignorePersistent */ FFlag::LuauFreezeIgnorePersistent);
|
||||||
auto tableTy = getMutable<TableType>(resultType);
|
auto tableTy = getMutable<TableType>(resultType);
|
||||||
// `clone` should not break this.
|
// `clone` should not break this.
|
||||||
LUAU_ASSERT(tableTy);
|
LUAU_ASSERT(tableTy);
|
||||||
|
@ -1408,10 +1503,13 @@ static std::optional<TypeId> freezeTable(TypeId inputType, MagicFunctionCallCont
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionFreeze(MagicFunctionCallContext context)
|
std::optional<WithPredicate<TypePackId>> MagicFreeze::handleOldSolver(struct TypeChecker &, const std::shared_ptr<struct Scope> &, const class AstExprCall &, WithPredicate<TypePackId>)
|
||||||
{
|
{
|
||||||
LUAU_ASSERT(FFlag::LuauTypestateBuiltins2);
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool MagicFreeze::infer(const MagicFunctionCallContext& context)
|
||||||
|
{
|
||||||
TypeArena* arena = context.solver->arena;
|
TypeArena* arena = context.solver->arena;
|
||||||
const DataFlowGraph* dfg = context.solver->dfg.get();
|
const DataFlowGraph* dfg = context.solver->dfg.get();
|
||||||
Scope* scope = context.constraint->scope.get();
|
Scope* scope = context.constraint->scope.get();
|
||||||
|
@ -1469,7 +1567,7 @@ static bool checkRequirePath(TypeChecker& typechecker, AstExpr* expr)
|
||||||
return good;
|
return good;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::optional<WithPredicate<TypePackId>> magicFunctionRequire(
|
std::optional<WithPredicate<TypePackId>> MagicRequire::handleOldSolver(
|
||||||
TypeChecker& typechecker,
|
TypeChecker& typechecker,
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
const AstExprCall& expr,
|
const AstExprCall& expr,
|
||||||
|
@ -1515,7 +1613,7 @@ static bool checkRequirePathDcr(NotNull<ConstraintSolver> solver, AstExpr* expr)
|
||||||
return good;
|
return good;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool dcrMagicFunctionRequire(MagicFunctionCallContext context)
|
bool MagicRequire::infer(const MagicFunctionCallContext& context)
|
||||||
{
|
{
|
||||||
if (context.callSite->args.size != 1)
|
if (context.callSite->args.size != 1)
|
||||||
{
|
{
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
#include "Luau/Unifiable.h"
|
#include "Luau/Unifiable.h"
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
|
LUAU_FASTFLAG(LuauFreezeIgnorePersistent)
|
||||||
|
|
||||||
// For each `Luau::clone` call, we will clone only up to N amount of types _and_ packs, as controlled by this limit.
|
// For each `Luau::clone` call, we will clone only up to N amount of types _and_ packs, as controlled by this limit.
|
||||||
LUAU_FASTINTVARIABLE(LuauTypeCloneIterationLimit, 100'000)
|
LUAU_FASTINTVARIABLE(LuauTypeCloneIterationLimit, 100'000)
|
||||||
|
@ -38,14 +39,26 @@ class TypeCloner
|
||||||
NotNull<SeenTypes> types;
|
NotNull<SeenTypes> types;
|
||||||
NotNull<SeenTypePacks> packs;
|
NotNull<SeenTypePacks> packs;
|
||||||
|
|
||||||
|
TypeId forceTy = nullptr;
|
||||||
|
TypePackId forceTp = nullptr;
|
||||||
|
|
||||||
int steps = 0;
|
int steps = 0;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
TypeCloner(NotNull<TypeArena> arena, NotNull<BuiltinTypes> builtinTypes, NotNull<SeenTypes> types, NotNull<SeenTypePacks> packs)
|
TypeCloner(
|
||||||
|
NotNull<TypeArena> arena,
|
||||||
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<SeenTypes> types,
|
||||||
|
NotNull<SeenTypePacks> packs,
|
||||||
|
TypeId forceTy,
|
||||||
|
TypePackId forceTp
|
||||||
|
)
|
||||||
: arena(arena)
|
: arena(arena)
|
||||||
, builtinTypes(builtinTypes)
|
, builtinTypes(builtinTypes)
|
||||||
, types(types)
|
, types(types)
|
||||||
, packs(packs)
|
, packs(packs)
|
||||||
|
, forceTy(forceTy)
|
||||||
|
, forceTp(forceTp)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -112,7 +125,7 @@ private:
|
||||||
ty = follow(ty, FollowOption::DisableLazyTypeThunks);
|
ty = follow(ty, FollowOption::DisableLazyTypeThunks);
|
||||||
if (auto it = types->find(ty); it != types->end())
|
if (auto it = types->find(ty); it != types->end())
|
||||||
return it->second;
|
return it->second;
|
||||||
else if (ty->persistent)
|
else if (ty->persistent && (!FFlag::LuauFreezeIgnorePersistent || ty != forceTy))
|
||||||
return ty;
|
return ty;
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
@ -122,7 +135,7 @@ private:
|
||||||
tp = follow(tp);
|
tp = follow(tp);
|
||||||
if (auto it = packs->find(tp); it != packs->end())
|
if (auto it = packs->find(tp); it != packs->end())
|
||||||
return it->second;
|
return it->second;
|
||||||
else if (tp->persistent)
|
else if (tp->persistent && (!FFlag::LuauFreezeIgnorePersistent || tp != forceTp))
|
||||||
return tp;
|
return tp;
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
@ -148,7 +161,7 @@ public:
|
||||||
|
|
||||||
if (auto clone = find(ty))
|
if (auto clone = find(ty))
|
||||||
return *clone;
|
return *clone;
|
||||||
else if (ty->persistent)
|
else if (ty->persistent && (!FFlag::LuauFreezeIgnorePersistent || ty != forceTy))
|
||||||
return ty;
|
return ty;
|
||||||
|
|
||||||
TypeId target = arena->addType(ty->ty);
|
TypeId target = arena->addType(ty->ty);
|
||||||
|
@ -174,7 +187,7 @@ public:
|
||||||
|
|
||||||
if (auto clone = find(tp))
|
if (auto clone = find(tp))
|
||||||
return *clone;
|
return *clone;
|
||||||
else if (tp->persistent)
|
else if (tp->persistent && (!FFlag::LuauFreezeIgnorePersistent || tp != forceTp))
|
||||||
return tp;
|
return tp;
|
||||||
|
|
||||||
TypePackId target = arena->addTypePack(tp->ty);
|
TypePackId target = arena->addTypePack(tp->ty);
|
||||||
|
@ -458,21 +471,37 @@ private:
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
TypePackId shallowClone(TypePackId tp, TypeArena& dest, CloneState& cloneState)
|
TypePackId shallowClone(TypePackId tp, TypeArena& dest, CloneState& cloneState, bool ignorePersistent)
|
||||||
{
|
{
|
||||||
if (tp->persistent)
|
if (tp->persistent && (!FFlag::LuauFreezeIgnorePersistent || !ignorePersistent))
|
||||||
return tp;
|
return tp;
|
||||||
|
|
||||||
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}};
|
TypeCloner cloner{
|
||||||
|
NotNull{&dest},
|
||||||
|
cloneState.builtinTypes,
|
||||||
|
NotNull{&cloneState.seenTypes},
|
||||||
|
NotNull{&cloneState.seenTypePacks},
|
||||||
|
nullptr,
|
||||||
|
FFlag::LuauFreezeIgnorePersistent && ignorePersistent ? tp : nullptr
|
||||||
|
};
|
||||||
|
|
||||||
return cloner.shallowClone(tp);
|
return cloner.shallowClone(tp);
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId shallowClone(TypeId typeId, TypeArena& dest, CloneState& cloneState)
|
TypeId shallowClone(TypeId typeId, TypeArena& dest, CloneState& cloneState, bool ignorePersistent)
|
||||||
{
|
{
|
||||||
if (typeId->persistent)
|
if (typeId->persistent && (!FFlag::LuauFreezeIgnorePersistent || !ignorePersistent))
|
||||||
return typeId;
|
return typeId;
|
||||||
|
|
||||||
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}};
|
TypeCloner cloner{
|
||||||
|
NotNull{&dest},
|
||||||
|
cloneState.builtinTypes,
|
||||||
|
NotNull{&cloneState.seenTypes},
|
||||||
|
NotNull{&cloneState.seenTypePacks},
|
||||||
|
FFlag::LuauFreezeIgnorePersistent && ignorePersistent ? typeId : nullptr,
|
||||||
|
nullptr
|
||||||
|
};
|
||||||
|
|
||||||
return cloner.shallowClone(typeId);
|
return cloner.shallowClone(typeId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -481,7 +510,7 @@ TypePackId clone(TypePackId tp, TypeArena& dest, CloneState& cloneState)
|
||||||
if (tp->persistent)
|
if (tp->persistent)
|
||||||
return tp;
|
return tp;
|
||||||
|
|
||||||
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}};
|
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}, nullptr, nullptr};
|
||||||
return cloner.clone(tp);
|
return cloner.clone(tp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -490,13 +519,13 @@ TypeId clone(TypeId typeId, TypeArena& dest, CloneState& cloneState)
|
||||||
if (typeId->persistent)
|
if (typeId->persistent)
|
||||||
return typeId;
|
return typeId;
|
||||||
|
|
||||||
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}};
|
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}, nullptr, nullptr};
|
||||||
return cloner.clone(typeId);
|
return cloner.clone(typeId);
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeFun clone(const TypeFun& typeFun, TypeArena& dest, CloneState& cloneState)
|
TypeFun clone(const TypeFun& typeFun, TypeArena& dest, CloneState& cloneState)
|
||||||
{
|
{
|
||||||
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}};
|
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}, nullptr, nullptr};
|
||||||
|
|
||||||
TypeFun copy = typeFun;
|
TypeFun copy = typeFun;
|
||||||
|
|
||||||
|
@ -521,4 +550,18 @@ TypeFun clone(const TypeFun& typeFun, TypeArena& dest, CloneState& cloneState)
|
||||||
return copy;
|
return copy;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Binding clone(const Binding& binding, TypeArena& dest, CloneState& cloneState)
|
||||||
|
{
|
||||||
|
TypeCloner cloner{NotNull{&dest}, cloneState.builtinTypes, NotNull{&cloneState.seenTypes}, NotNull{&cloneState.seenTypePacks}, nullptr, nullptr};
|
||||||
|
|
||||||
|
Binding b;
|
||||||
|
b.deprecated = binding.deprecated;
|
||||||
|
b.deprecatedSuggestion = binding.deprecatedSuggestion;
|
||||||
|
b.documentationSymbol = binding.documentationSymbol;
|
||||||
|
b.location = binding.location;
|
||||||
|
b.typeId = cloner.clone(binding.typeId);
|
||||||
|
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Luau
|
} // namespace Luau
|
||||||
|
|
|
@ -3,8 +3,6 @@
|
||||||
#include "Luau/Constraint.h"
|
#include "Luau/Constraint.h"
|
||||||
#include "Luau/VisitType.h"
|
#include "Luau/VisitType.h"
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauDontRefCountTypesInTypeFunctions)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -60,9 +58,8 @@ struct ReferenceCountInitializer : TypeOnceVisitor
|
||||||
//
|
//
|
||||||
// The default behavior here is `true` for "visit the child types"
|
// The default behavior here is `true` for "visit the child types"
|
||||||
// of this type, hence:
|
// of this type, hence:
|
||||||
return !FFlag::LuauDontRefCountTypesInTypeFunctions;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
bool isReferenceCountedType(const TypeId typ)
|
bool isReferenceCountedType(const TypeId typ)
|
||||||
|
|
|
@ -31,17 +31,14 @@
|
||||||
LUAU_FASTINT(LuauCheckRecursionLimit)
|
LUAU_FASTINT(LuauCheckRecursionLimit)
|
||||||
LUAU_FASTFLAG(DebugLuauLogSolverToJson)
|
LUAU_FASTFLAG(DebugLuauLogSolverToJson)
|
||||||
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
||||||
LUAU_FASTFLAG(DebugLuauEqSatSimplification)
|
LUAU_FASTFLAG(LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
LUAU_FASTFLAG(LuauTypestateBuiltins2)
|
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunUpdateAllEnvs)
|
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverVisitErrorExprLvalues)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPrePopulateClasses)
|
LUAU_FASTFLAGVARIABLE(LuauNewSolverPrePopulateClasses)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunExportedAndLocal)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPopulateTableLocations)
|
LUAU_FASTFLAGVARIABLE(LuauNewSolverPopulateTableLocations)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunNoExtraConstraint)
|
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(InferGlobalTypes)
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauInferLocalTypesInMultipleAssignments)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -233,8 +230,17 @@ void ConstraintGenerator::visitModuleRoot(AstStatBlock* block)
|
||||||
Checkpoint end = checkpoint(this);
|
Checkpoint end = checkpoint(this);
|
||||||
|
|
||||||
TypeId result = arena->addType(BlockedType{});
|
TypeId result = arena->addType(BlockedType{});
|
||||||
NotNull<Constraint> genConstraint =
|
NotNull<Constraint> genConstraint = addConstraint(
|
||||||
addConstraint(scope, block->location, GeneralizationConstraint{result, moduleFnTy, std::move(interiorTypes.back())});
|
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);
|
getMutable<BlockedType>(result)->setOwner(genConstraint);
|
||||||
forEachConstraint(
|
forEachConstraint(
|
||||||
start,
|
start,
|
||||||
|
@ -303,9 +309,19 @@ void ConstraintGenerator::visitFragmentRoot(const ScopePtr& resumeScope, AstStat
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
TypeId ConstraintGenerator::freshType(const ScopePtr& scope)
|
TypeId ConstraintGenerator::freshType(const ScopePtr& scope)
|
||||||
{
|
{
|
||||||
|
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());
|
return Luau::freshType(arena, builtinTypes, scope.get());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePackId ConstraintGenerator::freshTypePack(const ScopePtr& scope)
|
TypePackId ConstraintGenerator::freshTypePack(const ScopePtr& scope)
|
||||||
|
@ -724,12 +740,6 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!FFlag::LuauUserTypeFunExportedAndLocal && scope->parent != globalScope)
|
|
||||||
{
|
|
||||||
reportError(function->location, GenericError{"Local user-defined functions are not supported yet"});
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
ScopePtr defnScope = childScope(function, scope);
|
ScopePtr defnScope = childScope(function, scope);
|
||||||
|
|
||||||
// Create TypeFunctionInstanceType
|
// Create TypeFunctionInstanceType
|
||||||
|
@ -755,11 +765,8 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
|
|
||||||
UserDefinedFunctionData udtfData;
|
UserDefinedFunctionData udtfData;
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
udtfData.owner = module;
|
udtfData.owner = module;
|
||||||
udtfData.definition = function;
|
udtfData.definition = function;
|
||||||
}
|
|
||||||
|
|
||||||
TypeId typeFunctionTy = arena->addType(
|
TypeId typeFunctionTy = arena->addType(
|
||||||
TypeFunctionInstanceType{NotNull{&builtinTypeFunctions().userFunc}, std::move(typeParams), {}, function->name, udtfData}
|
TypeFunctionInstanceType{NotNull{&builtinTypeFunctions().userFunc}, std::move(typeParams), {}, function->name, udtfData}
|
||||||
|
@ -768,7 +775,7 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
TypeFun typeFunction{std::move(quantifiedTypeParams), typeFunctionTy};
|
TypeFun typeFunction{std::move(quantifiedTypeParams), typeFunctionTy};
|
||||||
|
|
||||||
// Set type bindings and definition locations for this user-defined type function
|
// Set type bindings and definition locations for this user-defined type function
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal && function->exported)
|
if (function->exported)
|
||||||
scope->exportedTypeBindings[function->name.value] = std::move(typeFunction);
|
scope->exportedTypeBindings[function->name.value] = std::move(typeFunction);
|
||||||
else
|
else
|
||||||
scope->privateTypeBindings[function->name.value] = std::move(typeFunction);
|
scope->privateTypeBindings[function->name.value] = std::move(typeFunction);
|
||||||
|
@ -803,8 +810,6 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
// Additional pass for user-defined type functions to fill in their environments completely
|
// Additional pass for user-defined type functions to fill in their environments completely
|
||||||
for (AstStat* stat : block->body)
|
for (AstStat* stat : block->body)
|
||||||
{
|
{
|
||||||
|
@ -823,7 +828,7 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fill it with all visible type functions
|
// Fill it with all visible type functions
|
||||||
if (FFlag::LuauUserTypeFunUpdateAllEnvs && mainTypeFun)
|
if (mainTypeFun)
|
||||||
{
|
{
|
||||||
UserDefinedFunctionData& userFuncData = mainTypeFun->userFuncData;
|
UserDefinedFunctionData& userFuncData = mainTypeFun->userFuncData;
|
||||||
size_t level = 0;
|
size_t level = 0;
|
||||||
|
@ -878,7 +883,6 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ControlFlow ConstraintGenerator::visitBlockWithoutChildScope(const ScopePtr& scope, AstStatBlock* block)
|
ControlFlow ConstraintGenerator::visitBlockWithoutChildScope(const ScopePtr& scope, AstStatBlock* block)
|
||||||
|
@ -1021,6 +1025,66 @@ ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatLocal* stat
|
||||||
TypePackId rvaluePack = checkPack(scope, statLocal->values, expectedTypes).tp;
|
TypePackId rvaluePack = checkPack(scope, statLocal->values, expectedTypes).tp;
|
||||||
Checkpoint end = checkpoint(this);
|
Checkpoint end = checkpoint(this);
|
||||||
|
|
||||||
|
if (FFlag::LuauInferLocalTypesInMultipleAssignments)
|
||||||
|
{
|
||||||
|
std::vector<TypeId> deferredTypes;
|
||||||
|
auto [head, tail] = flatten(rvaluePack);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < statLocal->vars.size; ++i)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(get<BlockedType>(assignees[i]));
|
||||||
|
TypeIds* localDomain = localTypes.find(assignees[i]);
|
||||||
|
LUAU_ASSERT(localDomain);
|
||||||
|
|
||||||
|
if (statLocal->vars.data[i]->annotation)
|
||||||
|
{
|
||||||
|
localDomain->insert(annotatedTypes[i]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (i < head.size())
|
||||||
|
{
|
||||||
|
localDomain->insert(head[i]);
|
||||||
|
}
|
||||||
|
else if (tail)
|
||||||
|
{
|
||||||
|
deferredTypes.push_back(arena->addType(BlockedType{}));
|
||||||
|
localDomain->insert(deferredTypes.back());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
localDomain->insert(builtinTypes->nilType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasAnnotation)
|
||||||
|
{
|
||||||
|
TypePackId annotatedPack = arena->addTypePack(std::move(annotatedTypes));
|
||||||
|
addConstraint(scope, statLocal->location, PackSubtypeConstraint{rvaluePack, annotatedPack});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!deferredTypes.empty())
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(tail);
|
||||||
|
NotNull<Constraint> uc = addConstraint(scope, statLocal->location, UnpackConstraint{deferredTypes, *tail});
|
||||||
|
|
||||||
|
forEachConstraint(
|
||||||
|
start,
|
||||||
|
end,
|
||||||
|
this,
|
||||||
|
[&uc](const ConstraintPtr& runBefore)
|
||||||
|
{
|
||||||
|
uc->dependencies.emplace_back(runBefore.get());
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
for (TypeId t : deferredTypes)
|
||||||
|
getMutable<BlockedType>(t)->setOwner(uc);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (hasAnnotation)
|
if (hasAnnotation)
|
||||||
{
|
{
|
||||||
for (size_t i = 0; i < statLocal->vars.size; ++i)
|
for (size_t i = 0; i < statLocal->vars.size; ++i)
|
||||||
|
@ -1075,6 +1139,7 @@ ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatLocal* stat
|
||||||
localDomain->insert(valueTypes[i]);
|
localDomain->insert(valueTypes[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (statLocal->vars.size == 1 && statLocal->values.size == 1 && firstValueType && scope.get() == rootScope && !hasAnnotation)
|
if (statLocal->vars.size == 1 && statLocal->values.size == 1 && firstValueType && scope.get() == rootScope && !hasAnnotation)
|
||||||
{
|
{
|
||||||
|
@ -1084,20 +1149,10 @@ ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatLocal* stat
|
||||||
if (value->is<AstExprTable>())
|
if (value->is<AstExprTable>())
|
||||||
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
||||||
else if (const AstExprCall* call = value->as<AstExprCall>())
|
else if (const AstExprCall* call = value->as<AstExprCall>())
|
||||||
{
|
|
||||||
if (FFlag::LuauTypestateBuiltins2)
|
|
||||||
{
|
{
|
||||||
if (matchSetMetatable(*call))
|
if (matchSetMetatable(*call))
|
||||||
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
if (const AstExprGlobal* global = call->func->as<AstExprGlobal>(); global && global->name == "setmetatable")
|
|
||||||
{
|
|
||||||
addConstraint(scope, value->location, NameConstraint{*firstValueType, var->name.value, /*synthetic*/ true});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (statLocal->values.size > 0)
|
if (statLocal->values.size > 0)
|
||||||
|
@ -1603,24 +1658,6 @@ ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatTypeAlias*
|
||||||
|
|
||||||
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatTypeFunction* function)
|
ControlFlow ConstraintGenerator::visit(const ScopePtr& scope, AstStatTypeFunction* function)
|
||||||
{
|
{
|
||||||
if (!FFlag::LuauUserTypeFunNoExtraConstraint)
|
|
||||||
{
|
|
||||||
// If a type function with the same name was already defined, we skip over
|
|
||||||
auto bindingIt = scope->privateTypeBindings.find(function->name.value);
|
|
||||||
if (bindingIt == scope->privateTypeBindings.end())
|
|
||||||
return ControlFlow::None;
|
|
||||||
|
|
||||||
TypeFun typeFunction = bindingIt->second;
|
|
||||||
|
|
||||||
// Adding typeAliasExpansionConstraint on user-defined type function for the constraint solver
|
|
||||||
if (auto typeFunctionTy = get<TypeFunctionInstanceType>(follow(typeFunction.type)))
|
|
||||||
{
|
|
||||||
TypeId expansionTy =
|
|
||||||
arena->addType(PendingExpansionType{{}, function->name, typeFunctionTy->typeArguments, typeFunctionTy->packArguments});
|
|
||||||
addConstraint(scope, function->location, TypeAliasExpansionConstraint{/* target */ expansionTy});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return ControlFlow::None;
|
return ControlFlow::None;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2083,7 +2120,7 @@ InferencePack ConstraintGenerator::checkPack(const ScopePtr& scope, AstExprCall*
|
||||||
return InferencePack{arena->addTypePack({resultTy}), {refinementArena.variadic(returnRefinements)}};
|
return InferencePack{arena->addTypePack({resultTy}), {refinementArena.variadic(returnRefinements)}};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::LuauTypestateBuiltins2 && shouldTypestateForFirstArgument(*call) && call->args.size > 0 && isLValue(call->args.data[0]))
|
if (shouldTypestateForFirstArgument(*call) && call->args.size > 0 && isLValue(call->args.data[0]))
|
||||||
{
|
{
|
||||||
AstExpr* targetExpr = call->args.data[0];
|
AstExpr* targetExpr = call->args.data[0];
|
||||||
auto resultTy = arena->addType(BlockedType{});
|
auto resultTy = arena->addType(BlockedType{});
|
||||||
|
@ -2232,7 +2269,8 @@ Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprConstantStrin
|
||||||
if (forceSingleton)
|
if (forceSingleton)
|
||||||
return Inference{arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}})};
|
return Inference{arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}})};
|
||||||
|
|
||||||
FreeType ft = FreeType{scope.get()};
|
FreeType ft =
|
||||||
|
FFlag::LuauFreeTypesMustHaveBounds ? FreeType{scope.get(), builtinTypes->neverType, builtinTypes->unknownType} : FreeType{scope.get()};
|
||||||
ft.lowerBound = arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}});
|
ft.lowerBound = arena->addType(SingletonType{StringSingleton{std::string{string->value.data, string->value.size}}});
|
||||||
ft.upperBound = builtinTypes->stringType;
|
ft.upperBound = builtinTypes->stringType;
|
||||||
const TypeId freeTy = arena->addType(ft);
|
const TypeId freeTy = arena->addType(ft);
|
||||||
|
@ -2246,7 +2284,8 @@ Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprConstantBool*
|
||||||
if (forceSingleton)
|
if (forceSingleton)
|
||||||
return Inference{singletonType};
|
return Inference{singletonType};
|
||||||
|
|
||||||
FreeType ft = FreeType{scope.get()};
|
FreeType ft =
|
||||||
|
FFlag::LuauFreeTypesMustHaveBounds ? FreeType{scope.get(), builtinTypes->neverType, builtinTypes->unknownType} : FreeType{scope.get()};
|
||||||
ft.lowerBound = singletonType;
|
ft.lowerBound = singletonType;
|
||||||
ft.upperBound = builtinTypes->booleanType;
|
ft.upperBound = builtinTypes->booleanType;
|
||||||
const TypeId freeTy = arena->addType(ft);
|
const TypeId freeTy = arena->addType(ft);
|
||||||
|
@ -2408,8 +2447,17 @@ Inference ConstraintGenerator::check(const ScopePtr& scope, AstExprFunction* fun
|
||||||
Checkpoint endCheckpoint = checkpoint(this);
|
Checkpoint endCheckpoint = checkpoint(this);
|
||||||
|
|
||||||
TypeId generalizedTy = arena->addType(BlockedType{});
|
TypeId generalizedTy = arena->addType(BlockedType{});
|
||||||
NotNull<Constraint> gc =
|
NotNull<Constraint> gc = addConstraint(
|
||||||
addConstraint(sig.signatureScope, func->location, GeneralizationConstraint{generalizedTy, sig.signature, std::move(interiorTypes.back())});
|
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);
|
getMutable<BlockedType>(generalizedTy)->setOwner(gc);
|
||||||
interiorTypes.pop_back();
|
interiorTypes.pop_back();
|
||||||
|
|
||||||
|
@ -2756,8 +2804,6 @@ void ConstraintGenerator::visitLValue(const ScopePtr& scope, AstExpr* expr, Type
|
||||||
else if (auto e = expr->as<AstExprIndexExpr>())
|
else if (auto e = expr->as<AstExprIndexExpr>())
|
||||||
visitLValue(scope, e, rhsType);
|
visitLValue(scope, e, rhsType);
|
||||||
else if (auto e = expr->as<AstExprError>())
|
else if (auto e = expr->as<AstExprError>())
|
||||||
{
|
|
||||||
if (FFlag::LuauNewSolverVisitErrorExprLvalues)
|
|
||||||
{
|
{
|
||||||
// If we end up with some sort of error expression in an lvalue
|
// If we end up with some sort of error expression in an lvalue
|
||||||
// position, at least go and check the expressions so that when
|
// position, at least go and check the expressions so that when
|
||||||
|
@ -2767,7 +2813,6 @@ void ConstraintGenerator::visitLValue(const ScopePtr& scope, AstExpr* expr, Type
|
||||||
check(scope, subExpr);
|
check(scope, subExpr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
else
|
||||||
ice->ice("Unexpected lvalue expression", expr->location);
|
ice->ice("Unexpected lvalue expression", expr->location);
|
||||||
}
|
}
|
||||||
|
@ -2826,13 +2871,10 @@ void ConstraintGenerator::visitLValue(const ScopePtr& scope, AstExprGlobal* glob
|
||||||
DefId def = dfg->getDef(global);
|
DefId def = dfg->getDef(global);
|
||||||
rootScope->lvalueTypes[def] = rhsType;
|
rootScope->lvalueTypes[def] = rhsType;
|
||||||
|
|
||||||
if (FFlag::InferGlobalTypes)
|
|
||||||
{
|
|
||||||
// Sketchy: We're specifically looking for BlockedTypes that were
|
// Sketchy: We're specifically looking for BlockedTypes that were
|
||||||
// initially created by ConstraintGenerator::prepopulateGlobalScope.
|
// initially created by ConstraintGenerator::prepopulateGlobalScope.
|
||||||
if (auto bt = get<BlockedType>(follow(*annotatedTy)); bt && !bt->getOwner())
|
if (auto bt = get<BlockedType>(follow(*annotatedTy)); bt && !bt->getOwner())
|
||||||
emplaceType<BoundType>(asMutable(*annotatedTy), rhsType);
|
emplaceType<BoundType>(asMutable(*annotatedTy), rhsType);
|
||||||
}
|
|
||||||
|
|
||||||
addConstraint(scope, global->location, SubtypeConstraint{rhsType, *annotatedTy});
|
addConstraint(scope, global->location, SubtypeConstraint{rhsType, *annotatedTy});
|
||||||
}
|
}
|
||||||
|
@ -3227,8 +3269,7 @@ TypeId ConstraintGenerator::resolveReferenceType(
|
||||||
if (alias.has_value())
|
if (alias.has_value())
|
||||||
{
|
{
|
||||||
// If the alias is not generic, we don't need to set up a blocked type and an instantiation constraint
|
// If the alias is not generic, we don't need to set up a blocked type and an instantiation constraint
|
||||||
if (alias.has_value() && alias->typeParams.empty() && alias->typePackParams.empty() &&
|
if (alias.has_value() && alias->typeParams.empty() && alias->typePackParams.empty() && !ref->hasParameterList)
|
||||||
(!FFlag::LuauUserTypeFunNoExtraConstraint || !ref->hasParameterList))
|
|
||||||
{
|
{
|
||||||
result = alias->type;
|
result = alias->type;
|
||||||
}
|
}
|
||||||
|
@ -3437,6 +3478,12 @@ TypeId ConstraintGenerator::resolveType(const ScopePtr& scope, AstType* ty, bool
|
||||||
}
|
}
|
||||||
else if (auto unionAnnotation = ty->as<AstTypeUnion>())
|
else if (auto unionAnnotation = ty->as<AstTypeUnion>())
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
|
{
|
||||||
|
if (unionAnnotation->types.size == 1)
|
||||||
|
return resolveType(scope, unionAnnotation->types.data[0], inTypeArguments);
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<TypeId> parts;
|
std::vector<TypeId> parts;
|
||||||
for (AstType* part : unionAnnotation->types)
|
for (AstType* part : unionAnnotation->types)
|
||||||
{
|
{
|
||||||
|
@ -3447,6 +3494,12 @@ TypeId ConstraintGenerator::resolveType(const ScopePtr& scope, AstType* ty, bool
|
||||||
}
|
}
|
||||||
else if (auto intersectionAnnotation = ty->as<AstTypeIntersection>())
|
else if (auto intersectionAnnotation = ty->as<AstTypeIntersection>())
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
|
{
|
||||||
|
if (intersectionAnnotation->types.size == 1)
|
||||||
|
return resolveType(scope, intersectionAnnotation->types.data[0], inTypeArguments);
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<TypeId> parts;
|
std::vector<TypeId> parts;
|
||||||
for (AstType* part : intersectionAnnotation->types)
|
for (AstType* part : intersectionAnnotation->types)
|
||||||
{
|
{
|
||||||
|
@ -3455,6 +3508,10 @@ TypeId ConstraintGenerator::resolveType(const ScopePtr& scope, AstType* ty, bool
|
||||||
|
|
||||||
result = arena->addType(IntersectionType{parts});
|
result = arena->addType(IntersectionType{parts});
|
||||||
}
|
}
|
||||||
|
else if (auto typeGroupAnnotation = ty->as<AstTypeGroup>())
|
||||||
|
{
|
||||||
|
result = resolveType(scope, typeGroupAnnotation->type, inTypeArguments);
|
||||||
|
}
|
||||||
else if (auto boolAnnotation = ty->as<AstTypeSingletonBool>())
|
else if (auto boolAnnotation = ty->as<AstTypeSingletonBool>())
|
||||||
{
|
{
|
||||||
if (boolAnnotation->value)
|
if (boolAnnotation->value)
|
||||||
|
@ -3536,33 +3593,34 @@ TypePackId ConstraintGenerator::resolveTypePack(const ScopePtr& scope, const Ast
|
||||||
|
|
||||||
std::vector<std::pair<Name, GenericTypeDefinition>> ConstraintGenerator::createGenerics(
|
std::vector<std::pair<Name, GenericTypeDefinition>> ConstraintGenerator::createGenerics(
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
AstArray<AstGenericType> generics,
|
AstArray<AstGenericType*> generics,
|
||||||
bool useCache,
|
bool useCache,
|
||||||
bool addTypes
|
bool addTypes
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
std::vector<std::pair<Name, GenericTypeDefinition>> result;
|
std::vector<std::pair<Name, GenericTypeDefinition>> result;
|
||||||
for (const auto& generic : generics)
|
for (const auto* generic : generics)
|
||||||
{
|
{
|
||||||
TypeId genericTy = nullptr;
|
TypeId genericTy = nullptr;
|
||||||
|
|
||||||
if (auto it = scope->parent->typeAliasTypeParameters.find(generic.name.value); useCache && it != scope->parent->typeAliasTypeParameters.end())
|
if (auto it = scope->parent->typeAliasTypeParameters.find(generic->name.value);
|
||||||
|
useCache && it != scope->parent->typeAliasTypeParameters.end())
|
||||||
genericTy = it->second;
|
genericTy = it->second;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
genericTy = arena->addType(GenericType{scope.get(), generic.name.value});
|
genericTy = arena->addType(GenericType{scope.get(), generic->name.value});
|
||||||
scope->parent->typeAliasTypeParameters[generic.name.value] = genericTy;
|
scope->parent->typeAliasTypeParameters[generic->name.value] = genericTy;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<TypeId> defaultTy = std::nullopt;
|
std::optional<TypeId> defaultTy = std::nullopt;
|
||||||
|
|
||||||
if (generic.defaultValue)
|
if (generic->defaultValue)
|
||||||
defaultTy = resolveType(scope, generic.defaultValue, /* inTypeArguments */ false);
|
defaultTy = resolveType(scope, generic->defaultValue, /* inTypeArguments */ false);
|
||||||
|
|
||||||
if (addTypes)
|
if (addTypes)
|
||||||
scope->privateTypeBindings[generic.name.value] = TypeFun{genericTy};
|
scope->privateTypeBindings[generic->name.value] = TypeFun{genericTy};
|
||||||
|
|
||||||
result.push_back({generic.name.value, GenericTypeDefinition{genericTy, defaultTy}});
|
result.emplace_back(generic->name.value, GenericTypeDefinition{genericTy, defaultTy});
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
@ -3570,34 +3628,34 @@ std::vector<std::pair<Name, GenericTypeDefinition>> ConstraintGenerator::createG
|
||||||
|
|
||||||
std::vector<std::pair<Name, GenericTypePackDefinition>> ConstraintGenerator::createGenericPacks(
|
std::vector<std::pair<Name, GenericTypePackDefinition>> ConstraintGenerator::createGenericPacks(
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
AstArray<AstGenericTypePack> generics,
|
AstArray<AstGenericTypePack*> generics,
|
||||||
bool useCache,
|
bool useCache,
|
||||||
bool addTypes
|
bool addTypes
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
std::vector<std::pair<Name, GenericTypePackDefinition>> result;
|
std::vector<std::pair<Name, GenericTypePackDefinition>> result;
|
||||||
for (const auto& generic : generics)
|
for (const auto* generic : generics)
|
||||||
{
|
{
|
||||||
TypePackId genericTy;
|
TypePackId genericTy;
|
||||||
|
|
||||||
if (auto it = scope->parent->typeAliasTypePackParameters.find(generic.name.value);
|
if (auto it = scope->parent->typeAliasTypePackParameters.find(generic->name.value);
|
||||||
useCache && it != scope->parent->typeAliasTypePackParameters.end())
|
useCache && it != scope->parent->typeAliasTypePackParameters.end())
|
||||||
genericTy = it->second;
|
genericTy = it->second;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
genericTy = arena->addTypePack(TypePackVar{GenericTypePack{scope.get(), generic.name.value}});
|
genericTy = arena->addTypePack(TypePackVar{GenericTypePack{scope.get(), generic->name.value}});
|
||||||
scope->parent->typeAliasTypePackParameters[generic.name.value] = genericTy;
|
scope->parent->typeAliasTypePackParameters[generic->name.value] = genericTy;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<TypePackId> defaultTy = std::nullopt;
|
std::optional<TypePackId> defaultTy = std::nullopt;
|
||||||
|
|
||||||
if (generic.defaultValue)
|
if (generic->defaultValue)
|
||||||
defaultTy = resolveTypePack(scope, generic.defaultValue, /* inTypeArguments */ false);
|
defaultTy = resolveTypePack(scope, generic->defaultValue, /* inTypeArguments */ false);
|
||||||
|
|
||||||
if (addTypes)
|
if (addTypes)
|
||||||
scope->privateTypePackBindings[generic.name.value] = genericTy;
|
scope->privateTypePackBindings[generic->name.value] = genericTy;
|
||||||
|
|
||||||
result.push_back({generic.name.value, GenericTypePackDefinition{genericTy, defaultTy}});
|
result.emplace_back(generic->name.value, GenericTypePackDefinition{genericTy, defaultTy});
|
||||||
}
|
}
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
@ -3739,8 +3797,6 @@ struct GlobalPrepopulator : AstVisitor
|
||||||
}
|
}
|
||||||
|
|
||||||
bool visit(AstStatAssign* assign) override
|
bool visit(AstStatAssign* assign) override
|
||||||
{
|
|
||||||
if (FFlag::InferGlobalTypes)
|
|
||||||
{
|
{
|
||||||
for (const Luau::AstExpr* expr : assign->vars)
|
for (const Luau::AstExpr* expr : assign->vars)
|
||||||
{
|
{
|
||||||
|
@ -3753,7 +3809,6 @@ struct GlobalPrepopulator : AstVisitor
|
||||||
globalScope->bindings[g->name] = Binding{bt, g->location};
|
globalScope->bindings[g->name] = Binding{bt, g->location};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -3941,19 +3996,6 @@ TypeId ConstraintGenerator::createTypeFunctionInstance(
|
||||||
|
|
||||||
TypeId ConstraintGenerator::simplifyUnion(const ScopePtr& scope, Location location, TypeId left, TypeId right)
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -31,11 +31,13 @@ LUAU_FASTFLAGVARIABLE(DebugLuauLogSolver)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauLogSolverIncludeDependencies)
|
LUAU_FASTFLAGVARIABLE(DebugLuauLogSolverIncludeDependencies)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauLogBindings)
|
LUAU_FASTFLAGVARIABLE(DebugLuauLogBindings)
|
||||||
LUAU_FASTINTVARIABLE(LuauSolverRecursionLimit, 500)
|
LUAU_FASTINTVARIABLE(LuauSolverRecursionLimit, 500)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauRemoveNotAnyHack)
|
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauEqSatSimplification)
|
LUAU_FASTFLAGVARIABLE(DebugLuauEqSatSimplification)
|
||||||
LUAU_FASTFLAG(LuauNewSolverPopulateTableLocations)
|
LUAU_FASTFLAG(LuauNewSolverPopulateTableLocations)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauAllowNilAssignmentToIndexer)
|
LUAU_FASTFLAGVARIABLE(LuauAllowNilAssignmentToIndexer)
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunNoExtraConstraint)
|
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauAlwaysFillInFunctionCallDiscriminantTypes)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTablesOnScope)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauPrecalculateMutatedFreeTypes)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -73,7 +75,7 @@ size_t HashBlockedConstraintId::operator()(const BlockedConstraintId& bci) const
|
||||||
{
|
{
|
||||||
if (auto blocked = get<BlockedType>(ty))
|
if (auto blocked = get<BlockedType>(ty))
|
||||||
{
|
{
|
||||||
Constraint* owner = blocked->getOwner();
|
const Constraint* owner = blocked->getOwner();
|
||||||
LUAU_ASSERT(owner);
|
LUAU_ASSERT(owner);
|
||||||
return owner == constraint;
|
return owner == constraint;
|
||||||
}
|
}
|
||||||
|
@ -437,6 +439,10 @@ void ConstraintSolver::run()
|
||||||
snapshot = logger->prepareStepSnapshot(rootScope, c, force, unsolvedConstraints);
|
snapshot = logger->prepareStepSnapshot(rootScope, c, force, unsolvedConstraints);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::optional<DenseHashSet<TypeId>> mutatedFreeTypes = std::nullopt;
|
||||||
|
if (FFlag::LuauPrecalculateMutatedFreeTypes)
|
||||||
|
mutatedFreeTypes = c->getMaybeMutatedFreeTypes();
|
||||||
|
|
||||||
bool success = tryDispatch(c, force);
|
bool success = tryDispatch(c, force);
|
||||||
|
|
||||||
progress |= success;
|
progress |= success;
|
||||||
|
@ -444,8 +450,29 @@ void ConstraintSolver::run()
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
unblock(c);
|
unblock(c);
|
||||||
unsolvedConstraints.erase(unsolvedConstraints.begin() + i);
|
unsolvedConstraints.erase(unsolvedConstraints.begin() + ptrdiff_t(i));
|
||||||
|
|
||||||
|
if (FFlag::LuauPrecalculateMutatedFreeTypes)
|
||||||
|
{
|
||||||
|
for (auto ty : c->getMaybeMutatedFreeTypes())
|
||||||
|
mutatedFreeTypes->insert(ty);
|
||||||
|
for (auto ty : *mutatedFreeTypes)
|
||||||
|
{
|
||||||
|
size_t& refCount = unresolvedConstraints[ty];
|
||||||
|
if (refCount > 0)
|
||||||
|
refCount -= 1;
|
||||||
|
|
||||||
|
// We have two constraints that are designed to wait for the
|
||||||
|
// refCount on a free type to be equal to 1: the
|
||||||
|
// PrimitiveTypeConstraint and ReduceConstraint. We
|
||||||
|
// therefore wake any constraint waiting for a free type's
|
||||||
|
// refcount to be 1 or 0.
|
||||||
|
if (refCount <= 1)
|
||||||
|
unblock(ty, Location{});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
// decrement the referenced free types for this constraint if we dispatched successfully!
|
// decrement the referenced free types for this constraint if we dispatched successfully!
|
||||||
for (auto ty : c->getMaybeMutatedFreeTypes())
|
for (auto ty : c->getMaybeMutatedFreeTypes())
|
||||||
{
|
{
|
||||||
|
@ -461,6 +488,7 @@ void ConstraintSolver::run()
|
||||||
if (refCount <= 1)
|
if (refCount <= 1)
|
||||||
unblock(ty, Location{});
|
unblock(ty, Location{});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (logger)
|
if (logger)
|
||||||
{
|
{
|
||||||
|
@ -551,7 +579,7 @@ void ConstraintSolver::finalizeTypeFunctions()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ConstraintSolver::isDone()
|
bool ConstraintSolver::isDone() const
|
||||||
{
|
{
|
||||||
return unsolvedConstraints.empty();
|
return unsolvedConstraints.empty();
|
||||||
}
|
}
|
||||||
|
@ -724,8 +752,20 @@ bool ConstraintSolver::tryDispatch(const GeneralizationConstraint& c, NotNull<co
|
||||||
bind(constraint, c.generalizedType, builtinTypes->errorRecoveryType());
|
bind(constraint, c.generalizedType, builtinTypes->errorRecoveryType());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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)
|
for (TypeId ty : c.interiorTypes)
|
||||||
generalize(NotNull{arena}, builtinTypes, constraint->scope, generalizedTypes, ty, /* avoidSealingTables */ false);
|
generalize(NotNull{arena}, builtinTypes, constraint->scope, generalizedTypes, ty, /* avoidSealingTables */ false);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -801,9 +841,17 @@ bool ConstraintSolver::tryDispatch(const IterableConstraint& c, NotNull<const Co
|
||||||
{
|
{
|
||||||
TypeId keyTy = freshType(arena, builtinTypes, constraint->scope);
|
TypeId keyTy = freshType(arena, builtinTypes, constraint->scope);
|
||||||
TypeId valueTy = 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 =
|
TypeId tableTy =
|
||||||
arena->addType(TableType{TableType::Props{}, TableIndexer{keyTy, valueTy}, TypeLevel{}, constraint->scope, TableState::Free});
|
arena->addType(TableType{TableType::Props{}, TableIndexer{keyTy, valueTy}, TypeLevel{}, constraint->scope, TableState::Free});
|
||||||
|
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope && FFlag::LuauTrackInteriorFreeTablesOnScope)
|
||||||
|
trackInteriorFreeType(constraint->scope, tableTy);
|
||||||
|
|
||||||
unify(constraint, nextTy, tableTy);
|
unify(constraint, nextTy, tableTy);
|
||||||
|
|
||||||
auto it = begin(c.variables);
|
auto it = begin(c.variables);
|
||||||
|
@ -940,16 +988,6 @@ bool ConstraintSolver::tryDispatch(const TypeAliasExpansionConstraint& c, NotNul
|
||||||
if (auto typeFn = get<TypeFunctionInstanceType>(follow(tf->type)))
|
if (auto typeFn = get<TypeFunctionInstanceType>(follow(tf->type)))
|
||||||
pushConstraint(NotNull(constraint->scope.get()), constraint->location, ReduceConstraint{tf->type});
|
pushConstraint(NotNull(constraint->scope.get()), constraint->location, ReduceConstraint{tf->type});
|
||||||
|
|
||||||
if (!FFlag::LuauUserTypeFunNoExtraConstraint)
|
|
||||||
{
|
|
||||||
// If there are no parameters to the type function we can just use the type directly
|
|
||||||
if (tf->typeParams.empty() && tf->typePackParams.empty())
|
|
||||||
{
|
|
||||||
bindResult(tf->type);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Due to how pending expansion types and TypeFun's are created
|
// Due to how pending expansion types and TypeFun's are created
|
||||||
// If this check passes, we have created a cyclic / corecursive type alias
|
// If this check passes, we have created a cyclic / corecursive type alias
|
||||||
// of size 0
|
// of size 0
|
||||||
|
@ -962,15 +1000,12 @@ bool ConstraintSolver::tryDispatch(const TypeAliasExpansionConstraint& c, NotNul
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunNoExtraConstraint)
|
|
||||||
{
|
|
||||||
// If there are no parameters to the type function we can just use the type directly
|
// If there are no parameters to the type function we can just use the type directly
|
||||||
if (tf->typeParams.empty() && tf->typePackParams.empty())
|
if (tf->typeParams.empty() && tf->typePackParams.empty())
|
||||||
{
|
{
|
||||||
bindResult(tf->type);
|
bindResult(tf->type);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
auto [typeArguments, packArguments] = saturateArguments(arena, builtinTypes, *tf, petv->typeArguments, petv->packArguments);
|
auto [typeArguments, packArguments] = saturateArguments(arena, builtinTypes, *tf, petv->typeArguments, petv->packArguments);
|
||||||
|
|
||||||
|
@ -1135,6 +1170,28 @@ bool ConstraintSolver::tryDispatch(const TypeAliasExpansionConstraint& c, NotNul
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ConstraintSolver::fillInDiscriminantTypes(
|
||||||
|
NotNull<const Constraint> constraint,
|
||||||
|
const std::vector<std::optional<TypeId>>& discriminantTypes
|
||||||
|
)
|
||||||
|
{
|
||||||
|
for (std::optional<TypeId> ty : discriminantTypes)
|
||||||
|
{
|
||||||
|
if (!ty)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
// If the discriminant type has been transmuted, we need to unblock them.
|
||||||
|
if (!isBlocked(*ty))
|
||||||
|
{
|
||||||
|
unblock(*ty, constraint->location);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We bind any unused discriminants to the `*no-refine*` type indicating that it can be safely ignored.
|
||||||
|
emplaceType<BoundType>(asMutable(follow(*ty)), builtinTypes->noRefineType);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool ConstraintSolver::tryDispatch(const FunctionCallConstraint& c, NotNull<const Constraint> constraint)
|
bool ConstraintSolver::tryDispatch(const FunctionCallConstraint& c, NotNull<const Constraint> constraint)
|
||||||
{
|
{
|
||||||
TypeId fn = follow(c.fn);
|
TypeId fn = follow(c.fn);
|
||||||
|
@ -1150,6 +1207,8 @@ bool ConstraintSolver::tryDispatch(const FunctionCallConstraint& c, NotNull<cons
|
||||||
{
|
{
|
||||||
emplaceTypePack<BoundTypePack>(asMutable(c.result), builtinTypes->anyTypePack);
|
emplaceTypePack<BoundTypePack>(asMutable(c.result), builtinTypes->anyTypePack);
|
||||||
unblock(c.result, constraint->location);
|
unblock(c.result, constraint->location);
|
||||||
|
if (FFlag::LuauAlwaysFillInFunctionCallDiscriminantTypes)
|
||||||
|
fillInDiscriminantTypes(constraint, c.discriminantTypes);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1157,12 +1216,16 @@ bool ConstraintSolver::tryDispatch(const FunctionCallConstraint& c, NotNull<cons
|
||||||
if (get<ErrorType>(fn))
|
if (get<ErrorType>(fn))
|
||||||
{
|
{
|
||||||
bind(constraint, c.result, builtinTypes->errorRecoveryTypePack());
|
bind(constraint, c.result, builtinTypes->errorRecoveryTypePack());
|
||||||
|
if (FFlag::LuauAlwaysFillInFunctionCallDiscriminantTypes)
|
||||||
|
fillInDiscriminantTypes(constraint, c.discriminantTypes);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (get<NeverType>(fn))
|
if (get<NeverType>(fn))
|
||||||
{
|
{
|
||||||
bind(constraint, c.result, builtinTypes->neverTypePack);
|
bind(constraint, c.result, builtinTypes->neverTypePack);
|
||||||
|
if (FFlag::LuauAlwaysFillInFunctionCallDiscriminantTypes)
|
||||||
|
fillInDiscriminantTypes(constraint, c.discriminantTypes);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1232,17 +1295,24 @@ bool ConstraintSolver::tryDispatch(const FunctionCallConstraint& c, NotNull<cons
|
||||||
|
|
||||||
if (ftv)
|
if (ftv)
|
||||||
{
|
{
|
||||||
if (ftv->dcrMagicFunction)
|
if (ftv->magic)
|
||||||
usedMagic = ftv->dcrMagicFunction(MagicFunctionCallContext{NotNull{this}, constraint, c.callSite, c.argsPack, result});
|
{
|
||||||
|
usedMagic = ftv->magic->infer(MagicFunctionCallContext{NotNull{this}, constraint, c.callSite, c.argsPack, result});
|
||||||
if (ftv->dcrMagicRefinement)
|
ftv->magic->refine(MagicRefinementContext{constraint->scope, c.callSite, c.discriminantTypes});
|
||||||
ftv->dcrMagicRefinement(MagicRefinementContext{constraint->scope, c.callSite, c.discriminantTypes});
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!usedMagic)
|
if (!usedMagic)
|
||||||
emplace<FreeTypePack>(constraint, c.result, constraint->scope);
|
emplace<FreeTypePack>(constraint, c.result, constraint->scope);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (FFlag::LuauAlwaysFillInFunctionCallDiscriminantTypes)
|
||||||
|
{
|
||||||
|
fillInDiscriminantTypes(constraint, c.discriminantTypes);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// NOTE: This is the body of the `fillInDiscriminantTypes` helper.
|
||||||
for (std::optional<TypeId> ty : c.discriminantTypes)
|
for (std::optional<TypeId> ty : c.discriminantTypes)
|
||||||
{
|
{
|
||||||
if (!ty)
|
if (!ty)
|
||||||
|
@ -1255,24 +1325,12 @@ bool ConstraintSolver::tryDispatch(const FunctionCallConstraint& c, NotNull<cons
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::LuauRemoveNotAnyHack)
|
|
||||||
{
|
|
||||||
// We bind any unused discriminants to the `*no-refine*` type indicating that it can be safely ignored.
|
// We bind any unused discriminants to the `*no-refine*` type indicating that it can be safely ignored.
|
||||||
emplaceType<BoundType>(asMutable(follow(*ty)), builtinTypes->noRefineType);
|
emplaceType<BoundType>(asMutable(follow(*ty)), builtinTypes->noRefineType);
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
// We use `any` here because the discriminant type may be pointed at by both branches,
|
|
||||||
// where the discriminant type is not negated, and the other where it is negated, i.e.
|
|
||||||
// `unknown ~ unknown` and `~unknown ~ never`, so `T & unknown ~ T` and `T & ~unknown ~ never`
|
|
||||||
// v.s.
|
|
||||||
// `any ~ any` and `~any ~ any`, so `T & any ~ T` and `T & ~any ~ T`
|
|
||||||
//
|
|
||||||
// In practice, users cannot negate `any`, so this is an implementation detail we can always change.
|
|
||||||
emplaceType<BoundType>(asMutable(follow(*ty)), builtinTypes->anyType);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
OverloadResolver resolver{
|
OverloadResolver resolver{
|
||||||
builtinTypes,
|
builtinTypes,
|
||||||
NotNull{arena},
|
NotNull{arena},
|
||||||
|
@ -1632,7 +1690,7 @@ bool ConstraintSolver::tryDispatchHasIndexer(
|
||||||
for (TypeId part : parts)
|
for (TypeId part : parts)
|
||||||
{
|
{
|
||||||
TypeId r = arena->addType(BlockedType{});
|
TypeId r = arena->addType(BlockedType{});
|
||||||
getMutable<BlockedType>(r)->setOwner(const_cast<Constraint*>(constraint.get()));
|
getMutable<BlockedType>(r)->setOwner(constraint.get());
|
||||||
|
|
||||||
bool ok = tryDispatchHasIndexer(recursionDepth, constraint, part, indexType, r, seen);
|
bool ok = tryDispatchHasIndexer(recursionDepth, constraint, part, indexType, r, seen);
|
||||||
// If we've cut a recursive loop short, skip it.
|
// If we've cut a recursive loop short, skip it.
|
||||||
|
@ -1664,7 +1722,7 @@ bool ConstraintSolver::tryDispatchHasIndexer(
|
||||||
for (TypeId part : parts)
|
for (TypeId part : parts)
|
||||||
{
|
{
|
||||||
TypeId r = arena->addType(BlockedType{});
|
TypeId r = arena->addType(BlockedType{});
|
||||||
getMutable<BlockedType>(r)->setOwner(const_cast<Constraint*>(constraint.get()));
|
getMutable<BlockedType>(r)->setOwner(constraint.get());
|
||||||
|
|
||||||
bool ok = tryDispatchHasIndexer(recursionDepth, constraint, part, indexType, r, seen);
|
bool ok = tryDispatchHasIndexer(recursionDepth, constraint, part, indexType, r, seen);
|
||||||
// If we've cut a recursive loop short, skip it.
|
// If we've cut a recursive loop short, skip it.
|
||||||
|
@ -1784,6 +1842,10 @@ bool ConstraintSolver::tryDispatch(const AssignPropConstraint& c, NotNull<const
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
TypeId newUpperBound = arena->addType(TableType{TableState::Free, TypeLevel{}, constraint->scope});
|
TypeId newUpperBound = arena->addType(TableType{TableState::Free, TypeLevel{}, constraint->scope});
|
||||||
|
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope && FFlag::LuauTrackInteriorFreeTablesOnScope)
|
||||||
|
trackInteriorFreeType(constraint->scope, newUpperBound);
|
||||||
|
|
||||||
TableType* upperTable = getMutable<TableType>(newUpperBound);
|
TableType* upperTable = getMutable<TableType>(newUpperBound);
|
||||||
LUAU_ASSERT(upperTable);
|
LUAU_ASSERT(upperTable);
|
||||||
|
|
||||||
|
@ -2062,6 +2124,8 @@ bool ConstraintSolver::tryDispatch(const UnpackConstraint& c, NotNull<const Cons
|
||||||
// constitute any meaningful constraint, so we replace it
|
// constitute any meaningful constraint, so we replace it
|
||||||
// with a free type.
|
// with a free type.
|
||||||
TypeId f = freshType(arena, builtinTypes, constraint->scope);
|
TypeId f = freshType(arena, builtinTypes, constraint->scope);
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
trackInteriorFreeType(constraint->scope, f);
|
||||||
shiftReferences(resultTy, f);
|
shiftReferences(resultTy, f);
|
||||||
emplaceType<BoundType>(asMutable(resultTy), f);
|
emplaceType<BoundType>(asMutable(resultTy), f);
|
||||||
}
|
}
|
||||||
|
@ -2197,6 +2261,11 @@ bool ConstraintSolver::tryDispatchIterableTable(TypeId iteratorTy, const Iterabl
|
||||||
{
|
{
|
||||||
TypeId keyTy = freshType(arena, builtinTypes, constraint->scope);
|
TypeId keyTy = freshType(arena, builtinTypes, constraint->scope);
|
||||||
TypeId valueTy = 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});
|
TypeId tableTy = arena->addType(TableType{TableState::Sealed, {}, constraint->scope});
|
||||||
getMutable<TableType>(tableTy)->indexer = TableIndexer{keyTy, valueTy};
|
getMutable<TableType>(tableTy)->indexer = TableIndexer{keyTy, valueTy};
|
||||||
|
|
||||||
|
@ -2453,6 +2522,8 @@ TablePropLookupResult ConstraintSolver::lookupTableProp(
|
||||||
if (ttv->state == TableState::Free)
|
if (ttv->state == TableState::Free)
|
||||||
{
|
{
|
||||||
TypeId result = freshType(arena, builtinTypes, ttv->scope);
|
TypeId result = freshType(arena, builtinTypes, ttv->scope);
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
trackInteriorFreeType(ttv->scope, result);
|
||||||
switch (context)
|
switch (context)
|
||||||
{
|
{
|
||||||
case ValueContext::RValue:
|
case ValueContext::RValue:
|
||||||
|
@ -2558,10 +2629,17 @@ TablePropLookupResult ConstraintSolver::lookupTableProp(
|
||||||
NotNull<Scope> scope{ft->scope};
|
NotNull<Scope> scope{ft->scope};
|
||||||
|
|
||||||
const TypeId newUpperBound = arena->addType(TableType{TableState::Free, TypeLevel{}, scope});
|
const TypeId newUpperBound = arena->addType(TableType{TableState::Free, TypeLevel{}, scope});
|
||||||
|
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope && FFlag::LuauTrackInteriorFreeTablesOnScope)
|
||||||
|
trackInteriorFreeType(constraint->scope, newUpperBound);
|
||||||
|
|
||||||
TableType* tt = getMutable<TableType>(newUpperBound);
|
TableType* tt = getMutable<TableType>(newUpperBound);
|
||||||
LUAU_ASSERT(tt);
|
LUAU_ASSERT(tt);
|
||||||
TypeId propType = freshType(arena, builtinTypes, scope);
|
TypeId propType = freshType(arena, builtinTypes, scope);
|
||||||
|
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
trackInteriorFreeType(scope, propType);
|
||||||
|
|
||||||
switch (context)
|
switch (context)
|
||||||
{
|
{
|
||||||
case ValueContext::RValue:
|
case ValueContext::RValue:
|
||||||
|
@ -2792,10 +2870,10 @@ bool ConstraintSolver::blockOnPendingTypes(TypeId target, NotNull<const Constrai
|
||||||
return !blocker.blocked;
|
return !blocker.blocked;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ConstraintSolver::blockOnPendingTypes(TypePackId pack, NotNull<const Constraint> constraint)
|
bool ConstraintSolver::blockOnPendingTypes(TypePackId targetPack, NotNull<const Constraint> constraint)
|
||||||
{
|
{
|
||||||
Blocker blocker{NotNull{this}, constraint};
|
Blocker blocker{NotNull{this}, constraint};
|
||||||
blocker.traverse(pack);
|
blocker.traverse(targetPack);
|
||||||
return !blocker.blocked;
|
return !blocker.blocked;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,6 @@
|
||||||
|
|
||||||
LUAU_FASTFLAG(DebugLuauFreezeArena)
|
LUAU_FASTFLAG(DebugLuauFreezeArena)
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAG(LuauTypestateBuiltins2)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -62,6 +61,12 @@ const RefinementKey* RefinementKeyArena::node(const RefinementKey* parent, DefId
|
||||||
return allocator.allocate(RefinementKey{parent, def, propName});
|
return allocator.allocate(RefinementKey{parent, def, propName});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DataFlowGraph::DataFlowGraph(NotNull<DefArena> defArena, NotNull<RefinementKeyArena> keyArena)
|
||||||
|
: defArena{defArena}
|
||||||
|
, keyArena{keyArena}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
DefId DataFlowGraph::getDef(const AstExpr* expr) const
|
DefId DataFlowGraph::getDef(const AstExpr* expr) const
|
||||||
{
|
{
|
||||||
auto def = astDefs.find(expr);
|
auto def = astDefs.find(expr);
|
||||||
|
@ -178,11 +183,23 @@ bool DfgScope::canUpdateDefinition(DefId def, const std::string& key) const
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
DataFlowGraph DataFlowGraphBuilder::build(AstStatBlock* block, NotNull<InternalErrorReporter> handle)
|
DataFlowGraphBuilder::DataFlowGraphBuilder(NotNull<DefArena> defArena, NotNull<RefinementKeyArena> keyArena)
|
||||||
|
: graph{defArena, keyArena}
|
||||||
|
, defArena{defArena}
|
||||||
|
, keyArena{keyArena}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
DataFlowGraph DataFlowGraphBuilder::build(
|
||||||
|
AstStatBlock* block,
|
||||||
|
NotNull<DefArena> defArena,
|
||||||
|
NotNull<RefinementKeyArena> keyArena,
|
||||||
|
NotNull<struct InternalErrorReporter> handle
|
||||||
|
)
|
||||||
{
|
{
|
||||||
LUAU_TIMETRACE_SCOPE("DataFlowGraphBuilder::build", "Typechecking");
|
LUAU_TIMETRACE_SCOPE("DataFlowGraphBuilder::build", "Typechecking");
|
||||||
|
|
||||||
DataFlowGraphBuilder builder;
|
DataFlowGraphBuilder builder(defArena, keyArena);
|
||||||
builder.handle = handle;
|
builder.handle = handle;
|
||||||
DfgScope* moduleScope = builder.makeChildScope();
|
DfgScope* moduleScope = builder.makeChildScope();
|
||||||
PushScope ps{builder.scopeStack, moduleScope};
|
PushScope ps{builder.scopeStack, moduleScope};
|
||||||
|
@ -198,30 +215,6 @@ DataFlowGraph DataFlowGraphBuilder::build(AstStatBlock* block, NotNull<InternalE
|
||||||
return std::move(builder.graph);
|
return std::move(builder.graph);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::pair<std::shared_ptr<DataFlowGraph>, std::vector<std::unique_ptr<DfgScope>>> DataFlowGraphBuilder::buildShared(
|
|
||||||
AstStatBlock* block,
|
|
||||||
NotNull<InternalErrorReporter> handle
|
|
||||||
)
|
|
||||||
{
|
|
||||||
|
|
||||||
LUAU_TIMETRACE_SCOPE("DataFlowGraphBuilder::build", "Typechecking");
|
|
||||||
|
|
||||||
DataFlowGraphBuilder builder;
|
|
||||||
builder.handle = handle;
|
|
||||||
DfgScope* moduleScope = builder.makeChildScope();
|
|
||||||
PushScope ps{builder.scopeStack, moduleScope};
|
|
||||||
builder.visitBlockWithoutChildScope(block);
|
|
||||||
builder.resolveCaptures();
|
|
||||||
|
|
||||||
if (FFlag::DebugLuauFreezeArena)
|
|
||||||
{
|
|
||||||
builder.defArena->allocator.freeze();
|
|
||||||
builder.keyArena->allocator.freeze();
|
|
||||||
}
|
|
||||||
|
|
||||||
return {std::make_shared<DataFlowGraph>(std::move(builder.graph)), std::move(builder.scopes)};
|
|
||||||
}
|
|
||||||
|
|
||||||
void DataFlowGraphBuilder::resolveCaptures()
|
void DataFlowGraphBuilder::resolveCaptures()
|
||||||
{
|
{
|
||||||
for (const auto& [_, capture] : captures)
|
for (const auto& [_, capture] : captures)
|
||||||
|
@ -885,7 +878,7 @@ DataFlowResult DataFlowGraphBuilder::visitExpr(AstExprCall* c)
|
||||||
{
|
{
|
||||||
visitExpr(c->func);
|
visitExpr(c->func);
|
||||||
|
|
||||||
if (FFlag::LuauTypestateBuiltins2 && shouldTypestateForFirstArgument(*c) && c->args.size > 1 && isLValue(*c->args.begin()))
|
if (shouldTypestateForFirstArgument(*c) && c->args.size > 1 && isLValue(*c->args.begin()))
|
||||||
{
|
{
|
||||||
AstExpr* firstArg = *c->args.begin();
|
AstExpr* firstArg = *c->args.begin();
|
||||||
|
|
||||||
|
@ -1176,6 +1169,8 @@ void DataFlowGraphBuilder::visitType(AstType* t)
|
||||||
return; // ok
|
return; // ok
|
||||||
else if (auto s = t->as<AstTypeSingletonString>())
|
else if (auto s = t->as<AstTypeSingletonString>())
|
||||||
return; // ok
|
return; // ok
|
||||||
|
else if (auto g = t->as<AstTypeGroup>())
|
||||||
|
return visitType(g->type);
|
||||||
else
|
else
|
||||||
handle->ice("Unknown AstType in DataFlowGraphBuilder::visitType");
|
handle->ice("Unknown AstType in DataFlowGraphBuilder::visitType");
|
||||||
}
|
}
|
||||||
|
@ -1265,21 +1260,21 @@ void DataFlowGraphBuilder::visitTypeList(AstTypeList l)
|
||||||
visitTypePack(l.tailType);
|
visitTypePack(l.tailType);
|
||||||
}
|
}
|
||||||
|
|
||||||
void DataFlowGraphBuilder::visitGenerics(AstArray<AstGenericType> g)
|
void DataFlowGraphBuilder::visitGenerics(AstArray<AstGenericType*> g)
|
||||||
{
|
{
|
||||||
for (AstGenericType generic : g)
|
for (AstGenericType* generic : g)
|
||||||
{
|
{
|
||||||
if (generic.defaultValue)
|
if (generic->defaultValue)
|
||||||
visitType(generic.defaultValue);
|
visitType(generic->defaultValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void DataFlowGraphBuilder::visitGenericPacks(AstArray<AstGenericTypePack> g)
|
void DataFlowGraphBuilder::visitGenericPacks(AstArray<AstGenericTypePack*> g)
|
||||||
{
|
{
|
||||||
for (AstGenericTypePack generic : g)
|
for (AstGenericTypePack* generic : g)
|
||||||
{
|
{
|
||||||
if (generic.defaultValue)
|
if (generic->defaultValue)
|
||||||
visitTypePack(generic.defaultValue);
|
visitTypePack(generic->defaultValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,7 +13,6 @@
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
std::string DiffPathNode::toString() const
|
std::string DiffPathNode::toString() const
|
||||||
{
|
{
|
||||||
switch (kind)
|
switch (kind)
|
||||||
|
@ -945,14 +944,12 @@ std::vector<std::pair<TypeId, TypeId>>::const_reverse_iterator DifferEnvironment
|
||||||
return visitingStack.crend();
|
return visitingStack.crend();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
DifferResult diff(TypeId ty1, TypeId ty2)
|
DifferResult diff(TypeId ty1, TypeId ty2)
|
||||||
{
|
{
|
||||||
DifferEnvironment differEnv{ty1, ty2, std::nullopt, std::nullopt};
|
DifferEnvironment differEnv{ty1, ty2, std::nullopt, std::nullopt};
|
||||||
return diffUsingEnv(differEnv, ty1, ty2);
|
return diffUsingEnv(differEnv, ty1, ty2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
DifferResult diffWithSymbols(TypeId ty1, TypeId ty2, std::optional<std::string> symbol1, std::optional<std::string> symbol2)
|
DifferResult diffWithSymbols(TypeId ty1, TypeId ty2, std::optional<std::string> symbol1, std::optional<std::string> symbol2)
|
||||||
{
|
{
|
||||||
DifferEnvironment differEnv{ty1, ty2, symbol1, symbol2};
|
DifferEnvironment differEnv{ty1, ty2, symbol1, symbol2};
|
||||||
|
|
|
@ -1,108 +1,13 @@
|
||||||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
#include "Luau/BuiltinDefinitions.h"
|
#include "Luau/BuiltinDefinitions.h"
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauMathMap)
|
LUAU_FASTFLAG(LuauBufferBitMethods2)
|
||||||
|
LUAU_FASTFLAG(LuauVector2Constructor)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitions)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitionsExtra)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
// TODO: there has to be a better way, like splitting up per library
|
static const std::string kBuiltinDefinitionBaseSrc = R"BUILTIN_SRC(
|
||||||
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 require(target: any): any
|
||||||
|
|
||||||
|
@ -150,88 +55,12 @@ declare function loadstring<A...>(src: string, chunkname: string?): (((A...) ->
|
||||||
|
|
||||||
@checked declare function newproxy(mt: boolean?): any
|
@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.
|
-- 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
|
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";
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
static const std::string kBuiltinDefinitionLuaSrcChecked = R"BUILTIN_SRC(
|
static const std::string kBuiltinDefinitionBit32Src = R"BUILTIN_SRC(
|
||||||
|
|
||||||
declare bit32: {
|
declare bit32: {
|
||||||
band: @checked (...number) -> number,
|
band: @checked (...number) -> number,
|
||||||
|
@ -251,6 +80,10 @@ declare bit32: {
|
||||||
byteswap: @checked (n: number) -> number,
|
byteswap: @checked (n: number) -> number,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionMathSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
declare math: {
|
declare math: {
|
||||||
frexp: @checked (n: number) -> (number, number),
|
frexp: @checked (n: number) -> (number, number),
|
||||||
ldexp: @checked (s: number, e: number) -> number,
|
ldexp: @checked (s: number, e: number) -> number,
|
||||||
|
@ -295,8 +128,13 @@ declare math: {
|
||||||
noise: @checked (x: number, y: number?, z: number?) -> number,
|
noise: @checked (x: number, y: number?, z: number?) -> number,
|
||||||
round: @checked (n: number) -> number,
|
round: @checked (n: number) -> number,
|
||||||
map: @checked (x: number, inmin: number, inmax: number, outmin: number, outmax: number) -> number,
|
map: @checked (x: number, inmin: number, inmax: number, outmin: number, outmax: number) -> number,
|
||||||
|
lerp: @checked (a: number, b: number, t: number) -> number,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionOsSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
type DateTypeArg = {
|
type DateTypeArg = {
|
||||||
year: number,
|
year: number,
|
||||||
month: number,
|
month: number,
|
||||||
|
@ -326,51 +164,9 @@ declare os: {
|
||||||
clock: () -> number,
|
clock: () -> number,
|
||||||
}
|
}
|
||||||
|
|
||||||
@checked declare function require(target: any): any
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
@checked declare function getfenv(target: any): { [string]: any }
|
static const std::string kBuiltinDefinitionCoroutineSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
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: {
|
declare coroutine: {
|
||||||
create: <A..., R...>(f: (A...) -> R...) -> thread,
|
create: <A..., R...>(f: (A...) -> R...) -> thread,
|
||||||
|
@ -383,6 +179,10 @@ declare coroutine: {
|
||||||
close: @checked (co: thread) -> (boolean, any)
|
close: @checked (co: thread) -> (boolean, any)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionTableSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
declare table: {
|
declare table: {
|
||||||
concat: <V>(t: {V}, sep: string?, i: number?, j: number?) -> string,
|
concat: <V>(t: {V}, sep: string?, i: number?, j: number?) -> string,
|
||||||
insert: (<V>(t: {V}, value: V) -> ()) & (<V>(t: {V}, pos: number, value: V) -> ()),
|
insert: (<V>(t: {V}, value: V) -> ()) & (<V>(t: {V}, pos: number, value: V) -> ()),
|
||||||
|
@ -405,11 +205,19 @@ declare table: {
|
||||||
isfrozen: <K, V>(t: {[K]: V}) -> boolean,
|
isfrozen: <K, V>(t: {[K]: V}) -> boolean,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionDebugSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
declare debug: {
|
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...),
|
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),
|
traceback: ((message: string?, level: number?) -> string) & ((thread: thread, message: string?, level: number?) -> string),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionUtf8Src = R"BUILTIN_SRC(
|
||||||
|
|
||||||
declare utf8: {
|
declare utf8: {
|
||||||
char: @checked (...number) -> string,
|
char: @checked (...number) -> string,
|
||||||
charpattern: string,
|
charpattern: string,
|
||||||
|
@ -419,10 +227,9 @@ declare utf8: {
|
||||||
offset: @checked (s: string, n: number?, i: 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.
|
)BUILTIN_SRC";
|
||||||
declare function unpack<V>(tab: {V}, i: number?, j: number?): ...V
|
|
||||||
|
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionBufferSrc_DEPRECATED = R"BUILTIN_SRC(
|
||||||
--- Buffer API
|
--- Buffer API
|
||||||
declare buffer: {
|
declare buffer: {
|
||||||
create: @checked (size: number) -> buffer,
|
create: @checked (size: number) -> buffer,
|
||||||
|
@ -453,10 +260,47 @@ declare buffer: {
|
||||||
|
|
||||||
)BUILTIN_SRC";
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
static const std::string kBuiltinDefinitionVectorSrc_DEPRECATED = R"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) -> (),
|
||||||
|
}
|
||||||
|
|
||||||
-- TODO: this will be replaced with a built-in primitive type
|
)BUILTIN_SRC";
|
||||||
declare class vector end
|
|
||||||
|
static const std::string kBuiltinDefinitionVectorSrc_NoVector2Ctor_DEPRECATED = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
-- While vector would have been better represented as a built-in primitive type, type solver class handling covers most of the properties
|
||||||
|
declare class vector
|
||||||
|
x: number
|
||||||
|
y: number
|
||||||
|
z: number
|
||||||
|
end
|
||||||
|
|
||||||
declare vector: {
|
declare vector: {
|
||||||
create: @checked (x: number, y: number, z: number) -> vector,
|
create: @checked (x: number, y: number, z: number) -> vector,
|
||||||
|
@ -489,7 +333,7 @@ declare class vector
|
||||||
end
|
end
|
||||||
|
|
||||||
declare vector: {
|
declare vector: {
|
||||||
create: @checked (x: number, y: number, z: number) -> vector,
|
create: @checked (x: number, y: number, z: number?) -> vector,
|
||||||
magnitude: @checked (vec: vector) -> number,
|
magnitude: @checked (vec: vector) -> number,
|
||||||
normalize: @checked (vec: vector) -> vector,
|
normalize: @checked (vec: vector) -> vector,
|
||||||
cross: @checked (vec1: vector, vec2: vector) -> vector,
|
cross: @checked (vec1: vector, vec2: vector) -> vector,
|
||||||
|
@ -511,12 +355,22 @@ declare vector: {
|
||||||
|
|
||||||
std::string getBuiltinDefinitionSource()
|
std::string getBuiltinDefinitionSource()
|
||||||
{
|
{
|
||||||
std::string result = FFlag::LuauMathMap ? kBuiltinDefinitionLuaSrcChecked : kBuiltinDefinitionLuaSrcChecked_DEPRECATED;
|
std::string result = kBuiltinDefinitionBaseSrc;
|
||||||
|
|
||||||
if (FFlag::LuauVectorDefinitionsExtra)
|
result += kBuiltinDefinitionBit32Src;
|
||||||
|
result += kBuiltinDefinitionMathSrc;
|
||||||
|
result += kBuiltinDefinitionOsSrc;
|
||||||
|
result += kBuiltinDefinitionCoroutineSrc;
|
||||||
|
result += kBuiltinDefinitionTableSrc;
|
||||||
|
result += kBuiltinDefinitionDebugSrc;
|
||||||
|
result += kBuiltinDefinitionUtf8Src;
|
||||||
|
|
||||||
|
result += FFlag::LuauBufferBitMethods2 ? kBuiltinDefinitionBufferSrc : kBuiltinDefinitionBufferSrc_DEPRECATED;
|
||||||
|
|
||||||
|
if (FFlag::LuauVector2Constructor)
|
||||||
result += kBuiltinDefinitionVectorSrc;
|
result += kBuiltinDefinitionVectorSrc;
|
||||||
else if (FFlag::LuauVectorDefinitions)
|
else
|
||||||
result += kBuiltinDefinitionVectorSrc_DEPRECATED;
|
result += kBuiltinDefinitionVectorSrc_NoVector2Ctor_DEPRECATED;
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
|
@ -92,18 +92,24 @@ size_t TTable::Hash::operator()(const TTable& value) const
|
||||||
return hash;
|
return hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t StringCache::add(std::string_view s)
|
StringId StringCache::add(std::string_view s)
|
||||||
{
|
{
|
||||||
size_t hash = std::hash<std::string_view>()(s);
|
/* Important subtlety: This use of DenseHashMap<std::string_view, StringId>
|
||||||
if (uint32_t* it = strings.find(hash))
|
* is okay because std::hash<std::string_view> works solely on the bytes
|
||||||
|
* referred by the string_view.
|
||||||
|
*
|
||||||
|
* In other words, two string views which contain the same bytes will have
|
||||||
|
* the same hash whether or not their addresses are the same.
|
||||||
|
*/
|
||||||
|
if (StringId* it = strings.find(s))
|
||||||
return *it;
|
return *it;
|
||||||
|
|
||||||
char* storage = static_cast<char*>(allocator.allocate(s.size()));
|
char* storage = static_cast<char*>(allocator.allocate(s.size()));
|
||||||
memcpy(storage, s.data(), s.size());
|
memcpy(storage, s.data(), s.size());
|
||||||
|
|
||||||
uint32_t result = uint32_t(views.size());
|
StringId result = StringId(views.size());
|
||||||
views.emplace_back(storage, s.size());
|
views.emplace_back(storage, s.size());
|
||||||
strings[hash] = result;
|
strings[s] = result;
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -193,9 +199,8 @@ static bool areTerminalAndDefinitelyDisjoint(const EType& lhs, const EType& rhs)
|
||||||
// - Whether one of the enodes is a large semantic set such as TAny,
|
// - Whether one of the enodes is a large semantic set such as TAny,
|
||||||
// TUnknown, or TError.
|
// TUnknown, or TError.
|
||||||
return !(
|
return !(
|
||||||
lhs.index() == rhs.index() ||
|
lhs.index() == rhs.index() || lhs.get<TUnknown>() || rhs.get<TUnknown>() || lhs.get<TAny>() || rhs.get<TAny>() || lhs.get<TNoRefine>() ||
|
||||||
lhs.get<TUnknown>() || rhs.get<TUnknown>() || lhs.get<TAny>() || rhs.get<TAny>() || lhs.get<TNoRefine>() || rhs.get<TNoRefine>() ||
|
rhs.get<TNoRefine>() || lhs.get<TError>() || rhs.get<TError>() || lhs.get<TOpaque>() || rhs.get<TOpaque>()
|
||||||
lhs.get<TError>() || rhs.get<TError>() || lhs.get<TOpaque>() || rhs.get<TOpaque>()
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -207,7 +212,7 @@ static bool isTerminal(const EGraph& egraph, Id eclass)
|
||||||
nodes.end(),
|
nodes.end(),
|
||||||
[](auto& a)
|
[](auto& a)
|
||||||
{
|
{
|
||||||
return isTerminal(a);
|
return isTerminal(a.node);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -391,6 +396,16 @@ Id toId(
|
||||||
{
|
{
|
||||||
LUAU_ASSERT(tfun->packArguments.empty());
|
LUAU_ASSERT(tfun->packArguments.empty());
|
||||||
|
|
||||||
|
if (tfun->userFuncName) {
|
||||||
|
// TODO: User defined type functions are pseudo-effectful: error
|
||||||
|
// reporting is done via the `print` statement, so running a
|
||||||
|
// UDTF multiple times may end up double erroring. egraphs
|
||||||
|
// currently may induce type functions to be reduced multiple
|
||||||
|
// times. We should probably opt _not_ to process user defined
|
||||||
|
// type functions at all.
|
||||||
|
return egraph.add(TOpaque{ty});
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<Id> parts;
|
std::vector<Id> parts;
|
||||||
parts.reserve(tfun->typeArguments.size());
|
parts.reserve(tfun->typeArguments.size());
|
||||||
for (TypeId part : tfun->typeArguments)
|
for (TypeId part : tfun->typeArguments)
|
||||||
|
@ -465,7 +480,7 @@ static size_t computeCost(std::unordered_map<Id, size_t>& bestNodes, const EGrap
|
||||||
if (auto it = costs.find(id); it != costs.end())
|
if (auto it = costs.find(id); it != costs.end())
|
||||||
return it->second;
|
return it->second;
|
||||||
|
|
||||||
const std::vector<EType>& nodes = egraph[id].nodes;
|
const std::vector<Node<EType>>& nodes = egraph[id].nodes;
|
||||||
|
|
||||||
size_t minCost = std::numeric_limits<size_t>::max();
|
size_t minCost = std::numeric_limits<size_t>::max();
|
||||||
size_t bestNode = std::numeric_limits<size_t>::max();
|
size_t bestNode = std::numeric_limits<size_t>::max();
|
||||||
|
@ -482,7 +497,7 @@ static size_t computeCost(std::unordered_map<Id, size_t>& bestNodes, const EGrap
|
||||||
// First, quickly scan for a terminal type. If we can find one, it is obviously the best.
|
// First, quickly scan for a terminal type. If we can find one, it is obviously the best.
|
||||||
for (size_t index = 0; index < nodes.size(); ++index)
|
for (size_t index = 0; index < nodes.size(); ++index)
|
||||||
{
|
{
|
||||||
if (isTerminal(nodes[index]))
|
if (isTerminal(nodes[index].node))
|
||||||
{
|
{
|
||||||
minCost = 1;
|
minCost = 1;
|
||||||
bestNode = index;
|
bestNode = index;
|
||||||
|
@ -534,44 +549,44 @@ static size_t computeCost(std::unordered_map<Id, size_t>& bestNodes, const EGrap
|
||||||
{
|
{
|
||||||
const auto& node = nodes[index];
|
const auto& node = nodes[index];
|
||||||
|
|
||||||
if (node.get<TBound>())
|
if (node.node.get<TBound>())
|
||||||
updateCost(BOUND_PENALTY, index); // TODO: This could probably be an assert now that we don't need rewrite rules to handle TBound.
|
updateCost(BOUND_PENALTY, index); // TODO: This could probably be an assert now that we don't need rewrite rules to handle TBound.
|
||||||
else if (node.get<TFunction>())
|
else if (node.node.get<TFunction>())
|
||||||
{
|
{
|
||||||
minCost = 1;
|
minCost = 1;
|
||||||
bestNode = index;
|
bestNode = index;
|
||||||
}
|
}
|
||||||
else if (auto tbl = node.get<TTable>())
|
else if (auto tbl = node.node.get<TTable>())
|
||||||
{
|
{
|
||||||
// TODO: We could make the penalty a parameter to computeChildren.
|
// TODO: We could make the penalty a parameter to computeChildren.
|
||||||
std::optional<size_t> maybeCost = computeChildren(tbl->operands(), minCost);
|
std::optional<size_t> maybeCost = computeChildren(tbl->operands(), minCost);
|
||||||
if (maybeCost)
|
if (maybeCost)
|
||||||
updateCost(TABLE_TYPE_PENALTY + *maybeCost, index);
|
updateCost(TABLE_TYPE_PENALTY + *maybeCost, index);
|
||||||
}
|
}
|
||||||
else if (node.get<TImportedTable>())
|
else if (node.node.get<TImportedTable>())
|
||||||
{
|
{
|
||||||
minCost = IMPORTED_TABLE_PENALTY;
|
minCost = IMPORTED_TABLE_PENALTY;
|
||||||
bestNode = index;
|
bestNode = index;
|
||||||
}
|
}
|
||||||
else if (auto u = node.get<Union>())
|
else if (auto u = node.node.get<Union>())
|
||||||
{
|
{
|
||||||
std::optional<size_t> maybeCost = computeChildren(u->operands(), minCost);
|
std::optional<size_t> maybeCost = computeChildren(u->operands(), minCost);
|
||||||
if (maybeCost)
|
if (maybeCost)
|
||||||
updateCost(SET_TYPE_PENALTY + *maybeCost, index);
|
updateCost(SET_TYPE_PENALTY + *maybeCost, index);
|
||||||
}
|
}
|
||||||
else if (auto i = node.get<Intersection>())
|
else if (auto i = node.node.get<Intersection>())
|
||||||
{
|
{
|
||||||
std::optional<size_t> maybeCost = computeChildren(i->operands(), minCost);
|
std::optional<size_t> maybeCost = computeChildren(i->operands(), minCost);
|
||||||
if (maybeCost)
|
if (maybeCost)
|
||||||
updateCost(SET_TYPE_PENALTY + *maybeCost, index);
|
updateCost(SET_TYPE_PENALTY + *maybeCost, index);
|
||||||
}
|
}
|
||||||
else if (auto negation = node.get<Negation>())
|
else if (auto negation = node.node.get<Negation>())
|
||||||
{
|
{
|
||||||
std::optional<size_t> maybeCost = computeChildren(negation->operands(), minCost);
|
std::optional<size_t> maybeCost = computeChildren(negation->operands(), minCost);
|
||||||
if (maybeCost)
|
if (maybeCost)
|
||||||
updateCost(NEGATION_PENALTY + *maybeCost, index);
|
updateCost(NEGATION_PENALTY + *maybeCost, index);
|
||||||
}
|
}
|
||||||
else if (auto tfun = node.get<TTypeFun>())
|
else if (auto tfun = node.node.get<TTypeFun>())
|
||||||
{
|
{
|
||||||
std::optional<size_t> maybeCost = computeChildren(tfun->operands(), minCost);
|
std::optional<size_t> maybeCost = computeChildren(tfun->operands(), minCost);
|
||||||
if (maybeCost)
|
if (maybeCost)
|
||||||
|
@ -644,7 +659,7 @@ TypeId flattenTableNode(
|
||||||
|
|
||||||
for (size_t i = 0; i < eclass.nodes.size(); ++i)
|
for (size_t i = 0; i < eclass.nodes.size(); ++i)
|
||||||
{
|
{
|
||||||
if (eclass.nodes[i].get<TImportedTable>())
|
if (eclass.nodes[i].node.get<TImportedTable>())
|
||||||
{
|
{
|
||||||
found = true;
|
found = true;
|
||||||
index = i;
|
index = i;
|
||||||
|
@ -661,13 +676,13 @@ TypeId flattenTableNode(
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto& node = eclass.nodes[index];
|
const auto& node = eclass.nodes[index];
|
||||||
if (const TTable* ttable = node.get<TTable>())
|
if (const TTable* ttable = node.node.get<TTable>())
|
||||||
{
|
{
|
||||||
stack.push_back(ttable);
|
stack.push_back(ttable);
|
||||||
id = ttable->getBasis();
|
id = ttable->getBasis();
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
else if (const TImportedTable* ti = node.get<TImportedTable>())
|
else if (const TImportedTable* ti = node.node.get<TImportedTable>())
|
||||||
{
|
{
|
||||||
importedTable = ti;
|
importedTable = ti;
|
||||||
break;
|
break;
|
||||||
|
@ -694,7 +709,8 @@ TypeId flattenTableNode(
|
||||||
StringId propName = t->propNames[i];
|
StringId propName = t->propNames[i];
|
||||||
const Id propType = t->propTypes()[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)};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -718,7 +734,7 @@ TypeId fromId(
|
||||||
size_t index = bestNodes.at(rootId);
|
size_t index = bestNodes.at(rootId);
|
||||||
LUAU_ASSERT(index <= egraph[rootId].nodes.size());
|
LUAU_ASSERT(index <= egraph[rootId].nodes.size());
|
||||||
|
|
||||||
const EType& node = egraph[rootId].nodes[index];
|
const EType& node = egraph[rootId].nodes[index].node;
|
||||||
|
|
||||||
if (node.get<TNil>())
|
if (node.get<TNil>())
|
||||||
return builtinTypes->nilType;
|
return builtinTypes->nilType;
|
||||||
|
@ -942,7 +958,15 @@ std::string mkDesc(
|
||||||
return rule + ":" + rulePadding + fromIdStr + toString(fromTy, opts) + " <=> " + toIdStr + toString(toTy, opts);
|
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)
|
if (!FFlag::DebugLuauLogSimplification)
|
||||||
return "";
|
return "";
|
||||||
|
@ -1017,8 +1041,9 @@ std::string toDot(const StringCache& strings, const EGraph& egraph)
|
||||||
|
|
||||||
for (const auto& [id, eclass] : egraph.getAllClasses())
|
for (const auto& [id, eclass] : egraph.getAllClasses())
|
||||||
{
|
{
|
||||||
for (const auto& node : eclass.nodes)
|
for (const auto& n : eclass.nodes)
|
||||||
{
|
{
|
||||||
|
const EType& node = n.node;
|
||||||
if (!node.operands().empty())
|
if (!node.operands().empty())
|
||||||
populated.insert(id);
|
populated.insert(id);
|
||||||
for (Id op : node.operands())
|
for (Id op : node.operands())
|
||||||
|
@ -1039,7 +1064,7 @@ std::string toDot(const StringCache& strings, const EGraph& egraph)
|
||||||
|
|
||||||
for (size_t index = 0; index < eclass.nodes.size(); ++index)
|
for (size_t index = 0; index < eclass.nodes.size(); ++index)
|
||||||
{
|
{
|
||||||
const auto& node = eclass.nodes[index];
|
const auto& node = eclass.nodes[index].node;
|
||||||
|
|
||||||
const std::string label = getNodeName(strings, node);
|
const std::string label = getNodeName(strings, node);
|
||||||
const std::string nodeName = "n" + std::to_string(uint32_t(id)) + "_" + std::to_string(index);
|
const std::string nodeName = "n" + std::to_string(uint32_t(id)) + "_" + std::to_string(index);
|
||||||
|
@ -1054,7 +1079,7 @@ std::string toDot(const StringCache& strings, const EGraph& egraph)
|
||||||
{
|
{
|
||||||
for (size_t index = 0; index < eclass.nodes.size(); ++index)
|
for (size_t index = 0; index < eclass.nodes.size(); ++index)
|
||||||
{
|
{
|
||||||
const auto& node = eclass.nodes[index];
|
const auto& node = eclass.nodes[index].node;
|
||||||
|
|
||||||
const std::string label = getNodeName(strings, node);
|
const std::string label = getNodeName(strings, node);
|
||||||
const std::string nodeName = "n" + std::to_string(uint32_t(egraph.find(id))) + "_" + std::to_string(index);
|
const std::string nodeName = "n" + std::to_string(uint32_t(egraph.find(id))) + "_" + std::to_string(index);
|
||||||
|
@ -1090,7 +1115,7 @@ static Tag const* isTag(const EGraph& egraph, Id id)
|
||||||
{
|
{
|
||||||
for (const auto& node : egraph[id].nodes)
|
for (const auto& node : egraph[id].nodes)
|
||||||
{
|
{
|
||||||
if (auto n = isTag<Tag>(node))
|
if (auto n = isTag<Tag>(node.node))
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
@ -1126,7 +1151,7 @@ protected:
|
||||||
{
|
{
|
||||||
for (const auto& node : (*egraph)[id].nodes)
|
for (const auto& node : (*egraph)[id].nodes)
|
||||||
{
|
{
|
||||||
if (auto n = node.get<Tag>())
|
if (auto n = node.node.get<Tag>())
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
@ -1314,8 +1339,10 @@ const EType* findSubtractableClass(const EGraph& egraph, std::unordered_set<Id>&
|
||||||
const EType* bestUnion = nullptr;
|
const EType* bestUnion = nullptr;
|
||||||
std::optional<size_t> unionSize;
|
std::optional<size_t> unionSize;
|
||||||
|
|
||||||
for (const auto& node : egraph[id].nodes)
|
for (const auto& n : egraph[id].nodes)
|
||||||
{
|
{
|
||||||
|
const EType& node = n.node;
|
||||||
|
|
||||||
if (isTerminal(node))
|
if (isTerminal(node))
|
||||||
return &node;
|
return &node;
|
||||||
|
|
||||||
|
@ -1431,14 +1458,14 @@ bool subtract(EGraph& egraph, CanonicalizedType& ct, Id part)
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Id fromCanonicalized(EGraph& egraph, CanonicalizedType& ct)
|
static std::pair<Id, size_t> fromCanonicalized(EGraph& egraph, CanonicalizedType& ct)
|
||||||
{
|
{
|
||||||
if (ct.isUnknown())
|
if (ct.isUnknown())
|
||||||
{
|
{
|
||||||
if (ct.errorPart)
|
if (ct.errorPart)
|
||||||
return egraph.add(TAny{});
|
return {egraph.add(TAny{}), 1};
|
||||||
else
|
else
|
||||||
return egraph.add(TUnknown{});
|
return {egraph.add(TUnknown{}), 1};
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<Id> parts;
|
std::vector<Id> parts;
|
||||||
|
@ -1476,7 +1503,12 @@ Id fromCanonicalized(EGraph& egraph, CanonicalizedType& ct)
|
||||||
parts.insert(parts.end(), ct.functionParts.begin(), ct.functionParts.end());
|
parts.insert(parts.end(), ct.functionParts.begin(), ct.functionParts.end());
|
||||||
parts.insert(parts.end(), ct.otherParts.begin(), ct.otherParts.end());
|
parts.insert(parts.end(), ct.otherParts.begin(), ct.otherParts.end());
|
||||||
|
|
||||||
return mkUnion(egraph, std::move(parts));
|
std::sort(parts.begin(), parts.end());
|
||||||
|
auto it = std::unique(parts.begin(), parts.end());
|
||||||
|
parts.erase(it, parts.end());
|
||||||
|
|
||||||
|
const size_t size = parts.size();
|
||||||
|
return {mkUnion(egraph, std::move(parts)), size};
|
||||||
}
|
}
|
||||||
|
|
||||||
void addChildren(const EGraph& egraph, const EType* enode, VecDeque<Id>& worklist)
|
void addChildren(const EGraph& egraph, const EType* enode, VecDeque<Id>& worklist)
|
||||||
|
@ -1522,7 +1554,7 @@ const Tag* Simplifier::isTag(Id id) const
|
||||||
{
|
{
|
||||||
for (const auto& node : get(id).nodes)
|
for (const auto& node : get(id).nodes)
|
||||||
{
|
{
|
||||||
if (const Tag* ty = node.get<Tag>())
|
if (const Tag* ty = node.node.get<Tag>())
|
||||||
return ty;
|
return ty;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1556,6 +1588,16 @@ void Simplifier::subst(Id from, Id to, const std::string& ruleName, const std::u
|
||||||
substs.emplace_back(from, to, desc);
|
substs.emplace_back(from, to, desc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Simplifier::subst(Id from, size_t boringIndex, Id to, const std::string& ruleName, const std::unordered_map<Id, size_t>& forceNodes)
|
||||||
|
{
|
||||||
|
std::string desc;
|
||||||
|
if (FFlag::DebugLuauLogSimplification)
|
||||||
|
desc = mkDesc(egraph, stringCache, arena, builtinTypes, from, to, forceNodes, ruleName);
|
||||||
|
|
||||||
|
egraph.markBoring(from, boringIndex);
|
||||||
|
substs.emplace_back(from, to, desc);
|
||||||
|
}
|
||||||
|
|
||||||
void Simplifier::unionClasses(std::vector<Id>& hereParts, Id there)
|
void Simplifier::unionClasses(std::vector<Id>& hereParts, Id there)
|
||||||
{
|
{
|
||||||
if (1 == hereParts.size() && isTag<TTopClass>(hereParts[0]))
|
if (1 == hereParts.size() && isTag<TTopClass>(hereParts[0]))
|
||||||
|
@ -1606,8 +1648,11 @@ void Simplifier::simplifyUnion(Id id)
|
||||||
for (Id part : u->operands())
|
for (Id part : u->operands())
|
||||||
unionWithType(egraph, canonicalized, find(part));
|
unionWithType(egraph, canonicalized, find(part));
|
||||||
|
|
||||||
Id resultId = fromCanonicalized(egraph, canonicalized);
|
const auto [resultId, newSize] = fromCanonicalized(egraph, canonicalized);
|
||||||
|
|
||||||
|
if (newSize < u->operands().size())
|
||||||
|
subst(id, unionIndex, resultId, "simplifyUnion", {{id, unionIndex}});
|
||||||
|
else
|
||||||
subst(id, resultId, "simplifyUnion", {{id, unionIndex}});
|
subst(id, resultId, "simplifyUnion", {{id, unionIndex}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1816,7 +1861,7 @@ void Simplifier::uninhabitedIntersection(Id id)
|
||||||
const auto& partNodes = egraph[partId].nodes;
|
const auto& partNodes = egraph[partId].nodes;
|
||||||
for (size_t partIndex = 0; partIndex < partNodes.size(); ++partIndex)
|
for (size_t partIndex = 0; partIndex < partNodes.size(); ++partIndex)
|
||||||
{
|
{
|
||||||
const EType& N = partNodes[partIndex];
|
const EType& N = partNodes[partIndex].node;
|
||||||
if (std::optional<EType> intersection = intersectOne(egraph, accumulator, &accumulatorNode, partId, &N))
|
if (std::optional<EType> intersection = intersectOne(egraph, accumulator, &accumulatorNode, partId, &N))
|
||||||
{
|
{
|
||||||
if (isTag<TNever>(*intersection))
|
if (isTag<TNever>(*intersection))
|
||||||
|
@ -1839,8 +1884,13 @@ void Simplifier::uninhabitedIntersection(Id id)
|
||||||
if ((unsimplified.empty() || !isTag<TUnknown>(accumulator)) && find(accumulator) != id)
|
if ((unsimplified.empty() || !isTag<TUnknown>(accumulator)) && find(accumulator) != id)
|
||||||
unsimplified.push_back(accumulator);
|
unsimplified.push_back(accumulator);
|
||||||
|
|
||||||
|
const bool isSmaller = unsimplified.size() < parts.size();
|
||||||
|
|
||||||
const Id result = mkIntersection(egraph, std::move(unsimplified));
|
const Id result = mkIntersection(egraph, std::move(unsimplified));
|
||||||
|
|
||||||
|
if (isSmaller)
|
||||||
|
subst(id, index, result, "uninhabitedIntersection", {{id, index}});
|
||||||
|
else
|
||||||
subst(id, result, "uninhabitedIntersection", {{id, index}});
|
subst(id, result, "uninhabitedIntersection", {{id, index}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1872,14 +1922,19 @@ void Simplifier::intersectWithNegatedClass(Id id)
|
||||||
const auto& iNodes = egraph[iId].nodes;
|
const auto& iNodes = egraph[iId].nodes;
|
||||||
for (size_t iIndex = 0; iIndex < iNodes.size(); ++iIndex)
|
for (size_t iIndex = 0; iIndex < iNodes.size(); ++iIndex)
|
||||||
{
|
{
|
||||||
const EType& iNode = iNodes[iIndex];
|
const EType& iNode = iNodes[iIndex].node;
|
||||||
if (isTag<TNil>(iNode) || isTag<TBoolean>(iNode) || isTag<TNumber>(iNode) || isTag<TString>(iNode) || isTag<TThread>(iNode) ||
|
if (isTag<TNil>(iNode) || isTag<TBoolean>(iNode) || isTag<TNumber>(iNode) || isTag<TString>(iNode) || isTag<TThread>(iNode) ||
|
||||||
isTag<TTopFunction>(iNode) ||
|
isTag<TTopFunction>(iNode) ||
|
||||||
// isTag<TTopTable>(iNode) || // I'm not sure about this one.
|
// isTag<TTopTable>(iNode) || // I'm not sure about this one.
|
||||||
isTag<SBoolean>(iNode) || isTag<SString>(iNode) || isTag<TFunction>(iNode) || isTag<TNever>(iNode))
|
isTag<SBoolean>(iNode) || isTag<SString>(iNode) || isTag<TFunction>(iNode) || isTag<TNever>(iNode))
|
||||||
{
|
{
|
||||||
// eg string & ~SomeClass
|
// 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;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1892,7 +1947,12 @@ void Simplifier::intersectWithNegatedClass(Id id)
|
||||||
// This cannot be meaningfully reduced.
|
// This cannot be meaningfully reduced.
|
||||||
continue;
|
continue;
|
||||||
case RightSuper:
|
case RightSuper:
|
||||||
subst(id, egraph.add(TNever{}), "intersectClassWithNegatedClass", {{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}});
|
subst(
|
||||||
|
id,
|
||||||
|
egraph.add(TNever{}),
|
||||||
|
"intersectClassWithNegatedClass",
|
||||||
|
{{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}}
|
||||||
|
);
|
||||||
return;
|
return;
|
||||||
case Unrelated:
|
case Unrelated:
|
||||||
// Part & ~Folder == Part
|
// Part & ~Folder == Part
|
||||||
|
@ -1905,8 +1965,13 @@ void Simplifier::intersectWithNegatedClass(Id id)
|
||||||
newParts.push_back(part);
|
newParts.push_back(part);
|
||||||
}
|
}
|
||||||
|
|
||||||
Id substId = egraph.add(Intersection{newParts.begin(), newParts.end()});
|
Id substId = mkIntersection(egraph, newParts);
|
||||||
subst(id, substId, "intersectClassWithNegatedClass", {{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}});
|
subst(
|
||||||
|
id,
|
||||||
|
substId,
|
||||||
|
"intersectClassWithNegatedClass",
|
||||||
|
{{id, intersectionIndex}, {iId, iIndex}, {jId, negationIndex}, {negated, negatedClassIndex}}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1942,7 +2007,7 @@ void Simplifier::intersectWithNegatedAtom(Id id)
|
||||||
{
|
{
|
||||||
for (size_t negationOperandIndex = 0; negationOperandIndex < egraph[negation->operands()[0]].nodes.size(); ++negationOperandIndex)
|
for (size_t negationOperandIndex = 0; negationOperandIndex < egraph[negation->operands()[0]].nodes.size(); ++negationOperandIndex)
|
||||||
{
|
{
|
||||||
const EType* negationOperand = &egraph[negation->operands()[0]].nodes[negationOperandIndex];
|
const EType* negationOperand = &egraph[negation->operands()[0]].nodes[negationOperandIndex].node;
|
||||||
if (!isTerminal(*negationOperand) || negationOperand->get<TOpaque>())
|
if (!isTerminal(*negationOperand) || negationOperand->get<TOpaque>())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -1953,7 +2018,7 @@ void Simplifier::intersectWithNegatedAtom(Id id)
|
||||||
|
|
||||||
for (size_t jNodeIndex = 0; jNodeIndex < egraph[intersectionOperands[j]].nodes.size(); ++jNodeIndex)
|
for (size_t jNodeIndex = 0; jNodeIndex < egraph[intersectionOperands[j]].nodes.size(); ++jNodeIndex)
|
||||||
{
|
{
|
||||||
const EType* jNode = &egraph[intersectionOperands[j]].nodes[jNodeIndex];
|
const EType* jNode = &egraph[intersectionOperands[j]].nodes[jNodeIndex].node;
|
||||||
if (!isTerminal(*jNode) || jNode->get<TOpaque>())
|
if (!isTerminal(*jNode) || jNode->get<TOpaque>())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
@ -1978,7 +2043,7 @@ void Simplifier::intersectWithNegatedAtom(Id id)
|
||||||
|
|
||||||
subst(
|
subst(
|
||||||
id,
|
id,
|
||||||
egraph.add(Intersection{newOperands}),
|
mkIntersection(egraph, std::move(newOperands)),
|
||||||
"intersectWithNegatedAtom",
|
"intersectWithNegatedAtom",
|
||||||
{{id, intersectionIndex}, {intersectionOperands[i], negationIndex}, {intersectionOperands[j], jNodeIndex}}
|
{{id, intersectionIndex}, {intersectionOperands[i], negationIndex}, {intersectionOperands[j], jNodeIndex}}
|
||||||
);
|
);
|
||||||
|
@ -2155,7 +2220,7 @@ void Simplifier::expandNegation(Id id)
|
||||||
if (!ok)
|
if (!ok)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
subst(id, fromCanonicalized(egraph, canonicalized), "expandNegation", {{id, index}});
|
subst(id, fromCanonicalized(egraph, canonicalized).first, "expandNegation", {{id, index}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2553,9 +2618,9 @@ std::optional<EqSatSimplificationResult> eqSatSimplify(NotNull<Simplifier> simpl
|
||||||
// try to run any rules on it.
|
// try to run any rules on it.
|
||||||
bool shouldAbort = false;
|
bool shouldAbort = false;
|
||||||
|
|
||||||
for (const EType& enode : egraph[id].nodes)
|
for (const auto& enode : egraph[id].nodes)
|
||||||
{
|
{
|
||||||
if (isTerminal(enode))
|
if (isTerminal(enode.node))
|
||||||
{
|
{
|
||||||
shouldAbort = true;
|
shouldAbort = true;
|
||||||
break;
|
break;
|
||||||
|
@ -2565,8 +2630,8 @@ std::optional<EqSatSimplificationResult> eqSatSimplify(NotNull<Simplifier> simpl
|
||||||
if (shouldAbort)
|
if (shouldAbort)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
for (const EType& enode : egraph[id].nodes)
|
for (const auto& enode : egraph[id].nodes)
|
||||||
addChildren(egraph, &enode, worklist);
|
addChildren(egraph, &enode.node, worklist);
|
||||||
|
|
||||||
for (Simplifier::RewriteRuleFn rule : rules)
|
for (Simplifier::RewriteRuleFn rule : rules)
|
||||||
(simplifier.get()->*rule)(id);
|
(simplifier.get()->*rule)(id);
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
#include "Luau/Autocomplete.h"
|
#include "Luau/Autocomplete.h"
|
||||||
#include "Luau/Common.h"
|
#include "Luau/Common.h"
|
||||||
#include "Luau/EqSatSimplification.h"
|
#include "Luau/EqSatSimplification.h"
|
||||||
|
#include "Luau/ModuleResolver.h"
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/ParseOptions.h"
|
#include "Luau/ParseOptions.h"
|
||||||
#include "Luau/Module.h"
|
#include "Luau/Module.h"
|
||||||
|
@ -19,16 +20,21 @@
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/ParseOptions.h"
|
#include "Luau/ParseOptions.h"
|
||||||
#include "Luau/Module.h"
|
#include "Luau/Module.h"
|
||||||
|
#include "Luau/Clone.h"
|
||||||
#include "AutocompleteCore.h"
|
#include "AutocompleteCore.h"
|
||||||
|
|
||||||
|
|
||||||
LUAU_FASTINT(LuauTypeInferRecursionLimit);
|
LUAU_FASTINT(LuauTypeInferRecursionLimit);
|
||||||
LUAU_FASTINT(LuauTypeInferIterationLimit);
|
LUAU_FASTINT(LuauTypeInferIterationLimit);
|
||||||
LUAU_FASTINT(LuauTarjanChildLimit)
|
LUAU_FASTINT(LuauTarjanChildLimit)
|
||||||
LUAU_FASTFLAG(LuauAllowFragmentParsing);
|
LUAU_FASTFLAG(LuauAllowFragmentParsing);
|
||||||
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauIncrementalAutocompleteBugfixes)
|
||||||
|
LUAU_FASTFLAG(LuauReferenceAllocatorInNewSolver)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauMixedModeDefFinderTraversesTypeOf)
|
||||||
|
LUAU_FASTFLAG(LuauBetterReverseDependencyTracking)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauCloneIncrementalModule)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LogFragmentsFromAutocomplete)
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
template<typename T>
|
template<typename T>
|
||||||
|
@ -49,6 +55,96 @@ void copyModuleMap(Luau::DenseHashMap<K, V>& result, const Luau::DenseHashMap<K,
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
template<typename K, typename V>
|
||||||
|
void cloneModuleMap(TypeArena& destArena, CloneState& cloneState, const Luau::DenseHashMap<K, V>& source, Luau::DenseHashMap<K, V>& dest)
|
||||||
|
{
|
||||||
|
for (auto [k, v] : source)
|
||||||
|
{
|
||||||
|
dest[k] = Luau::clone(v, destArena, cloneState);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct MixedModeIncrementalTCDefFinder : public AstVisitor
|
||||||
|
{
|
||||||
|
bool visit(AstExprLocal* local) override
|
||||||
|
{
|
||||||
|
referencedLocalDefs.emplace_back(local->local, local);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool visit(AstTypeTypeof* node) override
|
||||||
|
{
|
||||||
|
// We need to traverse typeof expressions because they may refer to locals that we need
|
||||||
|
// to populate the local environment for fragment typechecking. For example, `typeof(m)`
|
||||||
|
// requires that we find the local/global `m` and place it in the environment.
|
||||||
|
// The default behaviour here is to return false, and have individual visitors override
|
||||||
|
// the specific behaviour they need.
|
||||||
|
return FFlag::LuauMixedModeDefFinderTraversesTypeOf;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ast defs is just a mapping from expr -> def in general
|
||||||
|
// will get built up by the dfg builder
|
||||||
|
|
||||||
|
// localDefs, we need to copy over
|
||||||
|
std::vector<std::pair<AstLocal*, AstExpr*>> referencedLocalDefs;
|
||||||
|
};
|
||||||
|
|
||||||
|
void cloneAndSquashScopes(
|
||||||
|
CloneState& cloneState,
|
||||||
|
const Scope* staleScope,
|
||||||
|
const ModulePtr& staleModule,
|
||||||
|
NotNull<TypeArena> destArena,
|
||||||
|
NotNull<DataFlowGraph> dfg,
|
||||||
|
AstStatBlock* program,
|
||||||
|
Scope* destScope
|
||||||
|
)
|
||||||
|
{
|
||||||
|
std::vector<const Scope*> scopes;
|
||||||
|
for (const Scope* current = staleScope; current; current = current->parent.get())
|
||||||
|
{
|
||||||
|
scopes.emplace_back(current);
|
||||||
|
}
|
||||||
|
|
||||||
|
// in reverse order (we need to clone the parents and override defs as we go down the list)
|
||||||
|
for (auto it = scopes.rbegin(); it != scopes.rend(); ++it)
|
||||||
|
{
|
||||||
|
const Scope* curr = *it;
|
||||||
|
// Clone the lvalue types
|
||||||
|
for (const auto& [def, ty] : curr->lvalueTypes)
|
||||||
|
destScope->lvalueTypes[def] = Luau::clone(ty, *destArena, cloneState);
|
||||||
|
// Clone the rvalueRefinements
|
||||||
|
for (const auto& [def, ty] : curr->rvalueRefinements)
|
||||||
|
destScope->rvalueRefinements[def] = Luau::clone(ty, *destArena, cloneState);
|
||||||
|
for (const auto& [n, m] : curr->importedTypeBindings)
|
||||||
|
{
|
||||||
|
std::unordered_map<Name, TypeFun> importedBindingTypes;
|
||||||
|
for (const auto& [v, tf] : m)
|
||||||
|
importedBindingTypes[v] = Luau::clone(tf, *destArena, cloneState);
|
||||||
|
destScope->importedTypeBindings[n] = m;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Finally, clone up the bindings
|
||||||
|
for (const auto& [s, b] : curr->bindings)
|
||||||
|
{
|
||||||
|
destScope->bindings[s] = Luau::clone(b, *destArena, cloneState);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// The above code associates defs with TypeId's in the scope
|
||||||
|
// so that lookup to locals will succeed.
|
||||||
|
MixedModeIncrementalTCDefFinder finder;
|
||||||
|
program->visit(&finder);
|
||||||
|
std::vector<std::pair<AstLocal*, AstExpr*>> locals = std::move(finder.referencedLocalDefs);
|
||||||
|
for (auto [loc, expr] : locals)
|
||||||
|
{
|
||||||
|
if (std::optional<Binding> binding = staleScope->linearSearchForBinding(loc->name.value, true))
|
||||||
|
{
|
||||||
|
destScope->lvalueTypes[dfg->getDef(expr)] = Luau::clone(binding->typeId, *destArena, cloneState);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
static FrontendModuleResolver& getModuleResolver(Frontend& frontend, std::optional<FrontendOptions> options)
|
static FrontendModuleResolver& getModuleResolver(Frontend& frontend, std::optional<FrontendOptions> options)
|
||||||
{
|
{
|
||||||
if (FFlag::LuauSolverV2 || !options)
|
if (FFlag::LuauSolverV2 || !options)
|
||||||
|
@ -200,7 +296,7 @@ ScopePtr findClosestScope(const ModulePtr& module, const AstStat* nearestStateme
|
||||||
return closest;
|
return closest;
|
||||||
}
|
}
|
||||||
|
|
||||||
FragmentParseResult parseFragment(
|
std::optional<FragmentParseResult> parseFragment(
|
||||||
const SourceModule& srcModule,
|
const SourceModule& srcModule,
|
||||||
std::string_view src,
|
std::string_view src,
|
||||||
const Position& cursorPos,
|
const Position& cursorPos,
|
||||||
|
@ -239,12 +335,17 @@ FragmentParseResult parseFragment(
|
||||||
FragmentParseResult fragmentResult;
|
FragmentParseResult fragmentResult;
|
||||||
fragmentResult.fragmentToParse = std::string(dbg.data(), parseLength);
|
fragmentResult.fragmentToParse = std::string(dbg.data(), parseLength);
|
||||||
// For the duration of the incremental parse, we want to allow the name table to re-use duplicate names
|
// For the duration of the incremental parse, we want to allow the name table to re-use duplicate names
|
||||||
|
if (FFlag::LogFragmentsFromAutocomplete)
|
||||||
|
logLuau(dbg);
|
||||||
|
|
||||||
ParseOptions opts;
|
ParseOptions opts;
|
||||||
opts.allowDeclarationSyntax = false;
|
opts.allowDeclarationSyntax = false;
|
||||||
opts.captureComments = true;
|
opts.captureComments = true;
|
||||||
opts.parseFragment = FragmentParseResumeSettings{std::move(result.localMap), std::move(result.localStack), startPos};
|
opts.parseFragment = FragmentParseResumeSettings{std::move(result.localMap), std::move(result.localStack), startPos};
|
||||||
ParseResult p = Luau::Parser::parse(srcStart, parseLength, *nameTbl, *fragmentResult.alloc.get(), opts);
|
ParseResult p = Luau::Parser::parse(srcStart, parseLength, *nameTbl, *fragmentResult.alloc.get(), opts);
|
||||||
|
// This means we threw a ParseError and we should decline to offer autocomplete here.
|
||||||
|
if (p.root == nullptr)
|
||||||
|
return std::nullopt;
|
||||||
|
|
||||||
std::vector<AstNode*> fabricatedAncestry = std::move(result.ancestry);
|
std::vector<AstNode*> fabricatedAncestry = std::move(result.ancestry);
|
||||||
|
|
||||||
|
@ -258,16 +359,39 @@ FragmentParseResult parseFragment(
|
||||||
fragmentResult.root = std::move(p.root);
|
fragmentResult.root = std::move(p.root);
|
||||||
fragmentResult.ancestry = std::move(fabricatedAncestry);
|
fragmentResult.ancestry = std::move(fabricatedAncestry);
|
||||||
fragmentResult.nearestStatement = nearestStatement;
|
fragmentResult.nearestStatement = nearestStatement;
|
||||||
|
fragmentResult.commentLocations = std::move(p.commentLocations);
|
||||||
return fragmentResult;
|
return fragmentResult;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ModulePtr cloneModule(CloneState& cloneState, const ModulePtr& source, std::unique_ptr<Allocator> alloc)
|
||||||
|
{
|
||||||
|
freeze(source->internalTypes);
|
||||||
|
freeze(source->interfaceTypes);
|
||||||
|
ModulePtr incremental = std::make_shared<Module>();
|
||||||
|
incremental->name = source->name;
|
||||||
|
incremental->humanReadableName = source->humanReadableName;
|
||||||
|
incremental->allocator = std::move(alloc);
|
||||||
|
// Clone types
|
||||||
|
cloneModuleMap(incremental->internalTypes, cloneState, source->astTypes, incremental->astTypes);
|
||||||
|
cloneModuleMap(incremental->internalTypes, cloneState, source->astTypePacks, incremental->astTypePacks);
|
||||||
|
cloneModuleMap(incremental->internalTypes, cloneState, source->astExpectedTypes, incremental->astExpectedTypes);
|
||||||
|
|
||||||
|
cloneModuleMap(incremental->internalTypes, cloneState, source->astOverloadResolvedTypes, incremental->astOverloadResolvedTypes);
|
||||||
|
|
||||||
|
cloneModuleMap(incremental->internalTypes, cloneState, source->astForInNextTypes, incremental->astForInNextTypes);
|
||||||
|
|
||||||
|
copyModuleMap(incremental->astScopes, source->astScopes);
|
||||||
|
|
||||||
|
return incremental;
|
||||||
|
}
|
||||||
|
|
||||||
ModulePtr copyModule(const ModulePtr& result, std::unique_ptr<Allocator> alloc)
|
ModulePtr copyModule(const ModulePtr& result, std::unique_ptr<Allocator> alloc)
|
||||||
{
|
{
|
||||||
freeze(result->internalTypes);
|
|
||||||
freeze(result->interfaceTypes);
|
|
||||||
ModulePtr incrementalModule = std::make_shared<Module>();
|
ModulePtr incrementalModule = std::make_shared<Module>();
|
||||||
incrementalModule->name = result->name;
|
incrementalModule->name = result->name;
|
||||||
incrementalModule->humanReadableName = result->humanReadableName;
|
incrementalModule->humanReadableName = "Incremental$" + result->humanReadableName;
|
||||||
|
incrementalModule->internalTypes.owningModule = incrementalModule.get();
|
||||||
|
incrementalModule->interfaceTypes.owningModule = incrementalModule.get();
|
||||||
incrementalModule->allocator = std::move(alloc);
|
incrementalModule->allocator = std::move(alloc);
|
||||||
// Don't need to keep this alive (it's already on the source module)
|
// Don't need to keep this alive (it's already on the source module)
|
||||||
copyModuleVec(incrementalModule->scopes, result->scopes);
|
copyModuleVec(incrementalModule->scopes, result->scopes);
|
||||||
|
@ -286,21 +410,6 @@ ModulePtr copyModule(const ModulePtr& result, std::unique_ptr<Allocator> alloc)
|
||||||
return incrementalModule;
|
return incrementalModule;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct MixedModeIncrementalTCDefFinder : public AstVisitor
|
|
||||||
{
|
|
||||||
bool visit(AstExprLocal* local) override
|
|
||||||
{
|
|
||||||
referencedLocalDefs.push_back({local->local, local});
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// ast defs is just a mapping from expr -> def in general
|
|
||||||
// will get built up by the dfg builder
|
|
||||||
|
|
||||||
// localDefs, we need to copy over
|
|
||||||
std::vector<std::pair<AstLocal*, AstExpr*>> referencedLocalDefs;
|
|
||||||
};
|
|
||||||
|
|
||||||
void mixedModeCompatibility(
|
void mixedModeCompatibility(
|
||||||
const ScopePtr& bottomScopeStale,
|
const ScopePtr& bottomScopeStale,
|
||||||
const ScopePtr& myFakeScope,
|
const ScopePtr& myFakeScope,
|
||||||
|
@ -339,7 +448,9 @@ FragmentTypeCheckResult typecheckFragment_(
|
||||||
{
|
{
|
||||||
freeze(stale->internalTypes);
|
freeze(stale->internalTypes);
|
||||||
freeze(stale->interfaceTypes);
|
freeze(stale->interfaceTypes);
|
||||||
ModulePtr incrementalModule = copyModule(stale, std::move(astAllocator));
|
CloneState cloneState{frontend.builtinTypes};
|
||||||
|
ModulePtr incrementalModule =
|
||||||
|
FFlag::LuauCloneIncrementalModule ? cloneModule(cloneState, stale, std::move(astAllocator)) : copyModule(stale, std::move(astAllocator));
|
||||||
incrementalModule->checkedInNewSolver = true;
|
incrementalModule->checkedInNewSolver = true;
|
||||||
unfreeze(incrementalModule->internalTypes);
|
unfreeze(incrementalModule->internalTypes);
|
||||||
unfreeze(incrementalModule->interfaceTypes);
|
unfreeze(incrementalModule->interfaceTypes);
|
||||||
|
@ -366,7 +477,8 @@ FragmentTypeCheckResult typecheckFragment_(
|
||||||
TypeFunctionRuntime typeFunctionRuntime(iceHandler, NotNull{&limits});
|
TypeFunctionRuntime typeFunctionRuntime(iceHandler, NotNull{&limits});
|
||||||
|
|
||||||
/// Create a DataFlowGraph just for the surrounding context
|
/// Create a DataFlowGraph just for the surrounding context
|
||||||
auto dfg = DataFlowGraphBuilder::build(root, iceHandler);
|
DataFlowGraph dfg = DataFlowGraphBuilder::build(root, NotNull{&incrementalModule->defArena}, NotNull{&incrementalModule->keyArena}, iceHandler);
|
||||||
|
|
||||||
SimplifierPtr simplifier = newSimplifier(NotNull{&incrementalModule->internalTypes}, frontend.builtinTypes);
|
SimplifierPtr simplifier = newSimplifier(NotNull{&incrementalModule->internalTypes}, frontend.builtinTypes);
|
||||||
|
|
||||||
FrontendModuleResolver& resolver = getModuleResolver(frontend, opts);
|
FrontendModuleResolver& resolver = getModuleResolver(frontend, opts);
|
||||||
|
@ -386,25 +498,34 @@ FragmentTypeCheckResult typecheckFragment_(
|
||||||
NotNull{&dfg},
|
NotNull{&dfg},
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
std::shared_ptr<Scope> freshChildOfNearestScope = nullptr;
|
||||||
cg.rootScope = stale->getModuleScope().get();
|
if (FFlag::LuauCloneIncrementalModule)
|
||||||
// Any additions to the scope must occur in a fresh scope
|
{
|
||||||
auto freshChildOfNearestScope = std::make_shared<Scope>(closestScope);
|
freshChildOfNearestScope = std::make_shared<Scope>(closestScope);
|
||||||
incrementalModule->scopes.emplace_back(root->location, freshChildOfNearestScope);
|
incrementalModule->scopes.emplace_back(root->location, freshChildOfNearestScope);
|
||||||
|
cg.rootScope = freshChildOfNearestScope.get();
|
||||||
|
|
||||||
// Update freshChildOfNearestScope with the appropriate lvalueTypes
|
cloneAndSquashScopes(
|
||||||
|
cloneState, closestScope.get(), stale, NotNull{&incrementalModule->internalTypes}, NotNull{&dfg}, root, freshChildOfNearestScope.get()
|
||||||
|
);
|
||||||
|
cg.visitFragmentRoot(freshChildOfNearestScope, root);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Any additions to the scope must occur in a fresh scope
|
||||||
|
cg.rootScope = stale->getModuleScope().get();
|
||||||
|
freshChildOfNearestScope = std::make_shared<Scope>(closestScope);
|
||||||
|
incrementalModule->scopes.emplace_back(root->location, freshChildOfNearestScope);
|
||||||
mixedModeCompatibility(closestScope, freshChildOfNearestScope, stale, NotNull{&dfg}, root);
|
mixedModeCompatibility(closestScope, freshChildOfNearestScope, stale, NotNull{&dfg}, root);
|
||||||
|
|
||||||
// closest Scope -> children = { ...., freshChildOfNearestScope}
|
// closest Scope -> children = { ...., freshChildOfNearestScope}
|
||||||
// We need to trim nearestChild from the scope hierarcy
|
// We need to trim nearestChild from the scope hierarcy
|
||||||
closestScope->children.push_back(NotNull{freshChildOfNearestScope.get()});
|
closestScope->children.emplace_back(freshChildOfNearestScope.get());
|
||||||
// Visit just the root - we know the scope it should be in
|
|
||||||
cg.visitFragmentRoot(freshChildOfNearestScope, root);
|
cg.visitFragmentRoot(freshChildOfNearestScope, root);
|
||||||
// Trim nearestChild from the closestScope
|
// Trim nearestChild from the closestScope
|
||||||
Scope* back = closestScope->children.back().get();
|
Scope* back = closestScope->children.back().get();
|
||||||
LUAU_ASSERT(back == freshChildOfNearestScope.get());
|
LUAU_ASSERT(back == freshChildOfNearestScope.get());
|
||||||
closestScope->children.pop_back();
|
closestScope->children.pop_back();
|
||||||
|
}
|
||||||
|
|
||||||
/// Initialize the constraint solver and run it
|
/// Initialize the constraint solver and run it
|
||||||
ConstraintSolver cs{
|
ConstraintSolver cs{
|
||||||
|
@ -444,7 +565,7 @@ FragmentTypeCheckResult typecheckFragment_(
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
FragmentTypeCheckResult typecheckFragment(
|
std::pair<FragmentTypeCheckStatus, FragmentTypeCheckResult> typecheckFragment(
|
||||||
Frontend& frontend,
|
Frontend& frontend,
|
||||||
const ModuleName& moduleName,
|
const ModuleName& moduleName,
|
||||||
const Position& cursorPos,
|
const Position& cursorPos,
|
||||||
|
@ -453,6 +574,13 @@ FragmentTypeCheckResult typecheckFragment(
|
||||||
std::optional<Position> fragmentEndPosition
|
std::optional<Position> fragmentEndPosition
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
if (FFlag::LuauBetterReverseDependencyTracking)
|
||||||
|
{
|
||||||
|
if (!frontend.allModuleDependenciesValid(moduleName, opts && opts->forAutocomplete))
|
||||||
|
return {FragmentTypeCheckStatus::SkipAutocomplete, {}};
|
||||||
|
}
|
||||||
|
|
||||||
const SourceModule* sourceModule = frontend.getSourceModule(moduleName);
|
const SourceModule* sourceModule = frontend.getSourceModule(moduleName);
|
||||||
if (!sourceModule)
|
if (!sourceModule)
|
||||||
{
|
{
|
||||||
|
@ -468,13 +596,30 @@ FragmentTypeCheckResult typecheckFragment(
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
FragmentParseResult parseResult = parseFragment(*sourceModule, src, cursorPos, fragmentEndPosition);
|
if (FFlag::LuauIncrementalAutocompleteBugfixes && FFlag::LuauReferenceAllocatorInNewSolver)
|
||||||
|
{
|
||||||
|
if (sourceModule->allocator.get() != module->allocator.get())
|
||||||
|
{
|
||||||
|
return {FragmentTypeCheckStatus::SkipAutocomplete, {}};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
auto tryParse = parseFragment(*sourceModule, src, cursorPos, fragmentEndPosition);
|
||||||
|
|
||||||
|
if (!tryParse)
|
||||||
|
return {FragmentTypeCheckStatus::SkipAutocomplete, {}};
|
||||||
|
|
||||||
|
FragmentParseResult& parseResult = *tryParse;
|
||||||
|
|
||||||
|
if (isWithinComment(parseResult.commentLocations, fragmentEndPosition.value_or(cursorPos)))
|
||||||
|
return {FragmentTypeCheckStatus::SkipAutocomplete, {}};
|
||||||
|
|
||||||
FrontendOptions frontendOptions = opts.value_or(frontend.options);
|
FrontendOptions frontendOptions = opts.value_or(frontend.options);
|
||||||
const ScopePtr& closestScope = findClosestScope(module, parseResult.nearestStatement);
|
const ScopePtr& closestScope = findClosestScope(module, parseResult.nearestStatement);
|
||||||
FragmentTypeCheckResult result =
|
FragmentTypeCheckResult result =
|
||||||
typecheckFragment_(frontend, parseResult.root, module, closestScope, cursorPos, std::move(parseResult.alloc), frontendOptions);
|
typecheckFragment_(frontend, parseResult.root, module, closestScope, cursorPos, std::move(parseResult.alloc), frontendOptions);
|
||||||
result.ancestry = std::move(parseResult.ancestry);
|
result.ancestry = std::move(parseResult.ancestry);
|
||||||
return result;
|
return {FragmentTypeCheckStatus::Success, result};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -498,9 +643,17 @@ FragmentAutocompleteResult fragmentAutocomplete(
|
||||||
return {};
|
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
|
||||||
auto globalScope = (opts && opts->forAutocomplete) ? frontend.globalsForAutocomplete.globalScope.get() : frontend.globals.globalScope.get();
|
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();
|
||||||
|
if (FFlag::LogFragmentsFromAutocomplete)
|
||||||
|
logLuau(src);
|
||||||
TypeArena arenaForFragmentAutocomplete;
|
TypeArena arenaForFragmentAutocomplete;
|
||||||
auto result = Luau::autocomplete_(
|
auto result = Luau::autocomplete_(
|
||||||
tcResult.incrementalModule,
|
tcResult.incrementalModule,
|
||||||
|
|
|
@ -13,6 +13,7 @@
|
||||||
#include "Luau/EqSatSimplification.h"
|
#include "Luau/EqSatSimplification.h"
|
||||||
#include "Luau/FileResolver.h"
|
#include "Luau/FileResolver.h"
|
||||||
#include "Luau/NonStrictTypeChecker.h"
|
#include "Luau/NonStrictTypeChecker.h"
|
||||||
|
#include "Luau/NotNull.h"
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/Scope.h"
|
#include "Luau/Scope.h"
|
||||||
#include "Luau/StringUtils.h"
|
#include "Luau/StringUtils.h"
|
||||||
|
@ -38,7 +39,6 @@ LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
||||||
LUAU_FASTINT(LuauTarjanChildLimit)
|
LUAU_FASTINT(LuauTarjanChildLimit)
|
||||||
LUAU_FASTFLAG(LuauInferInNoCheckMode)
|
LUAU_FASTFLAG(LuauInferInNoCheckMode)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauKnowsTheDataModel3)
|
LUAU_FASTFLAGVARIABLE(LuauKnowsTheDataModel3)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauStoreCommentsForDefinitionFiles)
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauLogSolverToJson)
|
LUAU_FASTFLAGVARIABLE(DebugLuauLogSolverToJson)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauLogSolverToJsonFile)
|
LUAU_FASTFLAGVARIABLE(DebugLuauLogSolverToJsonFile)
|
||||||
|
@ -47,10 +47,13 @@ LUAU_FASTFLAGVARIABLE(DebugLuauForceStrictMode)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauForceNonStrictMode)
|
LUAU_FASTFLAGVARIABLE(DebugLuauForceNonStrictMode)
|
||||||
LUAU_DYNAMIC_FASTFLAGVARIABLE(LuauRunCustomModuleChecks, false)
|
LUAU_DYNAMIC_FASTFLAGVARIABLE(LuauRunCustomModuleChecks, false)
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauBetterReverseDependencyTracking)
|
||||||
|
|
||||||
LUAU_FASTFLAG(StudioReportLuauAny2)
|
LUAU_FASTFLAG(StudioReportLuauAny2)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauStoreSolverTypeOnModule)
|
LUAU_FASTFLAGVARIABLE(LuauStoreSolverTypeOnModule)
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauReferenceAllocatorInNewSolver)
|
LUAU_FASTFLAGVARIABLE(LuauReferenceAllocatorInNewSolver)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauSelectivelyRetainDFGArena)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -137,7 +140,7 @@ static ParseResult parseSourceForModule(std::string_view source, Luau::SourceMod
|
||||||
sourceModule.root = parseResult.root;
|
sourceModule.root = parseResult.root;
|
||||||
sourceModule.mode = Mode::Definition;
|
sourceModule.mode = Mode::Definition;
|
||||||
|
|
||||||
if (FFlag::LuauStoreCommentsForDefinitionFiles && options.captureComments)
|
if (options.captureComments)
|
||||||
{
|
{
|
||||||
sourceModule.hotcomments = parseResult.hotcomments;
|
sourceModule.hotcomments = parseResult.hotcomments;
|
||||||
sourceModule.commentLocations = parseResult.commentLocations;
|
sourceModule.commentLocations = parseResult.commentLocations;
|
||||||
|
@ -819,6 +822,16 @@ bool Frontend::parseGraph(
|
||||||
topseen = Permanent;
|
topseen = Permanent;
|
||||||
|
|
||||||
buildQueue.push_back(top->name);
|
buildQueue.push_back(top->name);
|
||||||
|
|
||||||
|
if (FFlag::LuauBetterReverseDependencyTracking)
|
||||||
|
{
|
||||||
|
// at this point we know all valid dependencies are processed into SourceNodes
|
||||||
|
for (const ModuleName& dep : top->requireSet)
|
||||||
|
{
|
||||||
|
if (auto it = sourceNodes.find(dep); it != sourceNodes.end())
|
||||||
|
it->second->dependents.insert(top->name);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -1048,6 +1061,11 @@ void Frontend::checkBuildQueueItem(BuildQueueItem& item)
|
||||||
freeze(module->interfaceTypes);
|
freeze(module->interfaceTypes);
|
||||||
|
|
||||||
module->internalTypes.clear();
|
module->internalTypes.clear();
|
||||||
|
if (FFlag::LuauSelectivelyRetainDFGArena)
|
||||||
|
{
|
||||||
|
module->defArena.allocator.clear();
|
||||||
|
module->keyArena.allocator.clear();
|
||||||
|
}
|
||||||
|
|
||||||
module->astTypes.clear();
|
module->astTypes.clear();
|
||||||
module->astTypePacks.clear();
|
module->astTypePacks.clear();
|
||||||
|
@ -1101,6 +1119,39 @@ void Frontend::recordItemResult(const BuildQueueItem& item)
|
||||||
if (item.exception)
|
if (item.exception)
|
||||||
std::rethrow_exception(item.exception);
|
std::rethrow_exception(item.exception);
|
||||||
|
|
||||||
|
if (FFlag::LuauBetterReverseDependencyTracking)
|
||||||
|
{
|
||||||
|
bool replacedModule = false;
|
||||||
|
if (item.options.forAutocomplete)
|
||||||
|
{
|
||||||
|
replacedModule = moduleResolverForAutocomplete.setModule(item.name, item.module);
|
||||||
|
item.sourceNode->dirtyModuleForAutocomplete = false;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
replacedModule = moduleResolver.setModule(item.name, item.module);
|
||||||
|
item.sourceNode->dirtyModule = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (replacedModule)
|
||||||
|
{
|
||||||
|
LUAU_TIMETRACE_SCOPE("Frontend::invalidateDependentModules", "Frontend");
|
||||||
|
LUAU_TIMETRACE_ARGUMENT("name", item.name.c_str());
|
||||||
|
traverseDependents(
|
||||||
|
item.name,
|
||||||
|
[forAutocomplete = item.options.forAutocomplete](SourceNode& sourceNode)
|
||||||
|
{
|
||||||
|
bool traverseSubtree = !sourceNode.hasInvalidModuleDependency(forAutocomplete);
|
||||||
|
sourceNode.setInvalidModuleDependency(true, forAutocomplete);
|
||||||
|
return traverseSubtree;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
item.sourceNode->setInvalidModuleDependency(false, item.options.forAutocomplete);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (item.options.forAutocomplete)
|
if (item.options.forAutocomplete)
|
||||||
{
|
{
|
||||||
moduleResolverForAutocomplete.setModule(item.name, item.module);
|
moduleResolverForAutocomplete.setModule(item.name, item.module);
|
||||||
|
@ -1111,6 +1162,7 @@ void Frontend::recordItemResult(const BuildQueueItem& item)
|
||||||
moduleResolver.setModule(item.name, item.module);
|
moduleResolver.setModule(item.name, item.module);
|
||||||
item.sourceNode->dirtyModule = false;
|
item.sourceNode->dirtyModule = false;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
stats.timeCheck += item.stats.timeCheck;
|
stats.timeCheck += item.stats.timeCheck;
|
||||||
stats.timeLint += item.stats.timeLint;
|
stats.timeLint += item.stats.timeLint;
|
||||||
|
@ -1146,6 +1198,13 @@ ScopePtr Frontend::getModuleEnvironment(const SourceModule& module, const Config
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Frontend::allModuleDependenciesValid(const ModuleName& name, bool forAutocomplete) const
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(FFlag::LuauBetterReverseDependencyTracking);
|
||||||
|
auto it = sourceNodes.find(name);
|
||||||
|
return it != sourceNodes.end() && !it->second->hasInvalidModuleDependency(forAutocomplete);
|
||||||
|
}
|
||||||
|
|
||||||
bool Frontend::isDirty(const ModuleName& name, bool forAutocomplete) const
|
bool Frontend::isDirty(const ModuleName& name, bool forAutocomplete) const
|
||||||
{
|
{
|
||||||
auto it = sourceNodes.find(name);
|
auto it = sourceNodes.find(name);
|
||||||
|
@ -1160,6 +1219,31 @@ bool Frontend::isDirty(const ModuleName& name, bool forAutocomplete) const
|
||||||
*/
|
*/
|
||||||
void Frontend::markDirty(const ModuleName& name, std::vector<ModuleName>* markedDirty)
|
void Frontend::markDirty(const ModuleName& name, std::vector<ModuleName>* markedDirty)
|
||||||
{
|
{
|
||||||
|
LUAU_TIMETRACE_SCOPE("Frontend::markDirty", "Frontend");
|
||||||
|
LUAU_TIMETRACE_ARGUMENT("name", name.c_str());
|
||||||
|
|
||||||
|
if (FFlag::LuauBetterReverseDependencyTracking)
|
||||||
|
{
|
||||||
|
traverseDependents(
|
||||||
|
name,
|
||||||
|
[markedDirty](SourceNode& sourceNode)
|
||||||
|
{
|
||||||
|
if (markedDirty)
|
||||||
|
markedDirty->push_back(sourceNode.name);
|
||||||
|
|
||||||
|
if (sourceNode.dirtySourceModule && sourceNode.dirtyModule && sourceNode.dirtyModuleForAutocomplete)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
sourceNode.dirtySourceModule = true;
|
||||||
|
sourceNode.dirtyModule = true;
|
||||||
|
sourceNode.dirtyModuleForAutocomplete = true;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (sourceNodes.count(name) == 0)
|
if (sourceNodes.count(name) == 0)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -1198,6 +1282,33 @@ void Frontend::markDirty(const ModuleName& name, std::vector<ModuleName>* marked
|
||||||
const std::vector<ModuleName>& dependents = reverseDeps[next];
|
const std::vector<ModuleName>& dependents = reverseDeps[next];
|
||||||
queue.insert(queue.end(), dependents.begin(), dependents.end());
|
queue.insert(queue.end(), dependents.begin(), dependents.end());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Frontend::traverseDependents(const ModuleName& name, std::function<bool(SourceNode&)> processSubtree)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(FFlag::LuauBetterReverseDependencyTracking);
|
||||||
|
LUAU_TIMETRACE_SCOPE("Frontend::traverseDependents", "Frontend");
|
||||||
|
|
||||||
|
if (sourceNodes.count(name) == 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
std::vector<ModuleName> queue{name};
|
||||||
|
|
||||||
|
while (!queue.empty())
|
||||||
|
{
|
||||||
|
ModuleName next = std::move(queue.back());
|
||||||
|
queue.pop_back();
|
||||||
|
|
||||||
|
LUAU_ASSERT(sourceNodes.count(next) > 0);
|
||||||
|
SourceNode& sourceNode = *sourceNodes[next];
|
||||||
|
|
||||||
|
if (!processSubtree(sourceNode))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const Set<ModuleName>& dependents = sourceNode.dependents;
|
||||||
|
queue.insert(queue.end(), dependents.begin(), dependents.end());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SourceModule* Frontend::getSourceModule(const ModuleName& moduleName)
|
SourceModule* Frontend::getSourceModule(const ModuleName& moduleName)
|
||||||
|
@ -1338,7 +1449,7 @@ ModulePtr check(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DataFlowGraph dfg = DataFlowGraphBuilder::build(sourceModule.root, iceHandler);
|
DataFlowGraph dfg = DataFlowGraphBuilder::build(sourceModule.root, NotNull{&result->defArena}, NotNull{&result->keyArena}, iceHandler);
|
||||||
|
|
||||||
UnifierSharedState unifierState{iceHandler};
|
UnifierSharedState unifierState{iceHandler};
|
||||||
unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit;
|
unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit;
|
||||||
|
@ -1637,6 +1748,17 @@ std::pair<SourceNode*, SourceModule*> Frontend::getSourceNode(const ModuleName&
|
||||||
|
|
||||||
sourceNode->name = sourceModule->name;
|
sourceNode->name = sourceModule->name;
|
||||||
sourceNode->humanReadableName = sourceModule->humanReadableName;
|
sourceNode->humanReadableName = sourceModule->humanReadableName;
|
||||||
|
|
||||||
|
if (FFlag::LuauBetterReverseDependencyTracking)
|
||||||
|
{
|
||||||
|
// clear all prior dependents. we will re-add them after parsing the rest of the graph
|
||||||
|
for (const auto& [moduleName, _] : sourceNode->requireLocations)
|
||||||
|
{
|
||||||
|
if (auto depIt = sourceNodes.find(moduleName); depIt != sourceNodes.end())
|
||||||
|
depIt->second->dependents.erase(sourceNode->name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
sourceNode->requireSet.clear();
|
sourceNode->requireSet.clear();
|
||||||
sourceNode->requireLocations.clear();
|
sourceNode->requireLocations.clear();
|
||||||
sourceNode->dirtySourceModule = false;
|
sourceNode->dirtySourceModule = false;
|
||||||
|
@ -1758,11 +1880,21 @@ std::string FrontendModuleResolver::getHumanReadableModuleName(const ModuleName&
|
||||||
return frontend->fileResolver->getHumanReadableModuleName(moduleName);
|
return frontend->fileResolver->getHumanReadableModuleName(moduleName);
|
||||||
}
|
}
|
||||||
|
|
||||||
void FrontendModuleResolver::setModule(const ModuleName& moduleName, ModulePtr module)
|
bool FrontendModuleResolver::setModule(const ModuleName& moduleName, ModulePtr module)
|
||||||
{
|
{
|
||||||
std::scoped_lock lock(moduleMutex);
|
std::scoped_lock lock(moduleMutex);
|
||||||
|
|
||||||
|
if (FFlag::LuauBetterReverseDependencyTracking)
|
||||||
|
{
|
||||||
|
bool replaced = modules.count(moduleName) > 0;
|
||||||
modules[moduleName] = std::move(module);
|
modules[moduleName] = std::move(module);
|
||||||
|
return replaced;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
modules[moduleName] = std::move(module);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void FrontendModuleResolver::clearModules()
|
void FrontendModuleResolver::clearModules()
|
||||||
|
|
|
@ -977,7 +977,8 @@ struct TypeCacher : TypeOnceVisitor
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool visit(TypePackId tp, const BoundTypePack& btp) override {
|
bool visit(TypePackId tp, const BoundTypePack& btp) override
|
||||||
|
{
|
||||||
traverse(btp.boundTo);
|
traverse(btp.boundTo);
|
||||||
if (isUncacheable(btp.boundTo))
|
if (isUncacheable(btp.boundTo))
|
||||||
markUncacheable(tp);
|
markUncacheable(tp);
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -61,9 +62,7 @@ TypeId Instantiation::clean(TypeId ty)
|
||||||
LUAU_ASSERT(ftv);
|
LUAU_ASSERT(ftv);
|
||||||
|
|
||||||
FunctionType clone = FunctionType{level, scope, ftv->argTypes, ftv->retTypes, ftv->definition, ftv->hasSelf};
|
FunctionType clone = FunctionType{level, scope, ftv->argTypes, ftv->retTypes, ftv->definition, ftv->hasSelf};
|
||||||
clone.magicFunction = ftv->magicFunction;
|
clone.magic = ftv->magic;
|
||||||
clone.dcrMagicFunction = ftv->dcrMagicFunction;
|
|
||||||
clone.dcrMagicRefinement = ftv->dcrMagicRefinement;
|
|
||||||
clone.tags = ftv->tags;
|
clone.tags = ftv->tags;
|
||||||
clone.argNames = ftv->argNames;
|
clone.argNames = ftv->argNames;
|
||||||
TypeId result = addType(std::move(clone));
|
TypeId result = addType(std::move(clone));
|
||||||
|
@ -165,7 +164,7 @@ TypeId ReplaceGenerics::clean(TypeId ty)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
return addType(FreeType{scope, level});
|
return FFlag::LuauFreeTypesMustHaveBounds ? arena->freshType(builtinTypes, scope, level) : addType(FreeType{scope, level});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -15,11 +15,32 @@
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2);
|
LUAU_FASTFLAG(LuauSolverV2);
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauIncrementalAutocompleteCommentDetection)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
static bool contains(Position pos, Comment comment)
|
static void defaultLogLuau(std::string_view input)
|
||||||
|
{
|
||||||
|
// The default is to do nothing because we don't want to mess with
|
||||||
|
// the xml parsing done by the dcr script.
|
||||||
|
}
|
||||||
|
|
||||||
|
Luau::LogLuauProc logLuau = &defaultLogLuau;
|
||||||
|
|
||||||
|
void setLogLuau(LogLuauProc ll)
|
||||||
|
{
|
||||||
|
logLuau = ll;
|
||||||
|
}
|
||||||
|
|
||||||
|
void resetLogLuauProc()
|
||||||
|
{
|
||||||
|
logLuau = &defaultLogLuau;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static bool contains_DEPRECATED(Position pos, Comment comment)
|
||||||
{
|
{
|
||||||
if (comment.location.contains(pos))
|
if (comment.location.contains(pos))
|
||||||
return true;
|
return true;
|
||||||
|
@ -32,7 +53,22 @@ static bool contains(Position pos, Comment comment)
|
||||||
return false;
|
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(
|
auto iter = std::lower_bound(
|
||||||
commentLocations.begin(),
|
commentLocations.begin(),
|
||||||
|
@ -40,6 +76,11 @@ static bool isWithinComment(const std::vector<Comment>& commentLocations, Positi
|
||||||
Comment{Lexeme::Comment, Location{pos, pos}},
|
Comment{Lexeme::Comment, Location{pos, pos}},
|
||||||
[](const Comment& a, const Comment& b)
|
[](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;
|
return a.location.end < b.location.end;
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
@ -47,7 +88,7 @@ static bool isWithinComment(const std::vector<Comment>& commentLocations, Positi
|
||||||
if (iter == commentLocations.end())
|
if (iter == commentLocations.end())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (contains(pos, *iter))
|
if (FFlag::LuauIncrementalAutocompleteCommentDetection ? contains(pos, *iter) : contains_DEPRECATED(pos, *iter))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
// Due to the nature of std::lower_bound, it is possible that iter points at a comment that ends
|
// Due to the nature of std::lower_bound, it is possible that iter points at a comment that ends
|
||||||
|
|
|
@ -14,12 +14,15 @@
|
||||||
#include "Luau/TypeFunction.h"
|
#include "Luau/TypeFunction.h"
|
||||||
#include "Luau/Def.h"
|
#include "Luau/Def.h"
|
||||||
#include "Luau/ToString.h"
|
#include "Luau/ToString.h"
|
||||||
#include "Luau/TypeFwd.h"
|
#include "Luau/TypeUtils.h"
|
||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <iterator>
|
#include <iterator>
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauCountSelfCallsNonstrict)
|
LUAU_FASTFLAGVARIABLE(LuauCountSelfCallsNonstrict)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauNonStrictVisitorImprovements)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauNewNonStrictWarnOnUnknownGlobals)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -211,7 +214,7 @@ struct NonStrictTypeChecker
|
||||||
return *fst;
|
return *fst;
|
||||||
else if (auto ftp = get<FreeTypePack>(pack))
|
else if (auto ftp = get<FreeTypePack>(pack))
|
||||||
{
|
{
|
||||||
TypeId result = arena->addType(FreeType{ftp->scope});
|
TypeId result = FFlag::LuauFreeTypesMustHaveBounds ? arena->freshType(builtinTypes, ftp->scope) : arena->addType(FreeType{ftp->scope});
|
||||||
TypePackId freeTail = arena->addTypePack(FreeTypePack{ftp->scope});
|
TypePackId freeTail = arena->addTypePack(FreeTypePack{ftp->scope});
|
||||||
|
|
||||||
TypePack* resultPack = emplaceTypePack<TypePack>(asMutable(pack));
|
TypePack* resultPack = emplaceTypePack<TypePack>(asMutable(pack));
|
||||||
|
@ -341,8 +344,9 @@ struct NonStrictTypeChecker
|
||||||
|
|
||||||
NonStrictContext visit(AstStatIf* ifStatement)
|
NonStrictContext visit(AstStatIf* ifStatement)
|
||||||
{
|
{
|
||||||
NonStrictContext condB = visit(ifStatement->condition);
|
NonStrictContext condB = visit(ifStatement->condition, ValueContext::RValue);
|
||||||
NonStrictContext branchContext;
|
NonStrictContext branchContext;
|
||||||
|
|
||||||
// If there is no else branch, don't bother generating warnings for the then branch - we can't prove there is an error
|
// If there is no else branch, don't bother generating warnings for the then branch - we can't prove there is an error
|
||||||
if (ifStatement->elsebody)
|
if (ifStatement->elsebody)
|
||||||
{
|
{
|
||||||
|
@ -350,16 +354,31 @@ struct NonStrictTypeChecker
|
||||||
NonStrictContext elseBody = visit(ifStatement->elsebody);
|
NonStrictContext elseBody = visit(ifStatement->elsebody);
|
||||||
branchContext = NonStrictContext::conjunction(builtinTypes, arena, thenBody, elseBody);
|
branchContext = NonStrictContext::conjunction(builtinTypes, arena, thenBody, elseBody);
|
||||||
}
|
}
|
||||||
|
|
||||||
return NonStrictContext::disjunction(builtinTypes, arena, condB, branchContext);
|
return NonStrictContext::disjunction(builtinTypes, arena, condB, branchContext);
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatWhile* whileStatement)
|
NonStrictContext visit(AstStatWhile* whileStatement)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
NonStrictContext condition = visit(whileStatement->condition, ValueContext::RValue);
|
||||||
|
NonStrictContext body = visit(whileStatement->body);
|
||||||
|
return NonStrictContext::disjunction(builtinTypes, arena, condition, body);
|
||||||
|
}
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatRepeat* repeatStatement)
|
NonStrictContext visit(AstStatRepeat* repeatStatement)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
NonStrictContext body = visit(repeatStatement->body);
|
||||||
|
NonStrictContext condition = visit(repeatStatement->condition, ValueContext::RValue);
|
||||||
|
return NonStrictContext::disjunction(builtinTypes, arena, body, condition);
|
||||||
|
}
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -375,49 +394,94 @@ struct NonStrictTypeChecker
|
||||||
|
|
||||||
NonStrictContext visit(AstStatReturn* returnStatement)
|
NonStrictContext visit(AstStatReturn* returnStatement)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
// TODO: this is believing existing code, but i'm not sure if this makes sense
|
||||||
|
// for how the contexts are handled
|
||||||
|
for (AstExpr* expr : returnStatement->list)
|
||||||
|
visit(expr, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatExpr* expr)
|
NonStrictContext visit(AstStatExpr* expr)
|
||||||
{
|
{
|
||||||
return visit(expr->expr);
|
return visit(expr->expr, ValueContext::RValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatLocal* local)
|
NonStrictContext visit(AstStatLocal* local)
|
||||||
{
|
{
|
||||||
for (AstExpr* rhs : local->values)
|
for (AstExpr* rhs : local->values)
|
||||||
visit(rhs);
|
visit(rhs, ValueContext::RValue);
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatFor* forStatement)
|
NonStrictContext visit(AstStatFor* forStatement)
|
||||||
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
// TODO: throwing out context based on same principle as existing code?
|
||||||
|
if (forStatement->from)
|
||||||
|
visit(forStatement->from, ValueContext::RValue);
|
||||||
|
if (forStatement->to)
|
||||||
|
visit(forStatement->to, ValueContext::RValue);
|
||||||
|
if (forStatement->step)
|
||||||
|
visit(forStatement->step, ValueContext::RValue);
|
||||||
|
return visit(forStatement->body);
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatForIn* forInStatement)
|
NonStrictContext visit(AstStatForIn* forInStatement)
|
||||||
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
for (AstExpr* rhs : forInStatement->values)
|
||||||
|
visit(rhs, ValueContext::RValue);
|
||||||
|
return visit(forInStatement->body);
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatAssign* assign)
|
NonStrictContext visit(AstStatAssign* assign)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
for (AstExpr* lhs : assign->vars)
|
||||||
|
visit(lhs, ValueContext::LValue);
|
||||||
|
for (AstExpr* rhs : assign->values)
|
||||||
|
visit(rhs, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatCompoundAssign* compoundAssign)
|
NonStrictContext visit(AstStatCompoundAssign* compoundAssign)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
visit(compoundAssign->var, ValueContext::LValue);
|
||||||
|
visit(compoundAssign->value, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatFunction* statFn)
|
NonStrictContext visit(AstStatFunction* statFn)
|
||||||
{
|
{
|
||||||
return visit(statFn->func);
|
return visit(statFn->func, ValueContext::RValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatLocalFunction* localFn)
|
NonStrictContext visit(AstStatLocalFunction* localFn)
|
||||||
{
|
{
|
||||||
return visit(localFn->func);
|
return visit(localFn->func, ValueContext::RValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstStatTypeAlias* typeAlias)
|
NonStrictContext visit(AstStatTypeAlias* typeAlias)
|
||||||
|
@ -447,14 +511,22 @@ struct NonStrictTypeChecker
|
||||||
|
|
||||||
NonStrictContext visit(AstStatError* error)
|
NonStrictContext visit(AstStatError* error)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
for (AstStat* stat : error->statements)
|
||||||
|
visit(stat);
|
||||||
|
for (AstExpr* expr : error->expressions)
|
||||||
|
visit(expr, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExpr* expr)
|
NonStrictContext visit(AstExpr* expr, ValueContext context)
|
||||||
{
|
{
|
||||||
auto pusher = pushStack(expr);
|
auto pusher = pushStack(expr);
|
||||||
if (auto e = expr->as<AstExprGroup>())
|
if (auto e = expr->as<AstExprGroup>())
|
||||||
return visit(e);
|
return visit(e, context);
|
||||||
else if (auto e = expr->as<AstExprConstantNil>())
|
else if (auto e = expr->as<AstExprConstantNil>())
|
||||||
return visit(e);
|
return visit(e);
|
||||||
else if (auto e = expr->as<AstExprConstantBool>())
|
else if (auto e = expr->as<AstExprConstantBool>())
|
||||||
|
@ -464,17 +536,17 @@ struct NonStrictTypeChecker
|
||||||
else if (auto e = expr->as<AstExprConstantString>())
|
else if (auto e = expr->as<AstExprConstantString>())
|
||||||
return visit(e);
|
return visit(e);
|
||||||
else if (auto e = expr->as<AstExprLocal>())
|
else if (auto e = expr->as<AstExprLocal>())
|
||||||
return visit(e);
|
return visit(e, context);
|
||||||
else if (auto e = expr->as<AstExprGlobal>())
|
else if (auto e = expr->as<AstExprGlobal>())
|
||||||
return visit(e);
|
return visit(e, context);
|
||||||
else if (auto e = expr->as<AstExprVarargs>())
|
else if (auto e = expr->as<AstExprVarargs>())
|
||||||
return visit(e);
|
return visit(e);
|
||||||
else if (auto e = expr->as<AstExprCall>())
|
else if (auto e = expr->as<AstExprCall>())
|
||||||
return visit(e);
|
return visit(e);
|
||||||
else if (auto e = expr->as<AstExprIndexName>())
|
else if (auto e = expr->as<AstExprIndexName>())
|
||||||
return visit(e);
|
return visit(e, context);
|
||||||
else if (auto e = expr->as<AstExprIndexExpr>())
|
else if (auto e = expr->as<AstExprIndexExpr>())
|
||||||
return visit(e);
|
return visit(e, context);
|
||||||
else if (auto e = expr->as<AstExprFunction>())
|
else if (auto e = expr->as<AstExprFunction>())
|
||||||
return visit(e);
|
return visit(e);
|
||||||
else if (auto e = expr->as<AstExprTable>())
|
else if (auto e = expr->as<AstExprTable>())
|
||||||
|
@ -498,8 +570,11 @@ struct NonStrictTypeChecker
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprGroup* group)
|
NonStrictContext visit(AstExprGroup* group, ValueContext context)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
return visit(group->expr, context);
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -523,17 +598,30 @@ struct NonStrictTypeChecker
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprLocal* local)
|
NonStrictContext visit(AstExprLocal* local, ValueContext context)
|
||||||
{
|
{
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprGlobal* global)
|
NonStrictContext visit(AstExprGlobal* global, ValueContext context)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNewNonStrictWarnOnUnknownGlobals)
|
||||||
|
{
|
||||||
|
// We don't file unknown symbols for LValues.
|
||||||
|
if (context == ValueContext::LValue)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
NotNull<Scope> scope = stack.back();
|
||||||
|
if (!scope->lookup(global->name))
|
||||||
|
{
|
||||||
|
reportError(UnknownSymbol{global->name.value, UnknownSymbol::Binding}, global->location);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprVarargs* global)
|
NonStrictContext visit(AstExprVarargs* varargs)
|
||||||
{
|
{
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
@ -762,13 +850,23 @@ struct NonStrictTypeChecker
|
||||||
return fresh;
|
return fresh;
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprIndexName* indexName)
|
NonStrictContext visit(AstExprIndexName* indexName, ValueContext context)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
return visit(indexName->expr, context);
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprIndexExpr* indexExpr)
|
NonStrictContext visit(AstExprIndexExpr* indexExpr, ValueContext context)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
NonStrictContext expr = visit(indexExpr->expr, context);
|
||||||
|
NonStrictContext index = visit(indexExpr->index, ValueContext::RValue);
|
||||||
|
return NonStrictContext::disjunction(builtinTypes, arena, expr, index);
|
||||||
|
}
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -788,39 +886,74 @@ struct NonStrictTypeChecker
|
||||||
|
|
||||||
NonStrictContext visit(AstExprTable* table)
|
NonStrictContext visit(AstExprTable* table)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
for (auto [_, key, value] : table->items)
|
||||||
|
{
|
||||||
|
if (key)
|
||||||
|
visit(key, ValueContext::RValue);
|
||||||
|
visit(value, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprUnary* unary)
|
NonStrictContext visit(AstExprUnary* unary)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
return visit(unary->expr, ValueContext::RValue);
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprBinary* binary)
|
NonStrictContext visit(AstExprBinary* binary)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
NonStrictContext lhs = visit(binary->left, ValueContext::RValue);
|
||||||
|
NonStrictContext rhs = visit(binary->right, ValueContext::RValue);
|
||||||
|
return NonStrictContext::disjunction(builtinTypes, arena, lhs, rhs);
|
||||||
|
}
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprTypeAssertion* typeAssertion)
|
NonStrictContext visit(AstExprTypeAssertion* typeAssertion)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
return visit(typeAssertion->expr, ValueContext::RValue);
|
||||||
|
else
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprIfElse* ifElse)
|
NonStrictContext visit(AstExprIfElse* ifElse)
|
||||||
{
|
{
|
||||||
NonStrictContext condB = visit(ifElse->condition);
|
NonStrictContext condB = visit(ifElse->condition, ValueContext::RValue);
|
||||||
NonStrictContext thenB = visit(ifElse->trueExpr);
|
NonStrictContext thenB = visit(ifElse->trueExpr, ValueContext::RValue);
|
||||||
NonStrictContext elseB = visit(ifElse->falseExpr);
|
NonStrictContext elseB = visit(ifElse->falseExpr, ValueContext::RValue);
|
||||||
return NonStrictContext::disjunction(builtinTypes, arena, condB, NonStrictContext::conjunction(builtinTypes, arena, thenB, elseB));
|
return NonStrictContext::disjunction(builtinTypes, arena, condB, NonStrictContext::conjunction(builtinTypes, arena, thenB, elseB));
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprInterpString* interpString)
|
NonStrictContext visit(AstExprInterpString* interpString)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
for (AstExpr* expr : interpString->expressions)
|
||||||
|
visit(expr, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
NonStrictContext visit(AstExprError* error)
|
NonStrictContext visit(AstExprError* error)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauNonStrictVisitorImprovements)
|
||||||
|
{
|
||||||
|
for (AstExpr* expr : error->expressions)
|
||||||
|
visit(expr, ValueContext::RValue);
|
||||||
|
}
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -17,12 +17,11 @@
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauCheckNormalizeInvariant)
|
LUAU_FASTFLAGVARIABLE(DebugLuauCheckNormalizeInvariant)
|
||||||
|
|
||||||
LUAU_FASTINTVARIABLE(LuauNormalizeCacheLimit, 100000);
|
LUAU_FASTINTVARIABLE(LuauNormalizeCacheLimit, 100000)
|
||||||
LUAU_FASTFLAG(LuauSolverV2);
|
|
||||||
|
|
||||||
LUAU_FASTINTVARIABLE(LuauNormalizeIntersectionLimit, 200)
|
LUAU_FASTINTVARIABLE(LuauNormalizeIntersectionLimit, 200)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauNormalizationTracksCyclicPairsThroughInhabitance);
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauIntersectNormalsNeedsToTrackResourceLimits);
|
LUAU_FASTFLAGVARIABLE(LuauFixInfiniteRecursionInNormalization)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauFixNormalizedIntersectionOfNegatedClass)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -1809,7 +1808,8 @@ NormalizationResult Normalizer::unionNormalWithTy(
|
||||||
}
|
}
|
||||||
else if (get<UnknownType>(here.tops))
|
else if (get<UnknownType>(here.tops))
|
||||||
return NormalizationResult::True;
|
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)
|
if (tyvarIndex(there) <= ignoreSmallerTyvars)
|
||||||
return NormalizationResult::True;
|
return NormalizationResult::True;
|
||||||
|
@ -2284,9 +2284,24 @@ void Normalizer::intersectClassesWithClass(NormalizedClassType& heres, TypeId th
|
||||||
else if (isSubclass(there, hereTy))
|
else if (isSubclass(there, hereTy))
|
||||||
{
|
{
|
||||||
TypeIds negations = std::move(hereNegations);
|
TypeIds negations = std::move(hereNegations);
|
||||||
|
bool emptyIntersectWithNegation = false;
|
||||||
|
|
||||||
for (auto nIt = negations.begin(); nIt != negations.end();)
|
for (auto nIt = negations.begin(); nIt != negations.end();)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauFixNormalizedIntersectionOfNegatedClass && isSubclass(there, *nIt))
|
||||||
|
{
|
||||||
|
// Hitting this block means that the incoming class is a
|
||||||
|
// subclass of this type, _and_ one of its negations is a
|
||||||
|
// superclass of this type, e.g.:
|
||||||
|
//
|
||||||
|
// Dog & ~Animal
|
||||||
|
//
|
||||||
|
// Clearly this intersects to never, so we mark this class as
|
||||||
|
// being removed from the normalized class type.
|
||||||
|
emptyIntersectWithNegation = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
if (!isSubclass(*nIt, there))
|
if (!isSubclass(*nIt, there))
|
||||||
{
|
{
|
||||||
nIt = negations.erase(nIt);
|
nIt = negations.erase(nIt);
|
||||||
|
@ -2299,6 +2314,7 @@ void Normalizer::intersectClassesWithClass(NormalizedClassType& heres, TypeId th
|
||||||
|
|
||||||
it = heres.ordering.erase(it);
|
it = heres.ordering.erase(it);
|
||||||
heres.classes.erase(hereTy);
|
heres.classes.erase(hereTy);
|
||||||
|
if (!emptyIntersectWithNegation)
|
||||||
heres.pushPair(there, std::move(negations));
|
heres.pushPair(there, std::move(negations));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -2583,12 +2599,32 @@ std::optional<TypeId> Normalizer::intersectionOfTables(TypeId here, TypeId there
|
||||||
if (hprop.readTy.has_value())
|
if (hprop.readTy.has_value())
|
||||||
{
|
{
|
||||||
if (tprop.readTy.has_value())
|
if (tprop.readTy.has_value())
|
||||||
|
{
|
||||||
|
if (FFlag::LuauFixInfiniteRecursionInNormalization)
|
||||||
|
{
|
||||||
|
TypeId ty = simplifyIntersection(builtinTypes, NotNull{arena}, *hprop.readTy, *tprop.readTy).result;
|
||||||
|
|
||||||
|
// If any property is going to get mapped to `never`, we can just call the entire table `never`.
|
||||||
|
// Since this check is syntactic, we may sometimes miss simplifying tables with complex uninhabited properties.
|
||||||
|
// Prior versions of this code attempted to do this semantically using the normalization machinery, but this
|
||||||
|
// mistakenly causes infinite loops when giving more complex recursive table types. As it stands, this approach
|
||||||
|
// will continue to scale as simplification is improved, but we may wish to reintroduce the semantic approach
|
||||||
|
// once we have revisited the usage of seen sets systematically (and possibly with some additional guarding to recognize
|
||||||
|
// when types are infinitely-recursive with non-pointer identical instances of them, or some guard to prevent that
|
||||||
|
// construction altogether). See also: `gh1632_no_infinite_recursion_in_normalization`
|
||||||
|
if (get<NeverType>(ty))
|
||||||
|
return {builtinTypes->neverType};
|
||||||
|
|
||||||
|
prop.readTy = ty;
|
||||||
|
hereSubThere &= (ty == hprop.readTy);
|
||||||
|
thereSubHere &= (ty == tprop.readTy);
|
||||||
|
}
|
||||||
|
else
|
||||||
{
|
{
|
||||||
// if the intersection of the read types of a property is uninhabited, the whole table is `never`.
|
// if the intersection of the read types of a property is uninhabited, the whole table is `never`.
|
||||||
// We've seen these table prop elements before and we're about to ask if their intersection
|
// We've seen these table prop elements before and we're about to ask if their intersection
|
||||||
// is inhabited
|
// is inhabited
|
||||||
if (FFlag::LuauNormalizationTracksCyclicPairsThroughInhabitance)
|
|
||||||
{
|
|
||||||
auto pair1 = std::pair{*hprop.readTy, *tprop.readTy};
|
auto pair1 = std::pair{*hprop.readTy, *tprop.readTy};
|
||||||
auto pair2 = std::pair{*tprop.readTy, *hprop.readTy};
|
auto pair2 = std::pair{*tprop.readTy, *hprop.readTy};
|
||||||
if (seenTablePropPairs.contains(pair1) || seenTablePropPairs.contains(pair2))
|
if (seenTablePropPairs.contains(pair1) || seenTablePropPairs.contains(pair2))
|
||||||
|
@ -2603,6 +2639,8 @@ std::optional<TypeId> Normalizer::intersectionOfTables(TypeId here, TypeId there
|
||||||
seenTablePropPairs.insert(pair2);
|
seenTablePropPairs.insert(pair2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FIXME(ariel): this is being added in a flag removal, so not changing the semantics here, but worth noting that this
|
||||||
|
// fresh `seenSet` is definitely a bug. we already have `seenSet` from the parameter that _should_ have been used here.
|
||||||
Set<TypeId> seenSet{nullptr};
|
Set<TypeId> seenSet{nullptr};
|
||||||
NormalizationResult res = isIntersectionInhabited(*hprop.readTy, *tprop.readTy, seenTablePropPairs, seenSet);
|
NormalizationResult res = isIntersectionInhabited(*hprop.readTy, *tprop.readTy, seenTablePropPairs, seenSet);
|
||||||
|
|
||||||
|
@ -2616,34 +2654,6 @@ std::optional<TypeId> Normalizer::intersectionOfTables(TypeId here, TypeId there
|
||||||
hereSubThere &= (ty == hprop.readTy);
|
hereSubThere &= (ty == hprop.readTy);
|
||||||
thereSubHere &= (ty == tprop.readTy);
|
thereSubHere &= (ty == tprop.readTy);
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
|
|
||||||
if (seenSet.contains(*hprop.readTy) && seenSet.contains(*tprop.readTy))
|
|
||||||
{
|
|
||||||
seenSet.erase(*hprop.readTy);
|
|
||||||
seenSet.erase(*tprop.readTy);
|
|
||||||
return {builtinTypes->neverType};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
seenSet.insert(*hprop.readTy);
|
|
||||||
seenSet.insert(*tprop.readTy);
|
|
||||||
}
|
|
||||||
|
|
||||||
NormalizationResult res = isIntersectionInhabited(*hprop.readTy, *tprop.readTy);
|
|
||||||
|
|
||||||
seenSet.erase(*hprop.readTy);
|
|
||||||
seenSet.erase(*tprop.readTy);
|
|
||||||
|
|
||||||
if (NormalizationResult::True != res)
|
|
||||||
return {builtinTypes->neverType};
|
|
||||||
|
|
||||||
TypeId ty = simplifyIntersection(builtinTypes, NotNull{arena}, *hprop.readTy, *tprop.readTy).result;
|
|
||||||
prop.readTy = ty;
|
|
||||||
hereSubThere &= (ty == hprop.readTy);
|
|
||||||
thereSubHere &= (ty == tprop.readTy);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -3042,12 +3052,9 @@ NormalizationResult Normalizer::intersectTyvarsWithTy(
|
||||||
// See above for an explaination of `ignoreSmallerTyvars`.
|
// See above for an explaination of `ignoreSmallerTyvars`.
|
||||||
NormalizationResult Normalizer::intersectNormals(NormalizedType& here, const NormalizedType& there, int ignoreSmallerTyvars)
|
NormalizationResult Normalizer::intersectNormals(NormalizedType& here, const NormalizedType& there, int ignoreSmallerTyvars)
|
||||||
{
|
{
|
||||||
if (FFlag::LuauIntersectNormalsNeedsToTrackResourceLimits)
|
|
||||||
{
|
|
||||||
RecursionCounter _rc(&sharedState->counters.recursionCount);
|
RecursionCounter _rc(&sharedState->counters.recursionCount);
|
||||||
if (!withinResourceLimits())
|
if (!withinResourceLimits())
|
||||||
return NormalizationResult::HitLimits;
|
return NormalizationResult::HitLimits;
|
||||||
}
|
|
||||||
|
|
||||||
if (!get<NeverType>(there.tops))
|
if (!get<NeverType>(there.tops))
|
||||||
{
|
{
|
||||||
|
@ -3162,7 +3169,8 @@ NormalizationResult Normalizer::intersectNormalWithTy(
|
||||||
}
|
}
|
||||||
return NormalizationResult::True;
|
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 thereNorm{builtinTypes};
|
||||||
NormalizedType topNorm{builtinTypes};
|
NormalizedType topNorm{builtinTypes};
|
||||||
|
|
|
@ -420,7 +420,8 @@ static std::optional<TypeId> selectOverload(
|
||||||
TypePackId argsPack
|
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);
|
auto [status, overload] = resolver->selectOverload(fn, argsPack);
|
||||||
|
|
||||||
if (status == OverloadResolver::Analysis::Ok)
|
if (status == OverloadResolver::Analysis::Ok)
|
||||||
|
|
|
@ -31,16 +31,16 @@ struct TypeSimplifier
|
||||||
|
|
||||||
int recursionDepth = 0;
|
int recursionDepth = 0;
|
||||||
|
|
||||||
TypeId mkNegation(TypeId ty);
|
TypeId mkNegation(TypeId ty) const;
|
||||||
|
|
||||||
TypeId intersectFromParts(std::set<TypeId> parts);
|
TypeId intersectFromParts(std::set<TypeId> parts);
|
||||||
|
|
||||||
TypeId intersectUnionWithType(TypeId unionTy, TypeId right);
|
TypeId intersectUnionWithType(TypeId left, TypeId right);
|
||||||
TypeId intersectUnions(TypeId left, TypeId right);
|
TypeId intersectUnions(TypeId left, TypeId right);
|
||||||
TypeId intersectNegatedUnion(TypeId unionTy, TypeId right);
|
TypeId intersectNegatedUnion(TypeId left, TypeId right);
|
||||||
|
|
||||||
TypeId intersectTypeWithNegation(TypeId a, TypeId b);
|
TypeId intersectTypeWithNegation(TypeId left, TypeId right);
|
||||||
TypeId intersectNegations(TypeId a, TypeId b);
|
TypeId intersectNegations(TypeId left, TypeId right);
|
||||||
|
|
||||||
TypeId intersectIntersectionWithType(TypeId left, TypeId right);
|
TypeId intersectIntersectionWithType(TypeId left, TypeId right);
|
||||||
|
|
||||||
|
@ -48,8 +48,8 @@ struct TypeSimplifier
|
||||||
// unions, intersections, or negations.
|
// unions, intersections, or negations.
|
||||||
std::optional<TypeId> basicIntersect(TypeId left, TypeId right);
|
std::optional<TypeId> basicIntersect(TypeId left, TypeId right);
|
||||||
|
|
||||||
TypeId intersect(TypeId ty, TypeId discriminant);
|
TypeId intersect(TypeId left, TypeId right);
|
||||||
TypeId union_(TypeId ty, TypeId discriminant);
|
TypeId union_(TypeId left, TypeId right);
|
||||||
|
|
||||||
TypeId simplify(TypeId ty);
|
TypeId simplify(TypeId ty);
|
||||||
TypeId simplify(TypeId ty, DenseHashSet<TypeId>& seen);
|
TypeId simplify(TypeId ty, DenseHashSet<TypeId>& seen);
|
||||||
|
@ -573,7 +573,7 @@ Relation relate(TypeId left, TypeId right)
|
||||||
return relate(left, right, seen);
|
return relate(left, right, seen);
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId TypeSimplifier::mkNegation(TypeId ty)
|
TypeId TypeSimplifier::mkNegation(TypeId ty) const
|
||||||
{
|
{
|
||||||
TypeId result = nullptr;
|
TypeId result = nullptr;
|
||||||
|
|
||||||
|
|
|
@ -98,9 +98,7 @@ static TypeId shallowClone(TypeId ty, TypeArena& dest, const TxnLog* log, bool a
|
||||||
FunctionType clone = FunctionType{a.level, a.scope, a.argTypes, a.retTypes, a.definition, a.hasSelf};
|
FunctionType clone = FunctionType{a.level, a.scope, a.argTypes, a.retTypes, a.definition, a.hasSelf};
|
||||||
clone.generics = a.generics;
|
clone.generics = a.generics;
|
||||||
clone.genericPacks = a.genericPacks;
|
clone.genericPacks = a.genericPacks;
|
||||||
clone.magicFunction = a.magicFunction;
|
clone.magic = a.magic;
|
||||||
clone.dcrMagicFunction = a.dcrMagicFunction;
|
|
||||||
clone.dcrMagicRefinement = a.dcrMagicRefinement;
|
|
||||||
clone.tags = a.tags;
|
clone.tags = a.tags;
|
||||||
clone.argNames = a.argNames;
|
clone.argNames = a.argNames;
|
||||||
clone.isCheckedFunction = a.isCheckedFunction;
|
clone.isCheckedFunction = a.isCheckedFunction;
|
||||||
|
|
|
@ -22,7 +22,6 @@
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauSubtypingCheckPathValidity)
|
LUAU_FASTFLAGVARIABLE(DebugLuauSubtypingCheckPathValidity)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauRetrySubtypingWithoutHiddenPack)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -1474,15 +1473,14 @@ SubtypingResult Subtyping::isCovariantWith(
|
||||||
|
|
||||||
// If subtyping failed in the argument packs, we should check if there's a hidden variadic tail and try ignoring it.
|
// If subtyping failed in the argument packs, we should check if there's a hidden variadic tail and try ignoring it.
|
||||||
// This might cause subtyping correctly because the sub type here may not have a hidden variadic tail or equivalent.
|
// This might cause subtyping correctly because the sub type here may not have a hidden variadic tail or equivalent.
|
||||||
if (FFlag::LuauRetrySubtypingWithoutHiddenPack && !result.isSubtype)
|
if (!result.isSubtype)
|
||||||
{
|
{
|
||||||
auto [arguments, tail] = flatten(superFunction->argTypes);
|
auto [arguments, tail] = flatten(superFunction->argTypes);
|
||||||
|
|
||||||
if (auto variadic = get<VariadicTypePack>(tail); variadic && variadic->hidden)
|
if (auto variadic = get<VariadicTypePack>(tail); variadic && variadic->hidden)
|
||||||
{
|
{
|
||||||
result.orElse(
|
result.orElse(isContravariantWith(env, subFunction->argTypes, arena->addTypePack(TypePack{arguments}), scope)
|
||||||
isContravariantWith(env, subFunction->argTypes, arena->addTypePack(TypePack{arguments}), scope).withBothComponent(TypePath::PackField::Arguments)
|
.withBothComponent(TypePath::PackField::Arguments));
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,6 +9,9 @@
|
||||||
#include "Luau/TypeUtils.h"
|
#include "Luau/TypeUtils.h"
|
||||||
#include "Luau/Unifier2.h"
|
#include "Luau/Unifier2.h"
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauDontInPlaceMutateTableType)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauAllowNonSharedTableTypesInLiteral)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -236,6 +239,8 @@ TypeId matchLiteralType(
|
||||||
return exprType;
|
return exprType;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
DenseHashSet<AstExprConstantString*> keysToDelete{nullptr};
|
||||||
|
|
||||||
for (const AstExprTable::Item& item : exprTable->items)
|
for (const AstExprTable::Item& item : exprTable->items)
|
||||||
{
|
{
|
||||||
if (isRecord(item))
|
if (isRecord(item))
|
||||||
|
@ -247,8 +252,19 @@ TypeId matchLiteralType(
|
||||||
|
|
||||||
Property& prop = it->second;
|
Property& prop = it->second;
|
||||||
|
|
||||||
|
if (FFlag::LuauAllowNonSharedTableTypesInLiteral)
|
||||||
|
{
|
||||||
|
// If we encounter a duplcate property, we may have already
|
||||||
|
// set it to be read-only. If that's the case, the only thing
|
||||||
|
// that will definitely crash is trying to access a write
|
||||||
|
// only property.
|
||||||
|
LUAU_ASSERT(!prop.isWriteOnly());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
// Table literals always initially result in shared read-write types
|
// Table literals always initially result in shared read-write types
|
||||||
LUAU_ASSERT(prop.isShared());
|
LUAU_ASSERT(prop.isShared());
|
||||||
|
}
|
||||||
TypeId propTy = *prop.readTy;
|
TypeId propTy = *prop.readTy;
|
||||||
|
|
||||||
auto it2 = expectedTableTy->props.find(keyStr);
|
auto it2 = expectedTableTy->props.find(keyStr);
|
||||||
|
@ -280,6 +296,9 @@ TypeId matchLiteralType(
|
||||||
else
|
else
|
||||||
tableTy->indexer = TableIndexer{expectedTableTy->indexer->indexType, matchedType};
|
tableTy->indexer = TableIndexer{expectedTableTy->indexer->indexType, matchedType};
|
||||||
|
|
||||||
|
if (FFlag::LuauDontInPlaceMutateTableType)
|
||||||
|
keysToDelete.insert(item.key->as<AstExprConstantString>());
|
||||||
|
else
|
||||||
tableTy->props.erase(keyStr);
|
tableTy->props.erase(keyStr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -387,6 +406,16 @@ TypeId matchLiteralType(
|
||||||
LUAU_ASSERT(!"Unexpected");
|
LUAU_ASSERT(!"Unexpected");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (FFlag::LuauDontInPlaceMutateTableType)
|
||||||
|
{
|
||||||
|
for (const auto& key: keysToDelete)
|
||||||
|
{
|
||||||
|
const AstArray<char>& s = key->value;
|
||||||
|
std::string keyStr{s.data, s.data + s.size};
|
||||||
|
tableTy->props.erase(keyStr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Keys that the expectedType says we should have, but that aren't
|
// Keys that the expectedType says we should have, but that aren't
|
||||||
// specified by the AST fragment.
|
// specified by the AST fragment.
|
||||||
//
|
//
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -27,6 +27,7 @@ LUAU_FASTINTVARIABLE(LuauTypeMaximumStringifierLength, 500)
|
||||||
LUAU_FASTINTVARIABLE(LuauTableTypeMaximumStringifierLength, 0)
|
LUAU_FASTINTVARIABLE(LuauTableTypeMaximumStringifierLength, 0)
|
||||||
LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
LUAU_FASTINT(LuauTypeInferRecursionLimit)
|
||||||
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -478,24 +479,12 @@ bool hasLength(TypeId ty, DenseHashSet<TypeId>& seen, int* recursionCount)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
FreeType::FreeType(TypeLevel level)
|
// New constructors
|
||||||
|
FreeType::FreeType(TypeLevel level, TypeId lowerBound, TypeId upperBound)
|
||||||
: index(Unifiable::freshIndex())
|
: index(Unifiable::freshIndex())
|
||||||
, level(level)
|
, level(level)
|
||||||
, scope(nullptr)
|
, lowerBound(lowerBound)
|
||||||
{
|
, upperBound(upperBound)
|
||||||
}
|
|
||||||
|
|
||||||
FreeType::FreeType(Scope* scope)
|
|
||||||
: index(Unifiable::freshIndex())
|
|
||||||
, level{}
|
|
||||||
, scope(scope)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
FreeType::FreeType(Scope* scope, TypeLevel level)
|
|
||||||
: index(Unifiable::freshIndex())
|
|
||||||
, level(level)
|
|
||||||
, scope(scope)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -507,6 +496,40 @@ FreeType::FreeType(Scope* scope, TypeId lowerBound, TypeId upperBound)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
FreeType::FreeType(Scope* scope, TypeLevel level, TypeId lowerBound, TypeId upperBound)
|
||||||
|
: index(Unifiable::freshIndex())
|
||||||
|
, level(level)
|
||||||
|
, scope(scope)
|
||||||
|
, lowerBound(lowerBound)
|
||||||
|
, upperBound(upperBound)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
// Old constructors
|
||||||
|
FreeType::FreeType(TypeLevel level)
|
||||||
|
: index(Unifiable::freshIndex())
|
||||||
|
, level(level)
|
||||||
|
, scope(nullptr)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(!FFlag::LuauFreeTypesMustHaveBounds);
|
||||||
|
}
|
||||||
|
|
||||||
|
FreeType::FreeType(Scope* scope)
|
||||||
|
: index(Unifiable::freshIndex())
|
||||||
|
, level{}
|
||||||
|
, scope(scope)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(!FFlag::LuauFreeTypesMustHaveBounds);
|
||||||
|
}
|
||||||
|
|
||||||
|
FreeType::FreeType(Scope* scope, TypeLevel level)
|
||||||
|
: index(Unifiable::freshIndex())
|
||||||
|
, level(level)
|
||||||
|
, scope(scope)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(!FFlag::LuauFreeTypesMustHaveBounds);
|
||||||
|
}
|
||||||
|
|
||||||
GenericType::GenericType()
|
GenericType::GenericType()
|
||||||
: index(Unifiable::freshIndex())
|
: index(Unifiable::freshIndex())
|
||||||
, name("g" + std::to_string(index))
|
, name("g" + std::to_string(index))
|
||||||
|
@ -554,12 +577,12 @@ BlockedType::BlockedType()
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
Constraint* BlockedType::getOwner() const
|
const Constraint* BlockedType::getOwner() const
|
||||||
{
|
{
|
||||||
return owner;
|
return owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
void BlockedType::setOwner(Constraint* newOwner)
|
void BlockedType::setOwner(const Constraint* newOwner)
|
||||||
{
|
{
|
||||||
LUAU_ASSERT(owner == nullptr);
|
LUAU_ASSERT(owner == nullptr);
|
||||||
|
|
||||||
|
@ -569,7 +592,7 @@ void BlockedType::setOwner(Constraint* newOwner)
|
||||||
owner = newOwner;
|
owner = newOwner;
|
||||||
}
|
}
|
||||||
|
|
||||||
void BlockedType::replaceOwner(Constraint* newOwner)
|
void BlockedType::replaceOwner(const Constraint* newOwner)
|
||||||
{
|
{
|
||||||
owner = newOwner;
|
owner = newOwner;
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
#include "Luau/TypeArena.h"
|
#include "Luau/TypeArena.h"
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeArena);
|
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeArena);
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -22,7 +23,34 @@ TypeId TypeArena::addTV(Type&& tv)
|
||||||
return allocated;
|
return allocated;
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId TypeArena::freshType(TypeLevel level)
|
TypeId TypeArena::freshType(NotNull<BuiltinTypes> builtins, TypeLevel level)
|
||||||
|
{
|
||||||
|
TypeId allocated = types.allocate(FreeType{level, builtins->neverType, builtins->unknownType});
|
||||||
|
|
||||||
|
asMutable(allocated)->owningArena = this;
|
||||||
|
|
||||||
|
return allocated;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId TypeArena::freshType(NotNull<BuiltinTypes> builtins, Scope* scope)
|
||||||
|
{
|
||||||
|
TypeId allocated = types.allocate(FreeType{scope, builtins->neverType, builtins->unknownType});
|
||||||
|
|
||||||
|
asMutable(allocated)->owningArena = this;
|
||||||
|
|
||||||
|
return allocated;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId TypeArena::freshType(NotNull<BuiltinTypes> builtins, Scope* scope, TypeLevel level)
|
||||||
|
{
|
||||||
|
TypeId allocated = types.allocate(FreeType{scope, level, builtins->neverType, builtins->unknownType});
|
||||||
|
|
||||||
|
asMutable(allocated)->owningArena = this;
|
||||||
|
|
||||||
|
return allocated;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId TypeArena::freshType_DEPRECATED(TypeLevel level)
|
||||||
{
|
{
|
||||||
TypeId allocated = types.allocate(FreeType{level});
|
TypeId allocated = types.allocate(FreeType{level});
|
||||||
|
|
||||||
|
@ -31,7 +59,7 @@ TypeId TypeArena::freshType(TypeLevel level)
|
||||||
return allocated;
|
return allocated;
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId TypeArena::freshType(Scope* scope)
|
TypeId TypeArena::freshType_DEPRECATED(Scope* scope)
|
||||||
{
|
{
|
||||||
TypeId allocated = types.allocate(FreeType{scope});
|
TypeId allocated = types.allocate(FreeType{scope});
|
||||||
|
|
||||||
|
@ -40,7 +68,7 @@ TypeId TypeArena::freshType(Scope* scope)
|
||||||
return allocated;
|
return allocated;
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId TypeArena::freshType(Scope* scope, TypeLevel level)
|
TypeId TypeArena::freshType_DEPRECATED(Scope* scope, TypeLevel level)
|
||||||
{
|
{
|
||||||
TypeId allocated = types.allocate(FreeType{scope, level});
|
TypeId allocated = types.allocate(FreeType{scope, level});
|
||||||
|
|
||||||
|
|
|
@ -261,24 +261,24 @@ public:
|
||||||
if (hasSeen(&ftv))
|
if (hasSeen(&ftv))
|
||||||
return allocator->alloc<AstTypeReference>(Location(), std::nullopt, AstName("<Cycle>"), std::nullopt, Location());
|
return allocator->alloc<AstTypeReference>(Location(), std::nullopt, AstName("<Cycle>"), std::nullopt, Location());
|
||||||
|
|
||||||
AstArray<AstGenericType> generics;
|
AstArray<AstGenericType*> generics;
|
||||||
generics.size = ftv.generics.size();
|
generics.size = ftv.generics.size();
|
||||||
generics.data = static_cast<AstGenericType*>(allocator->allocate(sizeof(AstGenericType) * generics.size));
|
generics.data = static_cast<AstGenericType**>(allocator->allocate(sizeof(AstGenericType) * generics.size));
|
||||||
size_t numGenerics = 0;
|
size_t numGenerics = 0;
|
||||||
for (auto it = ftv.generics.begin(); it != ftv.generics.end(); ++it)
|
for (auto it = ftv.generics.begin(); it != ftv.generics.end(); ++it)
|
||||||
{
|
{
|
||||||
if (auto gtv = get<GenericType>(*it))
|
if (auto gtv = get<GenericType>(*it))
|
||||||
generics.data[numGenerics++] = {AstName(gtv->name.c_str()), Location(), nullptr};
|
generics.data[numGenerics++] = allocator->alloc<AstGenericType>(Location(), AstName(gtv->name.c_str()), nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
AstArray<AstGenericTypePack> genericPacks;
|
AstArray<AstGenericTypePack*> genericPacks;
|
||||||
genericPacks.size = ftv.genericPacks.size();
|
genericPacks.size = ftv.genericPacks.size();
|
||||||
genericPacks.data = static_cast<AstGenericTypePack*>(allocator->allocate(sizeof(AstGenericTypePack) * genericPacks.size));
|
genericPacks.data = static_cast<AstGenericTypePack**>(allocator->allocate(sizeof(AstGenericTypePack) * genericPacks.size));
|
||||||
size_t numGenericPacks = 0;
|
size_t numGenericPacks = 0;
|
||||||
for (auto it = ftv.genericPacks.begin(); it != ftv.genericPacks.end(); ++it)
|
for (auto it = ftv.genericPacks.begin(); it != ftv.genericPacks.end(); ++it)
|
||||||
{
|
{
|
||||||
if (auto gtv = get<GenericTypePack>(*it))
|
if (auto gtv = get<GenericTypePack>(*it))
|
||||||
genericPacks.data[numGenericPacks++] = {AstName(gtv->name.c_str()), Location(), nullptr};
|
genericPacks.data[numGenericPacks++] = allocator->alloc<AstGenericTypePack>(Location(), AstName(gtv->name.c_str()), nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
AstArray<AstType*> argTypes;
|
AstArray<AstType*> argTypes;
|
||||||
|
|
|
@ -7,7 +7,6 @@
|
||||||
#include "Luau/DcrLogger.h"
|
#include "Luau/DcrLogger.h"
|
||||||
#include "Luau/DenseHash.h"
|
#include "Luau/DenseHash.h"
|
||||||
#include "Luau/Error.h"
|
#include "Luau/Error.h"
|
||||||
#include "Luau/InsertionOrderedMap.h"
|
|
||||||
#include "Luau/Instantiation.h"
|
#include "Luau/Instantiation.h"
|
||||||
#include "Luau/Metamethods.h"
|
#include "Luau/Metamethods.h"
|
||||||
#include "Luau/Normalize.h"
|
#include "Luau/Normalize.h"
|
||||||
|
@ -27,13 +26,11 @@
|
||||||
#include "Luau/VisitType.h"
|
#include "Luau/VisitType.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <iostream>
|
|
||||||
#include <ostream>
|
|
||||||
|
|
||||||
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
LUAU_FASTFLAG(DebugLuauMagicTypes)
|
||||||
|
|
||||||
LUAU_FASTFLAG(InferGlobalTypes)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauTableKeysAreRValues)
|
LUAU_FASTFLAGVARIABLE(LuauTableKeysAreRValues)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -176,7 +173,7 @@ struct InternalTypeFunctionFinder : TypeOnceVisitor
|
||||||
DenseHashSet<TypeId> mentionedFunctions{nullptr};
|
DenseHashSet<TypeId> mentionedFunctions{nullptr};
|
||||||
DenseHashSet<TypePackId> mentionedFunctionPacks{nullptr};
|
DenseHashSet<TypePackId> mentionedFunctionPacks{nullptr};
|
||||||
|
|
||||||
InternalTypeFunctionFinder(std::vector<TypeId>& declStack)
|
explicit InternalTypeFunctionFinder(std::vector<TypeId>& declStack)
|
||||||
{
|
{
|
||||||
TypeFunctionFinder f;
|
TypeFunctionFinder f;
|
||||||
for (TypeId fn : declStack)
|
for (TypeId fn : declStack)
|
||||||
|
@ -507,7 +504,7 @@ TypeId TypeChecker2::checkForTypeFunctionInhabitance(TypeId instance, Location l
|
||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePackId TypeChecker2::lookupPack(AstExpr* expr)
|
TypePackId TypeChecker2::lookupPack(AstExpr* expr) const
|
||||||
{
|
{
|
||||||
// If a type isn't in the type graph, it probably means that a recursion limit was exceeded.
|
// If a type isn't in the type graph, it probably means that a recursion limit was exceeded.
|
||||||
// We'll just return anyType in these cases. Typechecking against any is very fast and this
|
// We'll just return anyType in these cases. Typechecking against any is very fast and this
|
||||||
|
@ -557,7 +554,7 @@ TypeId TypeChecker2::lookupAnnotation(AstType* annotation)
|
||||||
return checkForTypeFunctionInhabitance(follow(*ty), annotation->location);
|
return checkForTypeFunctionInhabitance(follow(*ty), annotation->location);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<TypePackId> TypeChecker2::lookupPackAnnotation(AstTypePack* annotation)
|
std::optional<TypePackId> TypeChecker2::lookupPackAnnotation(AstTypePack* annotation) const
|
||||||
{
|
{
|
||||||
TypePackId* tp = module->astResolvedTypePacks.find(annotation);
|
TypePackId* tp = module->astResolvedTypePacks.find(annotation);
|
||||||
if (tp != nullptr)
|
if (tp != nullptr)
|
||||||
|
@ -565,7 +562,7 @@ std::optional<TypePackId> TypeChecker2::lookupPackAnnotation(AstTypePack* annota
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId TypeChecker2::lookupExpectedType(AstExpr* expr)
|
TypeId TypeChecker2::lookupExpectedType(AstExpr* expr) const
|
||||||
{
|
{
|
||||||
if (TypeId* ty = module->astExpectedTypes.find(expr))
|
if (TypeId* ty = module->astExpectedTypes.find(expr))
|
||||||
return follow(*ty);
|
return follow(*ty);
|
||||||
|
@ -573,7 +570,7 @@ TypeId TypeChecker2::lookupExpectedType(AstExpr* expr)
|
||||||
return builtinTypes->anyType;
|
return builtinTypes->anyType;
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePackId TypeChecker2::lookupExpectedPack(AstExpr* expr, TypeArena& arena)
|
TypePackId TypeChecker2::lookupExpectedPack(AstExpr* expr, TypeArena& arena) const
|
||||||
{
|
{
|
||||||
if (TypeId* ty = module->astExpectedTypes.find(expr))
|
if (TypeId* ty = module->astExpectedTypes.find(expr))
|
||||||
return arena.addTypePack(TypePack{{follow(*ty)}, std::nullopt});
|
return arena.addTypePack(TypePack{{follow(*ty)}, std::nullopt});
|
||||||
|
@ -597,7 +594,7 @@ TypePackId TypeChecker2::reconstructPack(AstArray<AstExpr*> exprs, TypeArena& ar
|
||||||
return arena.addTypePack(TypePack{head, tail});
|
return arena.addTypePack(TypePack{head, tail});
|
||||||
}
|
}
|
||||||
|
|
||||||
Scope* TypeChecker2::findInnermostScope(Location location)
|
Scope* TypeChecker2::findInnermostScope(Location location) const
|
||||||
{
|
{
|
||||||
Scope* bestScope = module->getModuleScope().get();
|
Scope* bestScope = module->getModuleScope().get();
|
||||||
|
|
||||||
|
@ -1020,7 +1017,8 @@ void TypeChecker2::visit(AstStatForIn* forInStatement)
|
||||||
{
|
{
|
||||||
reportError(OptionalValueAccess{iteratorTy}, forInStatement->values.data[0]->location);
|
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};
|
Instantiation instantiation{TxnLog::empty(), &arena, builtinTypes, TypeLevel{}, scope};
|
||||||
|
|
||||||
|
@ -1358,7 +1356,7 @@ void TypeChecker2::visit(AstExprGlobal* expr)
|
||||||
{
|
{
|
||||||
reportError(UnknownSymbol{expr->name.value, UnknownSymbol::Binding}, expr->location);
|
reportError(UnknownSymbol{expr->name.value, UnknownSymbol::Binding}, expr->location);
|
||||||
}
|
}
|
||||||
else if (FFlag::InferGlobalTypes)
|
else
|
||||||
{
|
{
|
||||||
if (scope->shouldWarnGlobal(expr->name.value) && !warnedGlobals.contains(expr->name.value))
|
if (scope->shouldWarnGlobal(expr->name.value) && !warnedGlobals.contains(expr->name.value))
|
||||||
{
|
{
|
||||||
|
@ -1453,9 +1451,10 @@ void TypeChecker2::visitCall(AstExprCall* call)
|
||||||
TypePackId argsTp = module->internalTypes.addTypePack(args);
|
TypePackId argsTp = module->internalTypes.addTypePack(args);
|
||||||
if (auto ftv = get<FunctionType>(follow(*originalCallTy)))
|
if (auto ftv = get<FunctionType>(follow(*originalCallTy)))
|
||||||
{
|
{
|
||||||
if (ftv->dcrMagicTypeCheck)
|
if (ftv->magic)
|
||||||
{
|
{
|
||||||
ftv->dcrMagicTypeCheck(MagicFunctionTypeCheckContext{NotNull{this}, builtinTypes, call, argsTp, scope});
|
bool usedMagic = ftv->magic->typeCheck(MagicFunctionTypeCheckContext{NotNull{this}, builtinTypes, call, argsTp, scope});
|
||||||
|
if (usedMagic)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1562,7 +1561,7 @@ void TypeChecker2::visit(AstExprCall* call)
|
||||||
visitCall(call);
|
visitCall(call);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<TypeId> TypeChecker2::tryStripUnionFromNil(TypeId ty)
|
std::optional<TypeId> TypeChecker2::tryStripUnionFromNil(TypeId ty) const
|
||||||
{
|
{
|
||||||
if (const UnionType* utv = get<UnionType>(ty))
|
if (const UnionType* utv = get<UnionType>(ty))
|
||||||
{
|
{
|
||||||
|
@ -2106,7 +2105,10 @@ TypeId TypeChecker2::visit(AstExprBinary* expr, AstNode* overrideKey)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
expectedRets = module->internalTypes.addTypePack({module->internalTypes.freshType(scope, TypeLevel{})});
|
expectedRets = module->internalTypes.addTypePack(
|
||||||
|
{FFlag::LuauFreeTypesMustHaveBounds ? module->internalTypes.freshType(builtinTypes, scope, TypeLevel{})
|
||||||
|
: module->internalTypes.freshType_DEPRECATED(scope, TypeLevel{})}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId expectedTy = module->internalTypes.addType(FunctionType(expectedArgs, expectedRets));
|
TypeId expectedTy = module->internalTypes.addType(FunctionType(expectedArgs, expectedRets));
|
||||||
|
@ -2358,7 +2360,8 @@ TypeId TypeChecker2::flattenPack(TypePackId pack)
|
||||||
return *fst;
|
return *fst;
|
||||||
else if (auto ftp = get<FreeTypePack>(pack))
|
else if (auto ftp = get<FreeTypePack>(pack))
|
||||||
{
|
{
|
||||||
TypeId result = module->internalTypes.addType(FreeType{ftp->scope});
|
TypeId result = FFlag::LuauFreeTypesMustHaveBounds ? module->internalTypes.freshType(builtinTypes, ftp->scope)
|
||||||
|
: module->internalTypes.addType(FreeType{ftp->scope});
|
||||||
TypePackId freeTail = module->internalTypes.addTypePack(FreeTypePack{ftp->scope});
|
TypePackId freeTail = module->internalTypes.addTypePack(FreeTypePack{ftp->scope});
|
||||||
|
|
||||||
TypePack* resultPack = emplaceTypePack<TypePack>(asMutable(pack));
|
TypePack* resultPack = emplaceTypePack<TypePack>(asMutable(pack));
|
||||||
|
@ -2375,30 +2378,30 @@ TypeId TypeChecker2::flattenPack(TypePackId pack)
|
||||||
ice->ice("flattenPack got a weird pack!");
|
ice->ice("flattenPack got a weird pack!");
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeChecker2::visitGenerics(AstArray<AstGenericType> generics, AstArray<AstGenericTypePack> genericPacks)
|
void TypeChecker2::visitGenerics(AstArray<AstGenericType*> generics, AstArray<AstGenericTypePack*> genericPacks)
|
||||||
{
|
{
|
||||||
DenseHashSet<AstName> seen{AstName{}};
|
DenseHashSet<AstName> seen{AstName{}};
|
||||||
|
|
||||||
for (const auto& g : generics)
|
for (const auto* g : generics)
|
||||||
{
|
{
|
||||||
if (seen.contains(g.name))
|
if (seen.contains(g->name))
|
||||||
reportError(DuplicateGenericParameter{g.name.value}, g.location);
|
reportError(DuplicateGenericParameter{g->name.value}, g->location);
|
||||||
else
|
else
|
||||||
seen.insert(g.name);
|
seen.insert(g->name);
|
||||||
|
|
||||||
if (g.defaultValue)
|
if (g->defaultValue)
|
||||||
visit(g.defaultValue);
|
visit(g->defaultValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const auto& g : genericPacks)
|
for (const auto* g : genericPacks)
|
||||||
{
|
{
|
||||||
if (seen.contains(g.name))
|
if (seen.contains(g->name))
|
||||||
reportError(DuplicateGenericParameter{g.name.value}, g.location);
|
reportError(DuplicateGenericParameter{g->name.value}, g->location);
|
||||||
else
|
else
|
||||||
seen.insert(g.name);
|
seen.insert(g->name);
|
||||||
|
|
||||||
if (g.defaultValue)
|
if (g->defaultValue)
|
||||||
visit(g.defaultValue);
|
visit(g->defaultValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2420,6 +2423,8 @@ void TypeChecker2::visit(AstType* ty)
|
||||||
return visit(t);
|
return visit(t);
|
||||||
else if (auto t = ty->as<AstTypeIntersection>())
|
else if (auto t = ty->as<AstTypeIntersection>())
|
||||||
return visit(t);
|
return visit(t);
|
||||||
|
else if (auto t = ty->as<AstTypeGroup>())
|
||||||
|
return visit(t->type);
|
||||||
}
|
}
|
||||||
|
|
||||||
void TypeChecker2::visit(AstTypeReference* ty)
|
void TypeChecker2::visit(AstTypeReference* ty)
|
||||||
|
|
|
@ -47,10 +47,10 @@ LUAU_DYNAMIC_FASTINTVARIABLE(LuauTypeFamilyUseGuesserDepth, -1);
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauLogTypeFamilies)
|
LUAU_FASTFLAGVARIABLE(DebugLuauLogTypeFamilies)
|
||||||
LUAU_FASTFLAG(DebugLuauEqSatSimplification)
|
LUAU_FASTFLAG(DebugLuauEqSatSimplification)
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunPrintToError)
|
LUAU_FASTFLAGVARIABLE(LuauMetatableTypeFunctions)
|
||||||
LUAU_FASTFLAG(LuauRemoveNotAnyHack)
|
LUAU_FASTFLAGVARIABLE(LuauClipNestedAndRecursiveUnion)
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunExportedAndLocal)
|
LUAU_FASTFLAGVARIABLE(LuauDoNotGeneralizeInTypeFunctions)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunUpdateAllEnvs)
|
LUAU_FASTFLAGVARIABLE(LuauPreventReentrantTypeFunctionReduction)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -220,12 +220,9 @@ struct TypeFunctionReducer
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
void handleTypeFunctionReduction(T subject, TypeFunctionReductionResult<T> reduction)
|
void handleTypeFunctionReduction(T subject, TypeFunctionReductionResult<T> reduction)
|
||||||
{
|
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
{
|
{
|
||||||
for (auto& message : reduction.messages)
|
for (auto& message : reduction.messages)
|
||||||
result.messages.emplace_back(location, UserDefinedTypeFunctionError{std::move(message)});
|
result.messages.emplace_back(location, UserDefinedTypeFunctionError{std::move(message)});
|
||||||
}
|
|
||||||
|
|
||||||
if (reduction.result)
|
if (reduction.result)
|
||||||
replace(subject, *reduction.result);
|
replace(subject, *reduction.result);
|
||||||
|
@ -450,6 +447,18 @@ static FunctionGraphReductionResult reduceFunctionsInternal(
|
||||||
TypeFunctionReducer reducer{std::move(queuedTys), std::move(queuedTps), std::move(shouldGuess), std::move(cyclics), location, ctx, force};
|
TypeFunctionReducer reducer{std::move(queuedTys), std::move(queuedTps), std::move(shouldGuess), std::move(cyclics), location, ctx, force};
|
||||||
int iterationCount = 0;
|
int iterationCount = 0;
|
||||||
|
|
||||||
|
if (FFlag::LuauPreventReentrantTypeFunctionReduction)
|
||||||
|
{
|
||||||
|
// If we are reducing a type function while reducing a type function,
|
||||||
|
// we're probably doing something clowny. One known place this can
|
||||||
|
// occur is type function reduction => overload selection => subtyping
|
||||||
|
// => back to type function reduction. At worst, if there's a reduction
|
||||||
|
// that _doesn't_ loop forever and _needs_ reentrancy, we'll fail to
|
||||||
|
// handle that and potentially emit an error when we didn't need to.
|
||||||
|
if (ctx.normalizer->sharedState->reentrantTypeReduction)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
TypeReductionRentrancyGuard _{ctx.normalizer->sharedState};
|
||||||
while (!reducer.done())
|
while (!reducer.done())
|
||||||
{
|
{
|
||||||
reducer.step();
|
reducer.step();
|
||||||
|
@ -463,6 +472,24 @@ static FunctionGraphReductionResult reduceFunctionsInternal(
|
||||||
}
|
}
|
||||||
|
|
||||||
return std::move(reducer.result);
|
return std::move(reducer.result);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while (!reducer.done())
|
||||||
|
{
|
||||||
|
reducer.step();
|
||||||
|
|
||||||
|
++iterationCount;
|
||||||
|
if (iterationCount > DFInt::LuauTypeFamilyGraphReductionMaximumSteps)
|
||||||
|
{
|
||||||
|
reducer.result.errors.emplace_back(location, CodeTooComplex{});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return std::move(reducer.result);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
FunctionGraphReductionResult reduceTypeFunctions(TypeId entrypoint, Location location, TypeFunctionContext ctx, bool force)
|
FunctionGraphReductionResult reduceTypeFunctions(TypeId entrypoint, Location location, TypeFunctionContext ctx, bool force)
|
||||||
|
@ -617,8 +644,6 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
{
|
{
|
||||||
auto typeFunction = getMutable<TypeFunctionInstanceType>(instance);
|
auto typeFunction = getMutable<TypeFunctionInstanceType>(instance);
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
if (typeFunction->userFuncData.owner.expired())
|
if (typeFunction->userFuncData.owner.expired())
|
||||||
{
|
{
|
||||||
ctx->ice->ice("user-defined type function module has expired");
|
ctx->ice->ice("user-defined type function module has expired");
|
||||||
|
@ -630,15 +655,6 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
ctx->ice->ice("all user-defined type functions must have an associated function definition");
|
ctx->ice->ice("all user-defined type functions must have an associated function definition");
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!ctx->userFuncName)
|
|
||||||
{
|
|
||||||
ctx->ice->ice("all user-defined type functions must have an associated function definition");
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If type functions cannot be evaluated because of errors in the code, we do not generate any additional ones
|
// If type functions cannot be evaluated because of errors in the code, we do not generate any additional ones
|
||||||
if (!ctx->typeFunctionRuntime->allowEvaluation)
|
if (!ctx->typeFunctionRuntime->allowEvaluation)
|
||||||
|
@ -653,8 +669,6 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {ty}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {ty}, {}};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal && FFlag::LuauUserTypeFunUpdateAllEnvs)
|
|
||||||
{
|
|
||||||
// Ensure that whole type function environment is registered
|
// Ensure that whole type function environment is registered
|
||||||
for (auto& [name, definition] : typeFunction->userFuncData.environment)
|
for (auto& [name, definition] : typeFunction->userFuncData.environment)
|
||||||
{
|
{
|
||||||
|
@ -666,23 +680,8 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
// Ensure that whole type function environment is registered
|
|
||||||
for (auto& [name, definition] : typeFunction->userFuncData.environment_DEPRECATED)
|
|
||||||
{
|
|
||||||
if (std::optional<std::string> error = ctx->typeFunctionRuntime->registerFunction(definition))
|
|
||||||
{
|
|
||||||
// Failure to register at this point means that original definition had to error out and should not have been present in the
|
|
||||||
// environment
|
|
||||||
ctx->ice->ice("user-defined type function reference cannot be registered");
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
AstName name = FFlag::LuauUserTypeFunExportedAndLocal ? typeFunction->userFuncData.definition->name : *ctx->userFuncName;
|
AstName name = typeFunction->userFuncData.definition->name;
|
||||||
|
|
||||||
lua_State* global = ctx->typeFunctionRuntime->state.get();
|
lua_State* global = ctx->typeFunctionRuntime->state.get();
|
||||||
|
|
||||||
|
@ -693,8 +692,6 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
lua_State* L = lua_newthread(global);
|
lua_State* L = lua_newthread(global);
|
||||||
LuauTempThreadPopper popper(global);
|
LuauTempThreadPopper popper(global);
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal && FFlag::LuauUserTypeFunUpdateAllEnvs)
|
|
||||||
{
|
|
||||||
// Build up the environment table of each function we have visible
|
// Build up the environment table of each function we have visible
|
||||||
for (auto& [_, curr] : typeFunction->userFuncData.environment)
|
for (auto& [_, curr] : typeFunction->userFuncData.environment)
|
||||||
{
|
{
|
||||||
|
@ -744,45 +741,6 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
ctx->ice->ice("user-defined type function reference cannot be found in the registry");
|
ctx->ice->ice("user-defined type function reference cannot be found in the registry");
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
// Fetch the function we want to evaluate
|
|
||||||
lua_pushlightuserdata(L, typeFunction->userFuncData.definition);
|
|
||||||
lua_gettable(L, LUA_REGISTRYINDEX);
|
|
||||||
|
|
||||||
if (!lua_isfunction(L, -1))
|
|
||||||
{
|
|
||||||
ctx->ice->ice("user-defined type function reference cannot be found in the registry");
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Build up the environment
|
|
||||||
lua_getfenv(L, -1);
|
|
||||||
lua_setreadonly(L, -1, false);
|
|
||||||
|
|
||||||
for (auto& [name, definition] : typeFunction->userFuncData.environment_DEPRECATED)
|
|
||||||
{
|
|
||||||
lua_pushlightuserdata(L, definition);
|
|
||||||
lua_gettable(L, LUA_REGISTRYINDEX);
|
|
||||||
|
|
||||||
if (!lua_isfunction(L, -1))
|
|
||||||
{
|
|
||||||
ctx->ice->ice("user-defined type function reference cannot be found in the registry");
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
|
||||||
}
|
|
||||||
|
|
||||||
lua_setfield(L, -2, name.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
lua_setreadonly(L, -1, true);
|
|
||||||
lua_pop(L, 1);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
lua_getglobal(global, name.value);
|
|
||||||
lua_xmove(global, L, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
resetTypeFunctionState(L);
|
resetTypeFunctionState(L);
|
||||||
|
|
||||||
|
@ -816,26 +774,22 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
throw UserCancelError(ctx->ice->moduleName);
|
throw UserCancelError(ctx->ice->moduleName);
|
||||||
};
|
};
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
ctx->typeFunctionRuntime->messages.clear();
|
ctx->typeFunctionRuntime->messages.clear();
|
||||||
|
|
||||||
if (auto error = checkResultForError(L, name.value, lua_pcall(L, int(typeParams.size()), 1, 0)))
|
if (auto error = checkResultForError(L, name.value, lua_pcall(L, int(typeParams.size()), 1, 0)))
|
||||||
{
|
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}, error, ctx->typeFunctionRuntime->messages};
|
return {std::nullopt, Reduction::Erroneous, {}, {}, error, ctx->typeFunctionRuntime->messages};
|
||||||
else
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}, error};
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the return value is not a type userdata, return with error message
|
// If the return value is not a type userdata, return with error message
|
||||||
if (!isTypeUserData(L, 1))
|
if (!isTypeUserData(L, 1))
|
||||||
{
|
{
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
return {
|
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)};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeFunctionTypeId retTypeFunctionTypeId = getTypeUserData(L, 1);
|
TypeFunctionTypeId retTypeFunctionTypeId = getTypeUserData(L, 1);
|
||||||
|
@ -847,17 +801,9 @@ TypeFunctionReductionResult<TypeId> userDefinedTypeFunction(
|
||||||
|
|
||||||
// At least 1 error occurred while deserializing
|
// At least 1 error occurred while deserializing
|
||||||
if (runtimeBuilder->errors.size() > 0)
|
if (runtimeBuilder->errors.size() > 0)
|
||||||
{
|
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}, runtimeBuilder->errors.front(), ctx->typeFunctionRuntime->messages};
|
return {std::nullopt, Reduction::Erroneous, {}, {}, runtimeBuilder->errors.front(), ctx->typeFunctionRuntime->messages};
|
||||||
else
|
|
||||||
return {std::nullopt, Reduction::Erroneous, {}, {}, runtimeBuilder->errors.front()};
|
|
||||||
}
|
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
return {retTypeId, Reduction::MaybeOk, {}, {}, std::nullopt, ctx->typeFunctionRuntime->messages};
|
return {retTypeId, Reduction::MaybeOk, {}, {}, std::nullopt, ctx->typeFunctionRuntime->messages};
|
||||||
else
|
|
||||||
return {retTypeId, Reduction::MaybeOk, {}, {}};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeFunctionReductionResult<TypeId> notTypeFunction(
|
TypeFunctionReductionResult<TypeId> notTypeFunction(
|
||||||
|
@ -912,7 +858,7 @@ TypeFunctionReductionResult<TypeId> lenTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {operandTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {operandTy}, {}};
|
||||||
|
|
||||||
// if the type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if the type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> maybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, operandTy, /* avoidSealingTables */ true);
|
std::optional<TypeId> maybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, operandTy, /* avoidSealingTables */ true);
|
||||||
if (!maybeGeneralized)
|
if (!maybeGeneralized)
|
||||||
|
@ -1004,7 +950,7 @@ TypeFunctionReductionResult<TypeId> unmTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {operandTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {operandTy}, {}};
|
||||||
|
|
||||||
// if the type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if the type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> maybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, operandTy);
|
std::optional<TypeId> maybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, operandTy);
|
||||||
if (!maybeGeneralized)
|
if (!maybeGeneralized)
|
||||||
|
@ -1093,8 +1039,6 @@ std::optional<std::string> TypeFunctionRuntime::registerFunction(AstStatTypeFunc
|
||||||
|
|
||||||
lua_State* global = state.get();
|
lua_State* global = state.get();
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
// Fetch to check if function is already registered
|
// Fetch to check if function is already registered
|
||||||
lua_pushlightuserdata(global, function);
|
lua_pushlightuserdata(global, function);
|
||||||
lua_gettable(global, LUA_REGISTRYINDEX);
|
lua_gettable(global, LUA_REGISTRYINDEX);
|
||||||
|
@ -1106,7 +1050,6 @@ std::optional<std::string> TypeFunctionRuntime::registerFunction(AstStatTypeFunc
|
||||||
}
|
}
|
||||||
|
|
||||||
lua_pop(global, 1);
|
lua_pop(global, 1);
|
||||||
}
|
|
||||||
|
|
||||||
AstName name = function->name;
|
AstName name = function->name;
|
||||||
|
|
||||||
|
@ -1120,7 +1063,7 @@ std::optional<std::string> TypeFunctionRuntime::registerFunction(AstStatTypeFunc
|
||||||
AstStat* stmtArray[] = {&stmtReturn};
|
AstStat* stmtArray[] = {&stmtReturn};
|
||||||
AstArray<AstStat*> stmts{stmtArray, 1};
|
AstArray<AstStat*> stmts{stmtArray, 1};
|
||||||
AstStatBlock exec{Location{}, stmts};
|
AstStatBlock exec{Location{}, stmts};
|
||||||
ParseResult parseResult{&exec, 1};
|
ParseResult parseResult{&exec, 1, {}, {}, {}, CstNodeMap{nullptr}};
|
||||||
|
|
||||||
BytecodeBuilder builder;
|
BytecodeBuilder builder;
|
||||||
try
|
try
|
||||||
|
@ -1161,19 +1104,10 @@ std::optional<std::string> TypeFunctionRuntime::registerFunction(AstStatTypeFunc
|
||||||
return format("Could not find '%s' type function in the global scope", name.value);
|
return format("Could not find '%s' type function in the global scope", name.value);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunExportedAndLocal)
|
|
||||||
{
|
|
||||||
// Store resulting function in the registry
|
// Store resulting function in the registry
|
||||||
lua_pushlightuserdata(global, function);
|
lua_pushlightuserdata(global, function);
|
||||||
lua_xmove(L, global, 1);
|
lua_xmove(L, global, 1);
|
||||||
lua_settable(global, LUA_REGISTRYINDEX);
|
lua_settable(global, LUA_REGISTRYINDEX);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Store resulting function in the global environment
|
|
||||||
lua_xmove(L, global, 1);
|
|
||||||
lua_setglobal(global, name.value);
|
|
||||||
}
|
|
||||||
|
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
@ -1259,7 +1193,7 @@ TypeFunctionReductionResult<TypeId> numericBinopTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
||||||
|
|
||||||
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
||||||
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
||||||
|
@ -1496,7 +1430,7 @@ TypeFunctionReductionResult<TypeId> concatTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
||||||
|
|
||||||
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
||||||
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
||||||
|
@ -1611,7 +1545,7 @@ TypeFunctionReductionResult<TypeId> andTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
||||||
|
|
||||||
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
||||||
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
||||||
|
@ -1666,7 +1600,7 @@ TypeFunctionReductionResult<TypeId> orTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
||||||
|
|
||||||
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
||||||
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
||||||
|
@ -1752,7 +1686,7 @@ static TypeFunctionReductionResult<TypeId> comparisonTypeFunction(
|
||||||
rhsTy = follow(rhsTy);
|
rhsTy = follow(rhsTy);
|
||||||
|
|
||||||
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
||||||
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
||||||
|
@ -1890,7 +1824,7 @@ TypeFunctionReductionResult<TypeId> eqTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {rhsTy}, {}};
|
||||||
|
|
||||||
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if either type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
std::optional<TypeId> lhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, lhsTy);
|
||||||
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
std::optional<TypeId> rhsMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, rhsTy);
|
||||||
|
@ -2035,7 +1969,7 @@ TypeFunctionReductionResult<TypeId> refineTypeFunction(
|
||||||
auto stepRefine = [&ctx](TypeId target, TypeId discriminant) -> std::pair<TypeId, std::vector<TypeId>>
|
auto stepRefine = [&ctx](TypeId target, TypeId discriminant) -> std::pair<TypeId, std::vector<TypeId>>
|
||||||
{
|
{
|
||||||
std::vector<TypeId> toBlock;
|
std::vector<TypeId> toBlock;
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> targetMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, target);
|
std::optional<TypeId> targetMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, target);
|
||||||
std::optional<TypeId> discriminantMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, discriminant);
|
std::optional<TypeId> discriminantMaybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, discriminant);
|
||||||
|
@ -2064,7 +1998,7 @@ TypeFunctionReductionResult<TypeId> refineTypeFunction(
|
||||||
if (ctx->solver)
|
if (ctx->solver)
|
||||||
{
|
{
|
||||||
for (TypeId newTf : simplifyResult->newTypeFunctions)
|
for (TypeId newTf : simplifyResult->newTypeFunctions)
|
||||||
ctx->solver->pushConstraint(ctx->scope, ctx->constraint->location, ReduceConstraint{newTf});
|
ctx->pushConstraint(ReduceConstraint{newTf});
|
||||||
}
|
}
|
||||||
|
|
||||||
return {simplifyResult->result, {}};
|
return {simplifyResult->result, {}};
|
||||||
|
@ -2086,18 +2020,10 @@ TypeFunctionReductionResult<TypeId> refineTypeFunction(
|
||||||
* We need to treat T & ~any as T in this case.
|
* We need to treat T & ~any as T in this case.
|
||||||
*/
|
*/
|
||||||
if (auto nt = get<NegationType>(discriminant))
|
if (auto nt = get<NegationType>(discriminant))
|
||||||
{
|
|
||||||
if (FFlag::LuauRemoveNotAnyHack)
|
|
||||||
{
|
{
|
||||||
if (get<NoRefineType>(follow(nt->ty)))
|
if (get<NoRefineType>(follow(nt->ty)))
|
||||||
return {target, {}};
|
return {target, {}};
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
if (get<AnyType>(follow(nt->ty)))
|
|
||||||
return {target, {}};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the target type is a table, then simplification already implements the logic to deal with refinements properly since the
|
// If the target type is a table, then simplification already implements the logic to deal with refinements properly since the
|
||||||
// type of the discriminant is guaranteed to only ever be an (arbitrarily-nested) table of a single property type.
|
// type of the discriminant is guaranteed to only ever be an (arbitrarily-nested) table of a single property type.
|
||||||
|
@ -2169,7 +2095,7 @@ TypeFunctionReductionResult<TypeId> singletonTypeFunction(
|
||||||
return {std::nullopt, Reduction::MaybeOk, {type}, {}};
|
return {std::nullopt, Reduction::MaybeOk, {type}, {}};
|
||||||
|
|
||||||
// if the type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
// if the type is free but has only one remaining reference, we can generalize it to its upper bound here.
|
||||||
if (ctx->solver)
|
if (ctx->solver && !FFlag::LuauDoNotGeneralizeInTypeFunctions)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> maybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, type);
|
std::optional<TypeId> maybeGeneralized = ctx->solver->generalizeFreeType(ctx->scope, type);
|
||||||
if (!maybeGeneralized)
|
if (!maybeGeneralized)
|
||||||
|
@ -2190,6 +2116,43 @@ TypeFunctionReductionResult<TypeId> singletonTypeFunction(
|
||||||
return {ctx->builtins->unknownType, Reduction::MaybeOk, {}, {}};
|
return {ctx->builtins->unknownType, Reduction::MaybeOk, {}, {}};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct CollectUnionTypeOptions : TypeOnceVisitor
|
||||||
|
{
|
||||||
|
NotNull<TypeFunctionContext> ctx;
|
||||||
|
DenseHashSet<TypeId> options{nullptr};
|
||||||
|
DenseHashSet<TypeId> blockingTypes{nullptr};
|
||||||
|
|
||||||
|
explicit CollectUnionTypeOptions(NotNull<TypeFunctionContext> ctx)
|
||||||
|
: TypeOnceVisitor(/* skipBoundTypes */ true)
|
||||||
|
, ctx(ctx)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
bool visit(TypeId ty) override
|
||||||
|
{
|
||||||
|
options.insert(ty);
|
||||||
|
if (isPending(ty, ctx->solver))
|
||||||
|
blockingTypes.insert(ty);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool visit(TypePackId tp) override
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool visit(TypeId ty, const TypeFunctionInstanceType& tfit) override
|
||||||
|
{
|
||||||
|
if (tfit.function->name != builtinTypeFunctions().unionFunc.name)
|
||||||
|
{
|
||||||
|
options.insert(ty);
|
||||||
|
blockingTypes.insert(ty);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
TypeFunctionReductionResult<TypeId> unionTypeFunction(
|
TypeFunctionReductionResult<TypeId> unionTypeFunction(
|
||||||
TypeId instance,
|
TypeId instance,
|
||||||
const std::vector<TypeId>& typeParams,
|
const std::vector<TypeId>& typeParams,
|
||||||
|
@ -2207,6 +2170,35 @@ TypeFunctionReductionResult<TypeId> unionTypeFunction(
|
||||||
if (typeParams.size() == 1)
|
if (typeParams.size() == 1)
|
||||||
return {follow(typeParams[0]), Reduction::MaybeOk, {}, {}};
|
return {follow(typeParams[0]), Reduction::MaybeOk, {}, {}};
|
||||||
|
|
||||||
|
if (FFlag::LuauClipNestedAndRecursiveUnion)
|
||||||
|
{
|
||||||
|
|
||||||
|
CollectUnionTypeOptions collector{ctx};
|
||||||
|
collector.traverse(instance);
|
||||||
|
|
||||||
|
if (!collector.blockingTypes.empty())
|
||||||
|
{
|
||||||
|
std::vector<TypeId> blockingTypes{collector.blockingTypes.begin(), collector.blockingTypes.end()};
|
||||||
|
return {std::nullopt, Reduction::MaybeOk, std::move(blockingTypes), {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId resultTy = ctx->builtins->neverType;
|
||||||
|
for (auto ty : collector.options)
|
||||||
|
{
|
||||||
|
SimplifyResult result = simplifyUnion(ctx->builtins, ctx->arena, resultTy, ty);
|
||||||
|
// This condition might fire if one of the arguments to this type
|
||||||
|
// function is a free type somewhere deep in a nested union or
|
||||||
|
// intersection type, even though we ran a pass above to capture
|
||||||
|
// some blocked types.
|
||||||
|
if (!result.blockedTypes.empty())
|
||||||
|
return {std::nullopt, Reduction::MaybeOk, {result.blockedTypes.begin(), result.blockedTypes.end()}, {}};
|
||||||
|
|
||||||
|
resultTy = result.result;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {resultTy, Reduction::MaybeOk, {}, {}};
|
||||||
|
}
|
||||||
|
|
||||||
// we need to follow all of the type parameters.
|
// we need to follow all of the type parameters.
|
||||||
std::vector<TypeId> types;
|
std::vector<TypeId> types;
|
||||||
types.reserve(typeParams.size());
|
types.reserve(typeParams.size());
|
||||||
|
@ -2278,14 +2270,11 @@ TypeFunctionReductionResult<TypeId> intersectTypeFunction(
|
||||||
for (auto ty : typeParams)
|
for (auto ty : typeParams)
|
||||||
types.emplace_back(follow(ty));
|
types.emplace_back(follow(ty));
|
||||||
|
|
||||||
if (FFlag::LuauRemoveNotAnyHack)
|
|
||||||
{
|
|
||||||
// if we only have two parameters and one is `*no-refine*`, we're all done.
|
// if we only have two parameters and one is `*no-refine*`, we're all done.
|
||||||
if (types.size() == 2 && get<NoRefineType>(types[1]))
|
if (types.size() == 2 && get<NoRefineType>(types[1]))
|
||||||
return {types[0], Reduction::MaybeOk, {}, {}};
|
return {types[0], Reduction::MaybeOk, {}, {}};
|
||||||
else if (types.size() == 2 && get<NoRefineType>(types[0]))
|
else if (types.size() == 2 && get<NoRefineType>(types[0]))
|
||||||
return {types[1], Reduction::MaybeOk, {}, {}};
|
return {types[1], Reduction::MaybeOk, {}, {}};
|
||||||
}
|
|
||||||
|
|
||||||
// check to see if the operand types are resolved enough, and wait to reduce if not
|
// check to see if the operand types are resolved enough, and wait to reduce if not
|
||||||
// if any of them are `never`, the intersection will always be `never`, so we can reduce directly.
|
// if any of them are `never`, the intersection will always be `never`, so we can reduce directly.
|
||||||
|
@ -2302,7 +2291,7 @@ TypeFunctionReductionResult<TypeId> intersectTypeFunction(
|
||||||
for (auto ty : types)
|
for (auto ty : types)
|
||||||
{
|
{
|
||||||
// skip any `*no-refine*` types.
|
// skip any `*no-refine*` types.
|
||||||
if (FFlag::LuauRemoveNotAnyHack && get<NoRefineType>(ty))
|
if (get<NoRefineType>(ty))
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
SimplifyResult result = simplifyIntersection(ctx->builtins, ctx->arena, resultTy, ty);
|
SimplifyResult result = simplifyIntersection(ctx->builtins, ctx->arena, resultTy, ty);
|
||||||
|
@ -2821,6 +2810,215 @@ TypeFunctionReductionResult<TypeId> rawgetTypeFunction(
|
||||||
return indexFunctionImpl(typeParams, packParams, ctx, /* isRaw */ true);
|
return indexFunctionImpl(typeParams, packParams, ctx, /* isRaw */ true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TypeFunctionReductionResult<TypeId> setmetatableTypeFunction(
|
||||||
|
TypeId instance,
|
||||||
|
const std::vector<TypeId>& typeParams,
|
||||||
|
const std::vector<TypePackId>& packParams,
|
||||||
|
NotNull<TypeFunctionContext> ctx
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (typeParams.size() != 2 || !packParams.empty())
|
||||||
|
{
|
||||||
|
ctx->ice->ice("setmetatable type function: encountered a type function instance without the required argument structure");
|
||||||
|
LUAU_ASSERT(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Location location = ctx->constraint ? ctx->constraint->location : Location{};
|
||||||
|
|
||||||
|
TypeId targetTy = follow(typeParams.at(0));
|
||||||
|
TypeId metatableTy = follow(typeParams.at(1));
|
||||||
|
|
||||||
|
std::shared_ptr<const NormalizedType> targetNorm = ctx->normalizer->normalize(targetTy);
|
||||||
|
|
||||||
|
// if the operand failed to normalize, we can't reduce, but know nothing about inhabitance.
|
||||||
|
if (!targetNorm)
|
||||||
|
return {std::nullopt, Reduction::MaybeOk, {}, {}};
|
||||||
|
|
||||||
|
// cannot setmetatable on something without table parts.
|
||||||
|
if (!targetNorm->hasTables())
|
||||||
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
|
|
||||||
|
// we're trying to reject any type that has not normalized to a table or a union/intersection of tables.
|
||||||
|
if (targetNorm->hasTops() || targetNorm->hasBooleans() || targetNorm->hasErrors() || targetNorm->hasNils() ||
|
||||||
|
targetNorm->hasNumbers() || targetNorm->hasStrings() || targetNorm->hasThreads() || targetNorm->hasBuffers() ||
|
||||||
|
targetNorm->hasFunctions() || targetNorm->hasTyvars() || targetNorm->hasClasses())
|
||||||
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
|
|
||||||
|
// if the supposed metatable is not a table, we will fail to reduce.
|
||||||
|
if (!get<TableType>(metatableTy) && !get<MetatableType>(metatableTy))
|
||||||
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
|
|
||||||
|
if (targetNorm->tables.size() == 1)
|
||||||
|
{
|
||||||
|
TypeId table = *targetNorm->tables.begin();
|
||||||
|
|
||||||
|
// findMetatableEntry demands the ability to emit errors, so we must give it
|
||||||
|
// the necessary state to do that, even if we intend to just eat the errors.
|
||||||
|
ErrorVec dummy;
|
||||||
|
|
||||||
|
std::optional<TypeId> metatableMetamethod = findMetatableEntry(ctx->builtins, dummy, table, "__metatable", location);
|
||||||
|
|
||||||
|
// if the `__metatable` metamethod is present, then the table is locked and we cannot `setmetatable` on it.
|
||||||
|
if (metatableMetamethod)
|
||||||
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
|
|
||||||
|
TypeId withMetatable = ctx->arena->addType(MetatableType{table, metatableTy});
|
||||||
|
|
||||||
|
return {withMetatable, Reduction::MaybeOk, {}, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeId result = ctx->builtins->neverType;
|
||||||
|
|
||||||
|
for (auto componentTy : targetNorm->tables)
|
||||||
|
{
|
||||||
|
// findMetatableEntry demands the ability to emit errors, so we must give it
|
||||||
|
// the necessary state to do that, even if we intend to just eat the errors.
|
||||||
|
ErrorVec dummy;
|
||||||
|
|
||||||
|
std::optional<TypeId> metatableMetamethod = findMetatableEntry(ctx->builtins, dummy, componentTy, "__metatable", location);
|
||||||
|
|
||||||
|
// if the `__metatable` metamethod is present, then the table is locked and we cannot `setmetatable` on it.
|
||||||
|
if (metatableMetamethod)
|
||||||
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
|
|
||||||
|
TypeId withMetatable = ctx->arena->addType(MetatableType{componentTy, metatableTy});
|
||||||
|
SimplifyResult simplified = simplifyUnion(ctx->builtins, ctx->arena, result, withMetatable);
|
||||||
|
|
||||||
|
if (!simplified.blockedTypes.empty())
|
||||||
|
{
|
||||||
|
std::vector<TypeId> blockedTypes{};
|
||||||
|
blockedTypes.reserve(simplified.blockedTypes.size());
|
||||||
|
for (auto ty : simplified.blockedTypes)
|
||||||
|
blockedTypes.push_back(ty);
|
||||||
|
return {std::nullopt, Reduction::MaybeOk, blockedTypes, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
result = simplified.result;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {result, Reduction::MaybeOk, {}, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
static TypeFunctionReductionResult<TypeId> getmetatableHelper(
|
||||||
|
TypeId targetTy,
|
||||||
|
const Location& location,
|
||||||
|
NotNull<TypeFunctionContext> ctx
|
||||||
|
)
|
||||||
|
{
|
||||||
|
targetTy = follow(targetTy);
|
||||||
|
|
||||||
|
std::optional<TypeId> metatable = std::nullopt;
|
||||||
|
bool erroneous = true;
|
||||||
|
|
||||||
|
if (auto table = get<TableType>(targetTy))
|
||||||
|
erroneous = false;
|
||||||
|
|
||||||
|
if (auto mt = get<MetatableType>(targetTy))
|
||||||
|
{
|
||||||
|
metatable = mt->metatable;
|
||||||
|
erroneous = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto clazz = get<ClassType>(targetTy))
|
||||||
|
{
|
||||||
|
metatable = clazz->metatable;
|
||||||
|
erroneous = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto primitive = get<PrimitiveType>(targetTy))
|
||||||
|
{
|
||||||
|
metatable = primitive->metatable;
|
||||||
|
erroneous = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto singleton = get<SingletonType>(targetTy))
|
||||||
|
{
|
||||||
|
if (get<StringSingleton>(singleton))
|
||||||
|
{
|
||||||
|
auto primitiveString = get<PrimitiveType>(ctx->builtins->stringType);
|
||||||
|
metatable = primitiveString->metatable;
|
||||||
|
}
|
||||||
|
erroneous = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (erroneous)
|
||||||
|
return {std::nullopt, Reduction::Erroneous, {}, {}};
|
||||||
|
|
||||||
|
// findMetatableEntry demands the ability to emit errors, so we must give it
|
||||||
|
// the necessary state to do that, even if we intend to just eat the errors.
|
||||||
|
ErrorVec dummy;
|
||||||
|
|
||||||
|
std::optional<TypeId> metatableMetamethod = findMetatableEntry(ctx->builtins, dummy, targetTy, "__metatable", location);
|
||||||
|
|
||||||
|
if (metatableMetamethod)
|
||||||
|
return {metatableMetamethod, Reduction::MaybeOk, {}, {}};
|
||||||
|
|
||||||
|
if (metatable)
|
||||||
|
return {metatable, Reduction::MaybeOk, {}, {}};
|
||||||
|
|
||||||
|
return {ctx->builtins->nilType, Reduction::MaybeOk, {}, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeFunctionReductionResult<TypeId> getmetatableTypeFunction(
|
||||||
|
TypeId instance,
|
||||||
|
const std::vector<TypeId>& typeParams,
|
||||||
|
const std::vector<TypePackId>& packParams,
|
||||||
|
NotNull<TypeFunctionContext> ctx
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (typeParams.size() != 1 || !packParams.empty())
|
||||||
|
{
|
||||||
|
ctx->ice->ice("getmetatable type function: encountered a type function instance without the required argument structure");
|
||||||
|
LUAU_ASSERT(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
const Location location = ctx->constraint ? ctx->constraint->location : Location{};
|
||||||
|
|
||||||
|
TypeId targetTy = follow(typeParams.at(0));
|
||||||
|
|
||||||
|
if (isPending(targetTy, ctx->solver))
|
||||||
|
return {std::nullopt, Reduction::MaybeOk, {targetTy}, {}};
|
||||||
|
|
||||||
|
if (auto ut = get<UnionType>(targetTy))
|
||||||
|
{
|
||||||
|
std::vector<TypeId> options{};
|
||||||
|
options.reserve(ut->options.size());
|
||||||
|
|
||||||
|
for (auto option : ut->options)
|
||||||
|
{
|
||||||
|
TypeFunctionReductionResult<TypeId> result = getmetatableHelper(option, location, ctx);
|
||||||
|
|
||||||
|
if (!result.result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
options.push_back(*result.result);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {ctx->arena->addType(UnionType{std::move(options)}), Reduction::MaybeOk, {}, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto it = get<IntersectionType>(targetTy))
|
||||||
|
{
|
||||||
|
std::vector<TypeId> parts{};
|
||||||
|
parts.reserve(it->parts.size());
|
||||||
|
|
||||||
|
for (auto part : it->parts)
|
||||||
|
{
|
||||||
|
TypeFunctionReductionResult<TypeId> result = getmetatableHelper(part, location, ctx);
|
||||||
|
|
||||||
|
if (!result.result)
|
||||||
|
return result;
|
||||||
|
|
||||||
|
parts.push_back(*result.result);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {ctx->arena->addType(IntersectionType{std::move(parts)}), Reduction::MaybeOk, {}, {}};
|
||||||
|
}
|
||||||
|
|
||||||
|
return getmetatableHelper(targetTy, location, ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
BuiltinTypeFunctions::BuiltinTypeFunctions()
|
BuiltinTypeFunctions::BuiltinTypeFunctions()
|
||||||
: userFunc{"user", userDefinedTypeFunction}
|
: userFunc{"user", userDefinedTypeFunction}
|
||||||
, notFunc{"not", notTypeFunction}
|
, notFunc{"not", notTypeFunction}
|
||||||
|
@ -2847,6 +3045,8 @@ BuiltinTypeFunctions::BuiltinTypeFunctions()
|
||||||
, rawkeyofFunc{"rawkeyof", rawkeyofTypeFunction}
|
, rawkeyofFunc{"rawkeyof", rawkeyofTypeFunction}
|
||||||
, indexFunc{"index", indexTypeFunction}
|
, indexFunc{"index", indexTypeFunction}
|
||||||
, rawgetFunc{"rawget", rawgetTypeFunction}
|
, rawgetFunc{"rawget", rawgetTypeFunction}
|
||||||
|
, setmetatableFunc{"setmetatable", setmetatableTypeFunction}
|
||||||
|
, getmetatableFunc{"getmetatable", getmetatableTypeFunction}
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2893,6 +3093,12 @@ void BuiltinTypeFunctions::addToScope(NotNull<TypeArena> arena, NotNull<Scope> s
|
||||||
|
|
||||||
scope->exportedTypeBindings[indexFunc.name] = mkBinaryTypeFunction(&indexFunc);
|
scope->exportedTypeBindings[indexFunc.name] = mkBinaryTypeFunction(&indexFunc);
|
||||||
scope->exportedTypeBindings[rawgetFunc.name] = mkBinaryTypeFunction(&rawgetFunc);
|
scope->exportedTypeBindings[rawgetFunc.name] = mkBinaryTypeFunction(&rawgetFunc);
|
||||||
|
|
||||||
|
if (FFlag::LuauMetatableTypeFunctions)
|
||||||
|
{
|
||||||
|
scope->exportedTypeBindings[setmetatableFunc.name] = mkBinaryTypeFunction(&setmetatableFunc);
|
||||||
|
scope->exportedTypeBindings[getmetatableFunc.name] = mkUnaryTypeFunction(&getmetatableFunc);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const BuiltinTypeFunctions& builtinTypeFunctions()
|
const BuiltinTypeFunctions& builtinTypeFunctions()
|
||||||
|
|
|
@ -14,9 +14,6 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunPrintToError)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixNoReadWrite)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunThreadBuffer)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunTypeofReturnsType)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunTypeofReturnsType)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
|
@ -135,11 +132,9 @@ static std::string getTag(lua_State* L, TypeFunctionTypeId ty)
|
||||||
return "number";
|
return "number";
|
||||||
else if (auto s = get<TypeFunctionPrimitiveType>(ty); s && s->type == TypeFunctionPrimitiveType::Type::String)
|
else if (auto s = get<TypeFunctionPrimitiveType>(ty); s && s->type == TypeFunctionPrimitiveType::Type::String)
|
||||||
return "string";
|
return "string";
|
||||||
else if (auto s = get<TypeFunctionPrimitiveType>(ty);
|
else if (auto s = get<TypeFunctionPrimitiveType>(ty); s && s->type == TypeFunctionPrimitiveType::Type::Thread)
|
||||||
FFlag::LuauUserTypeFunThreadBuffer && s && s->type == TypeFunctionPrimitiveType::Type::Thread)
|
|
||||||
return "thread";
|
return "thread";
|
||||||
else if (auto s = get<TypeFunctionPrimitiveType>(ty);
|
else if (auto s = get<TypeFunctionPrimitiveType>(ty); s && s->type == TypeFunctionPrimitiveType::Type::Buffer)
|
||||||
FFlag::LuauUserTypeFunThreadBuffer && s && s->type == TypeFunctionPrimitiveType::Type::Buffer)
|
|
||||||
return "buffer";
|
return "buffer";
|
||||||
else if (get<TypeFunctionUnknownType>(ty))
|
else if (get<TypeFunctionUnknownType>(ty))
|
||||||
return "unknown";
|
return "unknown";
|
||||||
|
@ -161,6 +156,8 @@ static std::string getTag(lua_State* L, TypeFunctionTypeId ty)
|
||||||
return "function";
|
return "function";
|
||||||
else if (get<TypeFunctionClassType>(ty))
|
else if (get<TypeFunctionClassType>(ty))
|
||||||
return "class";
|
return "class";
|
||||||
|
else if (get<TypeFunctionGenericType>(ty))
|
||||||
|
return "generic";
|
||||||
|
|
||||||
LUAU_UNREACHABLE();
|
LUAU_UNREACHABLE();
|
||||||
luaL_error(L, "VM encountered unexpected type variant when determining tag");
|
luaL_error(L, "VM encountered unexpected type variant when determining tag");
|
||||||
|
@ -267,6 +264,20 @@ static int createSingleton(lua_State* L)
|
||||||
luaL_error(L, "types.singleton: can't create singleton from `%s` type", lua_typename(L, 1));
|
luaL_error(L, "types.singleton: can't create singleton from `%s` type", lua_typename(L, 1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Luau: `types.generic(name: string, ispack: boolean?) -> type
|
||||||
|
// Create a generic type with the specified type. If an optinal boolean is set to true, result is a generic pack
|
||||||
|
static int createGeneric(lua_State* L)
|
||||||
|
{
|
||||||
|
const char* name = luaL_checkstring(L, 1);
|
||||||
|
bool isPack = luaL_optboolean(L, 2, false);
|
||||||
|
|
||||||
|
if (strlen(name) == 0)
|
||||||
|
luaL_error(L, "types.generic: generic name cannot be empty");
|
||||||
|
|
||||||
|
allocTypeUserData(L, TypeFunctionGenericType{/* isNamed */ true, isPack, name});
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
// Luau: `self:value() -> type`
|
// Luau: `self:value() -> type`
|
||||||
// Returns the value of a singleton
|
// Returns the value of a singleton
|
||||||
static int getSingletonValue(lua_State* L)
|
static int getSingletonValue(lua_State* L)
|
||||||
|
@ -414,7 +425,8 @@ static int getNegatedValue(lua_State* L)
|
||||||
luaL_error(L, "type.inner: expected 1 argument, but got %d", argumentCount);
|
luaL_error(L, "type.inner: expected 1 argument, but got %d", argumentCount);
|
||||||
|
|
||||||
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
||||||
if (auto tfnt = get<TypeFunctionNegationType>(self); !tfnt)
|
|
||||||
|
if (auto tfnt = get<TypeFunctionNegationType>(self); tfnt)
|
||||||
allocTypeUserData(L, tfnt->type->type);
|
allocTypeUserData(L, tfnt->type->type);
|
||||||
else
|
else
|
||||||
luaL_error(L, "type.inner: cannot call inner method on non-negation type: `%s` type", getTag(L, self).c_str());
|
luaL_error(L, "type.inner: cannot call inner method on non-negation type: `%s` type", getTag(L, self).c_str());
|
||||||
|
@ -658,10 +670,8 @@ static int readTableProp(lua_State* L)
|
||||||
auto prop = tftt->props.at(tfsst->value);
|
auto prop = tftt->props.at(tfsst->value);
|
||||||
if (prop.readTy)
|
if (prop.readTy)
|
||||||
allocTypeUserData(L, (*prop.readTy)->type);
|
allocTypeUserData(L, (*prop.readTy)->type);
|
||||||
else if (FFlag::LuauUserTypeFunFixNoReadWrite)
|
|
||||||
lua_pushnil(L);
|
|
||||||
else
|
else
|
||||||
luaL_error(L, "type.readproperty: property %s is write-only, and therefore does not have a read type.", tfsst->value.c_str());
|
lua_pushnil(L);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -698,10 +708,8 @@ static int writeTableProp(lua_State* L)
|
||||||
auto prop = tftt->props.at(tfsst->value);
|
auto prop = tftt->props.at(tfsst->value);
|
||||||
if (prop.writeTy)
|
if (prop.writeTy)
|
||||||
allocTypeUserData(L, (*prop.writeTy)->type);
|
allocTypeUserData(L, (*prop.writeTy)->type);
|
||||||
else if (FFlag::LuauUserTypeFunFixNoReadWrite)
|
|
||||||
lua_pushnil(L);
|
|
||||||
else
|
else
|
||||||
luaL_error(L, "type.writeproperty: property %s is read-only, and therefore does not have a write type.", tfsst->value.c_str());
|
lua_pushnil(L);
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -769,97 +777,209 @@ static int setTableMetatable(lua_State* L)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Luau: `types.newfunction(parameters: {head: {type}?, tail: type?}, returns: {head: {type}?, tail: type?}) -> type`
|
static std::tuple<std::vector<TypeFunctionTypeId>, std::vector<TypeFunctionTypePackId>> getGenerics(lua_State* L, int idx, const char* fname)
|
||||||
|
{
|
||||||
|
std::vector<TypeFunctionTypeId> types;
|
||||||
|
std::vector<TypeFunctionTypePackId> packs;
|
||||||
|
|
||||||
|
if (lua_istable(L, idx))
|
||||||
|
{
|
||||||
|
lua_pushvalue(L, idx);
|
||||||
|
|
||||||
|
for (int i = 1; i <= lua_objlen(L, -1); i++)
|
||||||
|
{
|
||||||
|
lua_pushinteger(L, i);
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
|
||||||
|
if (lua_isnil(L, -1))
|
||||||
|
{
|
||||||
|
lua_pop(L, 1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeFunctionTypeId ty = getTypeUserData(L, -1);
|
||||||
|
|
||||||
|
if (auto gty = get<TypeFunctionGenericType>(ty))
|
||||||
|
{
|
||||||
|
if (gty->isPack)
|
||||||
|
{
|
||||||
|
packs.push_back(allocateTypeFunctionTypePack(L, TypeFunctionGenericTypePack{gty->isNamed, gty->name}));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (!packs.empty())
|
||||||
|
luaL_error(L, "%s: generic type cannot follow a generic pack", fname);
|
||||||
|
|
||||||
|
types.push_back(ty);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
luaL_error(L, "%s: table member was not a generic type", fname);
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pop(L, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pop(L, 1);
|
||||||
|
}
|
||||||
|
else if (!lua_isnoneornil(L, idx))
|
||||||
|
{
|
||||||
|
luaL_typeerrorL(L, idx, "table");
|
||||||
|
}
|
||||||
|
|
||||||
|
return {types, packs};
|
||||||
|
}
|
||||||
|
|
||||||
|
static TypeFunctionTypePackId getTypePack(lua_State* L, int headIdx, int tailIdx)
|
||||||
|
{
|
||||||
|
TypeFunctionTypePackId result = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
||||||
|
|
||||||
|
std::vector<TypeFunctionTypeId> head;
|
||||||
|
|
||||||
|
if (lua_istable(L, headIdx))
|
||||||
|
{
|
||||||
|
lua_pushvalue(L, headIdx);
|
||||||
|
|
||||||
|
for (int i = 1; i <= lua_objlen(L, -1); i++)
|
||||||
|
{
|
||||||
|
lua_pushinteger(L, i);
|
||||||
|
lua_gettable(L, -2);
|
||||||
|
|
||||||
|
if (lua_isnil(L, -1))
|
||||||
|
{
|
||||||
|
lua_pop(L, 1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
head.push_back(getTypeUserData(L, -1));
|
||||||
|
lua_pop(L, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_pop(L, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<TypeFunctionTypePackId> tail;
|
||||||
|
|
||||||
|
if (auto type = optionalTypeUserData(L, tailIdx))
|
||||||
|
{
|
||||||
|
if (auto gty = get<TypeFunctionGenericType>(*type); gty && gty->isPack)
|
||||||
|
tail = allocateTypeFunctionTypePack(L, TypeFunctionGenericTypePack{gty->isNamed, gty->name});
|
||||||
|
else
|
||||||
|
tail = allocateTypeFunctionTypePack(L, TypeFunctionVariadicTypePack{*type});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (head.size() == 0 && tail.has_value())
|
||||||
|
result = *tail;
|
||||||
|
else
|
||||||
|
result = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void pushTypePack(lua_State* L, TypeFunctionTypePackId tp)
|
||||||
|
{
|
||||||
|
if (auto tftp = get<TypeFunctionTypePack>(tp))
|
||||||
|
{
|
||||||
|
lua_createtable(L, 0, 2);
|
||||||
|
|
||||||
|
if (!tftp->head.empty())
|
||||||
|
{
|
||||||
|
lua_createtable(L, int(tftp->head.size()), 0);
|
||||||
|
int pos = 1;
|
||||||
|
|
||||||
|
for (auto el : tftp->head)
|
||||||
|
{
|
||||||
|
allocTypeUserData(L, el->type);
|
||||||
|
lua_rawseti(L, -2, pos++);
|
||||||
|
}
|
||||||
|
|
||||||
|
lua_setfield(L, -2, "head");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tftp->tail.has_value())
|
||||||
|
{
|
||||||
|
if (auto tfvp = get<TypeFunctionVariadicTypePack>(*tftp->tail))
|
||||||
|
allocTypeUserData(L, tfvp->type->type);
|
||||||
|
else if (auto tfgp = get<TypeFunctionGenericTypePack>(*tftp->tail))
|
||||||
|
allocTypeUserData(L, TypeFunctionGenericType{tfgp->isNamed, true, tfgp->name});
|
||||||
|
else
|
||||||
|
luaL_error(L, "unsupported type pack type");
|
||||||
|
|
||||||
|
lua_setfield(L, -2, "tail");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (auto tfvp = get<TypeFunctionVariadicTypePack>(tp))
|
||||||
|
{
|
||||||
|
lua_createtable(L, 0, 1);
|
||||||
|
|
||||||
|
allocTypeUserData(L, tfvp->type->type);
|
||||||
|
lua_setfield(L, -2, "tail");
|
||||||
|
}
|
||||||
|
else if (auto tfgp = get<TypeFunctionGenericTypePack>(tp))
|
||||||
|
{
|
||||||
|
lua_createtable(L, 0, 1);
|
||||||
|
|
||||||
|
allocTypeUserData(L, TypeFunctionGenericType{tfgp->isNamed, true, tfgp->name});
|
||||||
|
lua_setfield(L, -2, "tail");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
luaL_error(L, "unsupported type pack type");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Luau: `types.newfunction(parameters: {head: {type}?, tail: type?}, returns: {head: {type}?, tail: type?}, generics: {type}?) -> type`
|
||||||
// Returns the type instance representing a function
|
// Returns the type instance representing a function
|
||||||
static int createFunction(lua_State* L)
|
static int createFunction(lua_State* L)
|
||||||
{
|
{
|
||||||
int argumentCount = lua_gettop(L);
|
int argumentCount = lua_gettop(L);
|
||||||
if (argumentCount > 2)
|
if (argumentCount > 3)
|
||||||
luaL_error(L, "types.newfunction: expected 0-2 arguments, but got %d", argumentCount);
|
luaL_error(L, "types.newfunction: expected 0-3 arguments, but got %d", argumentCount);
|
||||||
|
|
||||||
|
TypeFunctionTypePackId argTypes = nullptr;
|
||||||
|
|
||||||
TypeFunctionTypePackId argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
|
||||||
if (lua_istable(L, 1))
|
if (lua_istable(L, 1))
|
||||||
{
|
{
|
||||||
std::vector<TypeFunctionTypeId> head{};
|
|
||||||
lua_getfield(L, 1, "head");
|
lua_getfield(L, 1, "head");
|
||||||
if (lua_istable(L, -1))
|
|
||||||
{
|
|
||||||
int argSize = lua_objlen(L, -1);
|
|
||||||
for (int i = 1; i <= argSize; i++)
|
|
||||||
{
|
|
||||||
lua_pushinteger(L, i);
|
|
||||||
lua_gettable(L, -2);
|
|
||||||
|
|
||||||
if (lua_isnil(L, -1))
|
|
||||||
{
|
|
||||||
lua_pop(L, 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
TypeFunctionTypeId ty = getTypeUserData(L, -1);
|
|
||||||
head.push_back(ty);
|
|
||||||
|
|
||||||
lua_pop(L, 1); // Remove `ty` from stack
|
|
||||||
}
|
|
||||||
}
|
|
||||||
lua_pop(L, 1); // Pop the "head" field
|
|
||||||
|
|
||||||
std::optional<TypeFunctionTypePackId> tail;
|
|
||||||
lua_getfield(L, 1, "tail");
|
lua_getfield(L, 1, "tail");
|
||||||
if (auto type = optionalTypeUserData(L, -1))
|
|
||||||
tail = allocateTypeFunctionTypePack(L, TypeFunctionVariadicTypePack{*type});
|
|
||||||
lua_pop(L, 1); // Pop the "tail" field
|
|
||||||
|
|
||||||
if (head.size() == 0 && tail.has_value())
|
argTypes = getTypePack(L, -2, -1);
|
||||||
argTypes = *tail;
|
|
||||||
else
|
lua_pop(L, 2);
|
||||||
argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
|
||||||
}
|
}
|
||||||
else if (!lua_isnoneornil(L, 1))
|
else if (!lua_isnoneornil(L, 1))
|
||||||
|
{
|
||||||
luaL_typeerrorL(L, 1, "table");
|
luaL_typeerrorL(L, 1, "table");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeFunctionTypePackId retTypes = nullptr;
|
||||||
|
|
||||||
TypeFunctionTypePackId retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
|
||||||
if (lua_istable(L, 2))
|
if (lua_istable(L, 2))
|
||||||
{
|
{
|
||||||
std::vector<TypeFunctionTypeId> head{};
|
|
||||||
lua_getfield(L, 2, "head");
|
lua_getfield(L, 2, "head");
|
||||||
if (lua_istable(L, -1))
|
|
||||||
{
|
|
||||||
int argSize = lua_objlen(L, -1);
|
|
||||||
for (int i = 1; i <= argSize; i++)
|
|
||||||
{
|
|
||||||
lua_pushinteger(L, i);
|
|
||||||
lua_gettable(L, -2);
|
|
||||||
|
|
||||||
if (lua_isnil(L, -1))
|
|
||||||
{
|
|
||||||
lua_pop(L, 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
TypeFunctionTypeId ty = getTypeUserData(L, -1);
|
|
||||||
head.push_back(ty);
|
|
||||||
|
|
||||||
lua_pop(L, 1); // Remove `ty` from stack
|
|
||||||
}
|
|
||||||
}
|
|
||||||
lua_pop(L, 1); // Pop the "head" field
|
|
||||||
|
|
||||||
std::optional<TypeFunctionTypePackId> tail;
|
|
||||||
lua_getfield(L, 2, "tail");
|
lua_getfield(L, 2, "tail");
|
||||||
if (auto type = optionalTypeUserData(L, -1))
|
|
||||||
tail = allocateTypeFunctionTypePack(L, TypeFunctionVariadicTypePack{*type});
|
|
||||||
lua_pop(L, 1); // Pop the "tail" field
|
|
||||||
|
|
||||||
if (head.size() == 0 && tail.has_value())
|
retTypes = getTypePack(L, -2, -1);
|
||||||
retTypes = *tail;
|
|
||||||
else
|
lua_pop(L, 2);
|
||||||
retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
|
||||||
}
|
}
|
||||||
else if (!lua_isnoneornil(L, 2))
|
else if (!lua_isnoneornil(L, 2))
|
||||||
|
{
|
||||||
luaL_typeerrorL(L, 2, "table");
|
luaL_typeerrorL(L, 2, "table");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
||||||
|
}
|
||||||
|
|
||||||
allocTypeUserData(L, TypeFunctionFunctionType{argTypes, retTypes});
|
auto [genericTypes, genericPacks] = getGenerics(L, 3, "types.newfunction");
|
||||||
|
|
||||||
|
allocTypeUserData(L, TypeFunctionFunctionType{std::move(genericTypes), std::move(genericPacks), argTypes, retTypes});
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -877,38 +997,7 @@ static int setFunctionParameters(lua_State* L)
|
||||||
if (!tfft)
|
if (!tfft)
|
||||||
luaL_error(L, "type.setparameters: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
luaL_error(L, "type.setparameters: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
std::vector<TypeFunctionTypeId> head{};
|
tfft->argTypes = getTypePack(L, 2, 3);
|
||||||
if (lua_istable(L, 2))
|
|
||||||
{
|
|
||||||
int argSize = lua_objlen(L, 2);
|
|
||||||
for (int i = 1; i <= argSize; i++)
|
|
||||||
{
|
|
||||||
lua_pushinteger(L, i);
|
|
||||||
lua_gettable(L, 2);
|
|
||||||
|
|
||||||
if (lua_isnil(L, -1))
|
|
||||||
{
|
|
||||||
lua_pop(L, 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
TypeFunctionTypeId ty = getTypeUserData(L, -1);
|
|
||||||
head.push_back(ty);
|
|
||||||
|
|
||||||
lua_pop(L, 1); // Remove `ty` from stack
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (!lua_isnoneornil(L, 2))
|
|
||||||
luaL_typeerrorL(L, 2, "table");
|
|
||||||
|
|
||||||
std::optional<TypeFunctionTypePackId> tail;
|
|
||||||
if (auto type = optionalTypeUserData(L, 3))
|
|
||||||
tail = allocateTypeFunctionTypePack(L, TypeFunctionVariadicTypePack{*type});
|
|
||||||
|
|
||||||
if (head.size() == 0 && tail.has_value()) // Make argTypes a variadic type pack
|
|
||||||
tfft->argTypes = *tail;
|
|
||||||
else // Make argTypes a type pack
|
|
||||||
tfft->argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -926,52 +1015,8 @@ static int getFunctionParameters(lua_State* L)
|
||||||
if (!tfft)
|
if (!tfft)
|
||||||
luaL_error(L, "type.parameters: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
luaL_error(L, "type.parameters: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
if (auto tftp = get<TypeFunctionTypePack>(tfft->argTypes))
|
pushTypePack(L, tfft->argTypes);
|
||||||
{
|
|
||||||
int size = 0;
|
|
||||||
if (tftp->head.size() > 0)
|
|
||||||
size++;
|
|
||||||
if (tftp->tail.has_value())
|
|
||||||
size++;
|
|
||||||
|
|
||||||
lua_createtable(L, 0, size);
|
|
||||||
|
|
||||||
int argSize = (int)tftp->head.size();
|
|
||||||
if (argSize > 0)
|
|
||||||
{
|
|
||||||
lua_createtable(L, argSize, 0);
|
|
||||||
for (int i = 0; i < argSize; i++)
|
|
||||||
{
|
|
||||||
allocTypeUserData(L, tftp->head[i]->type);
|
|
||||||
lua_rawseti(L, -2, i + 1); // Luau is 1-indexed while C++ is 0-indexed
|
|
||||||
}
|
|
||||||
lua_setfield(L, -2, "head");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tftp->tail.has_value())
|
|
||||||
{
|
|
||||||
auto tfvp = get<TypeFunctionVariadicTypePack>(*tftp->tail);
|
|
||||||
if (!tfvp)
|
|
||||||
LUAU_ASSERT(!"We should only be supporting variadic packs as TypeFunctionTypePack.tail at the moment");
|
|
||||||
|
|
||||||
allocTypeUserData(L, tfvp->type->type);
|
|
||||||
lua_setfield(L, -2, "tail");
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (auto tfvp = get<TypeFunctionVariadicTypePack>(tfft->argTypes))
|
|
||||||
{
|
|
||||||
lua_createtable(L, 0, 1);
|
|
||||||
|
|
||||||
allocTypeUserData(L, tfvp->type->type);
|
|
||||||
lua_setfield(L, -2, "tail");
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
lua_createtable(L, 0, 0);
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -988,38 +1033,7 @@ static int setFunctionReturns(lua_State* L)
|
||||||
if (!tfft)
|
if (!tfft)
|
||||||
luaL_error(L, "type.setreturns: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
luaL_error(L, "type.setreturns: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
std::vector<TypeFunctionTypeId> head{};
|
tfft->retTypes = getTypePack(L, 2, 3);
|
||||||
if (lua_istable(L, 2))
|
|
||||||
{
|
|
||||||
int argSize = lua_objlen(L, 2);
|
|
||||||
for (int i = 1; i <= argSize; i++)
|
|
||||||
{
|
|
||||||
lua_pushinteger(L, i);
|
|
||||||
lua_gettable(L, 2);
|
|
||||||
|
|
||||||
if (lua_isnil(L, -1))
|
|
||||||
{
|
|
||||||
lua_pop(L, 1);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
TypeFunctionTypeId ty = getTypeUserData(L, -1);
|
|
||||||
head.push_back(ty);
|
|
||||||
|
|
||||||
lua_pop(L, 1); // Remove `ty` from stack
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (!lua_isnoneornil(L, 2))
|
|
||||||
luaL_typeerrorL(L, 2, "table");
|
|
||||||
|
|
||||||
std::optional<TypeFunctionTypePackId> tail;
|
|
||||||
if (auto type = optionalTypeUserData(L, 3))
|
|
||||||
tail = allocateTypeFunctionTypePack(L, TypeFunctionVariadicTypePack{*type});
|
|
||||||
|
|
||||||
if (head.size() == 0 && tail.has_value()) // Make retTypes a variadic type pack
|
|
||||||
tfft->retTypes = *tail;
|
|
||||||
else // Make retTypes a type pack
|
|
||||||
tfft->retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1037,52 +1051,57 @@ static int getFunctionReturns(lua_State* L)
|
||||||
if (!tfft)
|
if (!tfft)
|
||||||
luaL_error(L, "type.returns: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
luaL_error(L, "type.returns: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
if (auto tftp = get<TypeFunctionTypePack>(tfft->retTypes))
|
pushTypePack(L, tfft->retTypes);
|
||||||
{
|
|
||||||
int size = 0;
|
|
||||||
if (tftp->head.size() > 0)
|
|
||||||
size++;
|
|
||||||
if (tftp->tail.has_value())
|
|
||||||
size++;
|
|
||||||
|
|
||||||
lua_createtable(L, 0, size);
|
|
||||||
|
|
||||||
int argSize = (int)tftp->head.size();
|
|
||||||
if (argSize > 0)
|
|
||||||
{
|
|
||||||
lua_createtable(L, argSize, 0);
|
|
||||||
for (int i = 0; i < argSize; i++)
|
|
||||||
{
|
|
||||||
allocTypeUserData(L, tftp->head[i]->type);
|
|
||||||
lua_rawseti(L, -2, i + 1); // Luau is 1-indexed while C++ is 0-indexed
|
|
||||||
}
|
|
||||||
lua_setfield(L, -2, "head");
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tftp->tail.has_value())
|
|
||||||
{
|
|
||||||
auto tfvp = get<TypeFunctionVariadicTypePack>(*tftp->tail);
|
|
||||||
if (!tfvp)
|
|
||||||
LUAU_ASSERT(!"We should only be supporting variadic packs as TypeFunctionTypePack.tail at the moment");
|
|
||||||
|
|
||||||
allocTypeUserData(L, tfvp->type->type);
|
|
||||||
lua_setfield(L, -2, "tail");
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (auto tfvp = get<TypeFunctionVariadicTypePack>(tfft->retTypes))
|
// Luau: `self:setgenerics(generics: {type}?)`
|
||||||
|
static int setFunctionGenerics(lua_State* L)
|
||||||
|
{
|
||||||
|
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
||||||
|
auto tfft = getMutable<TypeFunctionFunctionType>(self);
|
||||||
|
if (!tfft)
|
||||||
|
luaL_error(L, "type.setgenerics: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
|
int argumentCount = lua_gettop(L);
|
||||||
|
if (argumentCount > 3)
|
||||||
|
luaL_error(L, "type.setgenerics: expected 3 arguments, but got %d", argumentCount);
|
||||||
|
|
||||||
|
auto [genericTypes, genericPacks] = getGenerics(L, 2, "types.setgenerics");
|
||||||
|
|
||||||
|
tfft->generics = std::move(genericTypes);
|
||||||
|
tfft->genericPacks = std::move(genericPacks);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Luau: `self:generics() -> {type}`
|
||||||
|
static int getFunctionGenerics(lua_State* L)
|
||||||
|
{
|
||||||
|
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
||||||
|
auto tfft = get<TypeFunctionFunctionType>(self);
|
||||||
|
if (!tfft)
|
||||||
|
luaL_error(L, "type.generics: expected self to be a function, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
|
lua_createtable(L, int(tfft->generics.size()) + int(tfft->genericPacks.size()), 0);
|
||||||
|
|
||||||
|
int pos = 1;
|
||||||
|
|
||||||
|
for (const auto& el : tfft->generics)
|
||||||
{
|
{
|
||||||
lua_createtable(L, 0, 1);
|
allocTypeUserData(L, el->type);
|
||||||
|
lua_rawseti(L, -2, pos++);
|
||||||
allocTypeUserData(L, tfvp->type->type);
|
}
|
||||||
lua_setfield(L, -2, "tail");
|
|
||||||
|
for (const auto& el : tfft->genericPacks)
|
||||||
return 1;
|
{
|
||||||
|
auto gty = get<TypeFunctionGenericTypePack>(el);
|
||||||
|
LUAU_ASSERT(gty);
|
||||||
|
allocTypeUserData(L, TypeFunctionGenericType{gty->isNamed, true, gty->name});
|
||||||
|
lua_rawseti(L, -2, pos++);
|
||||||
}
|
}
|
||||||
|
|
||||||
lua_createtable(L, 0, 0);
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1108,6 +1127,36 @@ static int getClassParent(lua_State* L)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Luau: `self:name() -> string?`
|
||||||
|
// Returns the name of the generic or 'nil' if the generic is unnamed
|
||||||
|
static int getGenericName(lua_State* L)
|
||||||
|
{
|
||||||
|
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
||||||
|
auto tfgt = get<TypeFunctionGenericType>(self);
|
||||||
|
if (!tfgt)
|
||||||
|
luaL_error(L, "type.name: expected self to be a generic, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
|
if (tfgt->isNamed)
|
||||||
|
lua_pushstring(L, tfgt->name.c_str());
|
||||||
|
else
|
||||||
|
lua_pushnil(L);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Luau: `self:ispack() -> boolean`
|
||||||
|
// Returns true if the generic is a pack
|
||||||
|
static int getGenericIsPack(lua_State* L)
|
||||||
|
{
|
||||||
|
TypeFunctionTypeId self = getTypeUserData(L, 1);
|
||||||
|
auto tfgt = get<TypeFunctionGenericType>(self);
|
||||||
|
if (!tfgt)
|
||||||
|
luaL_error(L, "type.ispack: expected self to be a generic, but got %s instead", getTag(L, self).c_str());
|
||||||
|
|
||||||
|
lua_pushboolean(L, tfgt->isPack);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
// Luau: `self:properties() -> {[type]: { read: type?, write: type? }}`
|
// Luau: `self:properties() -> {[type]: { read: type?, write: type? }}`
|
||||||
// Returns the properties of a table or class type
|
// Returns the properties of a table or class type
|
||||||
static int getProps(lua_State* L)
|
static int getProps(lua_State* L)
|
||||||
|
@ -1377,7 +1426,7 @@ static int checkTag(lua_State* L)
|
||||||
|
|
||||||
TypeFunctionTypeId deepClone(NotNull<TypeFunctionRuntime> runtime, TypeFunctionTypeId ty); // Forward declaration
|
TypeFunctionTypeId deepClone(NotNull<TypeFunctionRuntime> runtime, TypeFunctionTypeId ty); // Forward declaration
|
||||||
|
|
||||||
// Luau: `types.copy(arg: string) -> type`
|
// Luau: `types.copy(arg: type) -> type`
|
||||||
// Returns a deep copy of the argument
|
// Returns a deep copy of the argument
|
||||||
static int deepCopy(lua_State* L)
|
static int deepCopy(lua_State* L)
|
||||||
{
|
{
|
||||||
|
@ -1416,8 +1465,8 @@ void registerTypesLibrary(lua_State* L)
|
||||||
{"boolean", createBoolean},
|
{"boolean", createBoolean},
|
||||||
{"number", createNumber},
|
{"number", createNumber},
|
||||||
{"string", createString},
|
{"string", createString},
|
||||||
{FFlag::LuauUserTypeFunThreadBuffer ? "thread" : nullptr, FFlag::LuauUserTypeFunThreadBuffer ? createThread : nullptr},
|
{"thread", createThread},
|
||||||
{FFlag::LuauUserTypeFunThreadBuffer ? "buffer" : nullptr, FFlag::LuauUserTypeFunThreadBuffer ? createBuffer : nullptr},
|
{"buffer", createBuffer},
|
||||||
{nullptr, nullptr}
|
{nullptr, nullptr}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1429,6 +1478,7 @@ void registerTypesLibrary(lua_State* L)
|
||||||
{"newtable", createTable},
|
{"newtable", createTable},
|
||||||
{"newfunction", createFunction},
|
{"newfunction", createFunction},
|
||||||
{"copy", deepCopy},
|
{"copy", deepCopy},
|
||||||
|
{"generic", createGeneric},
|
||||||
|
|
||||||
{nullptr, nullptr}
|
{nullptr, nullptr}
|
||||||
};
|
};
|
||||||
|
@ -1493,6 +1543,8 @@ void registerTypeUserData(lua_State* L)
|
||||||
{"parameters", getFunctionParameters},
|
{"parameters", getFunctionParameters},
|
||||||
{"setreturns", setFunctionReturns},
|
{"setreturns", setFunctionReturns},
|
||||||
{"returns", getFunctionReturns},
|
{"returns", getFunctionReturns},
|
||||||
|
{"setgenerics", setFunctionGenerics},
|
||||||
|
{"generics", getFunctionGenerics},
|
||||||
|
|
||||||
// Union and Intersection type methods
|
// Union and Intersection type methods
|
||||||
{"components", getComponents},
|
{"components", getComponents},
|
||||||
|
@ -1500,6 +1552,14 @@ void registerTypeUserData(lua_State* L)
|
||||||
// Class type methods
|
// Class type methods
|
||||||
{"parent", getClassParent},
|
{"parent", getClassParent},
|
||||||
|
|
||||||
|
// Function type methods (cont.)
|
||||||
|
{"setgenerics", setFunctionGenerics},
|
||||||
|
{"generics", getFunctionGenerics},
|
||||||
|
|
||||||
|
// Generic type methods
|
||||||
|
{"name", getGenericName},
|
||||||
|
{"ispack", getGenericIsPack},
|
||||||
|
|
||||||
{nullptr, nullptr}
|
{nullptr, nullptr}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1593,8 +1653,6 @@ void setTypeFunctionEnvironment(lua_State* L)
|
||||||
luaopen_base(L);
|
luaopen_base(L);
|
||||||
lua_pop(L, 1);
|
lua_pop(L, 1);
|
||||||
|
|
||||||
if (FFlag::LuauUserTypeFunPrintToError)
|
|
||||||
{
|
|
||||||
// Remove certain global functions from the base library
|
// Remove certain global functions from the base library
|
||||||
static const char* unavailableGlobals[] = {"gcinfo", "getfenv", "newproxy", "setfenv", "pcall", "xpcall"};
|
static const char* unavailableGlobals[] = {"gcinfo", "getfenv", "newproxy", "setfenv", "pcall", "xpcall"};
|
||||||
for (auto& name : unavailableGlobals)
|
for (auto& name : unavailableGlobals)
|
||||||
|
@ -1605,17 +1663,6 @@ void setTypeFunctionEnvironment(lua_State* L)
|
||||||
|
|
||||||
lua_pushcfunction(L, print, "print");
|
lua_pushcfunction(L, print, "print");
|
||||||
lua_setglobal(L, "print");
|
lua_setglobal(L, "print");
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Remove certain global functions from the base library
|
|
||||||
static const std::string unavailableGlobals[] = {"gcinfo", "getfenv", "newproxy", "setfenv", "pcall", "xpcall"};
|
|
||||||
for (auto& name : unavailableGlobals)
|
|
||||||
{
|
|
||||||
lua_pushcfunction(L, unsupportedFunction, "Removing global function from type function environment");
|
|
||||||
lua_setglobal(L, name.c_str());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void resetTypeFunctionState(lua_State* L)
|
void resetTypeFunctionState(lua_State* L)
|
||||||
|
@ -1771,6 +1818,24 @@ bool areEqual(SeenSet& seen, const TypeFunctionFunctionType& lhs, const TypeFunc
|
||||||
if (seenSetContains(seen, &lhs, &rhs))
|
if (seenSetContains(seen, &lhs, &rhs))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
if (lhs.generics.size() != rhs.generics.size())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
for (auto l = lhs.generics.begin(), r = rhs.generics.begin(); l != lhs.generics.end() && r != rhs.generics.end(); ++l, ++r)
|
||||||
|
{
|
||||||
|
if (!areEqual(seen, **l, **r))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lhs.genericPacks.size() != rhs.genericPacks.size())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
for (auto l = lhs.genericPacks.begin(), r = rhs.genericPacks.begin(); l != lhs.genericPacks.end() && r != rhs.genericPacks.end(); ++l, ++r)
|
||||||
|
{
|
||||||
|
if (!areEqual(seen, **l, **r))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (bool(lhs.argTypes) != bool(rhs.argTypes))
|
if (bool(lhs.argTypes) != bool(rhs.argTypes))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
@ -1871,6 +1936,13 @@ bool areEqual(SeenSet& seen, const TypeFunctionType& lhs, const TypeFunctionType
|
||||||
return areEqual(seen, *lf, *rf);
|
return areEqual(seen, *lf, *rf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const TypeFunctionGenericType* lg = get<TypeFunctionGenericType>(&lhs);
|
||||||
|
const TypeFunctionGenericType* rg = get<TypeFunctionGenericType>(&rhs);
|
||||||
|
if (lg && rg)
|
||||||
|
return lg->isNamed == rg->isNamed && lg->isPack == rg->isPack && lg->name == rg->name;
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1917,6 +1989,13 @@ bool areEqual(SeenSet& seen, const TypeFunctionTypePackVar& lhs, const TypeFunct
|
||||||
return areEqual(seen, *lv, *rv);
|
return areEqual(seen, *lv, *rv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const TypeFunctionGenericTypePack* lg = get<TypeFunctionGenericTypePack>(&lhs);
|
||||||
|
const TypeFunctionGenericTypePack* rg = get<TypeFunctionGenericTypePack>(&rhs);
|
||||||
|
if (lg && rg)
|
||||||
|
return lg->isNamed == rg->isNamed && lg->name == rg->name;
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2106,11 +2185,9 @@ private:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
||||||
break;
|
break;
|
||||||
case TypeFunctionPrimitiveType::Thread:
|
case TypeFunctionPrimitiveType::Thread:
|
||||||
if (FFlag::LuauUserTypeFunThreadBuffer)
|
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Thread));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Thread));
|
||||||
break;
|
break;
|
||||||
case TypeFunctionPrimitiveType::Buffer:
|
case TypeFunctionPrimitiveType::Buffer:
|
||||||
if (FFlag::LuauUserTypeFunThreadBuffer)
|
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Buffer));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Buffer));
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -2141,10 +2218,14 @@ private:
|
||||||
else if (auto f = get<TypeFunctionFunctionType>(ty))
|
else if (auto f = get<TypeFunctionFunctionType>(ty))
|
||||||
{
|
{
|
||||||
TypeFunctionTypePackId emptyTypePack = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{});
|
TypeFunctionTypePackId emptyTypePack = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{});
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionFunctionType{emptyTypePack, emptyTypePack});
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionFunctionType{{}, {}, emptyTypePack, emptyTypePack});
|
||||||
}
|
}
|
||||||
else if (auto c = get<TypeFunctionClassType>(ty))
|
else if (auto c = get<TypeFunctionClassType>(ty))
|
||||||
target = ty; // Don't copy a class since they are immutable
|
target = ty; // Don't copy a class since they are immutable
|
||||||
|
else if (auto g = get<TypeFunctionGenericType>(ty))
|
||||||
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionGenericType{g->isNamed, g->isPack, g->name});
|
||||||
|
else
|
||||||
|
LUAU_ASSERT(!"Unknown type");
|
||||||
|
|
||||||
types[ty] = target;
|
types[ty] = target;
|
||||||
queue.emplace_back(ty, target);
|
queue.emplace_back(ty, target);
|
||||||
|
@ -2162,6 +2243,10 @@ private:
|
||||||
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{{}});
|
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{{}});
|
||||||
else if (auto vPack = get<TypeFunctionVariadicTypePack>(tp))
|
else if (auto vPack = get<TypeFunctionVariadicTypePack>(tp))
|
||||||
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionVariadicTypePack{});
|
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionVariadicTypePack{});
|
||||||
|
else if (auto gPack = get<TypeFunctionGenericTypePack>(tp))
|
||||||
|
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionGenericTypePack{gPack->isNamed, gPack->name});
|
||||||
|
else
|
||||||
|
LUAU_ASSERT(!"Unknown type");
|
||||||
|
|
||||||
packs[tp] = target;
|
packs[tp] = target;
|
||||||
queue.emplace_back(tp, target);
|
queue.emplace_back(tp, target);
|
||||||
|
@ -2192,6 +2277,8 @@ private:
|
||||||
cloneChildren(f1, f2);
|
cloneChildren(f1, f2);
|
||||||
else if (auto [c1, c2] = std::tuple{getMutable<TypeFunctionClassType>(ty), getMutable<TypeFunctionClassType>(tfti)}; c1 && c2)
|
else if (auto [c1, c2] = std::tuple{getMutable<TypeFunctionClassType>(ty), getMutable<TypeFunctionClassType>(tfti)}; c1 && c2)
|
||||||
cloneChildren(c1, c2);
|
cloneChildren(c1, c2);
|
||||||
|
else if (auto [g1, g2] = std::tuple{getMutable<TypeFunctionGenericType>(ty), getMutable<TypeFunctionGenericType>(tfti)}; g1 && g2)
|
||||||
|
cloneChildren(g1, g2);
|
||||||
else
|
else
|
||||||
LUAU_ASSERT(!"Unknown pair?"); // First and argument should always represent the same types
|
LUAU_ASSERT(!"Unknown pair?"); // First and argument should always represent the same types
|
||||||
}
|
}
|
||||||
|
@ -2203,6 +2290,9 @@ private:
|
||||||
else if (auto [vPack1, vPack2] = std::tuple{getMutable<TypeFunctionVariadicTypePack>(tp), getMutable<TypeFunctionVariadicTypePack>(tftp)};
|
else if (auto [vPack1, vPack2] = std::tuple{getMutable<TypeFunctionVariadicTypePack>(tp), getMutable<TypeFunctionVariadicTypePack>(tftp)};
|
||||||
vPack1 && vPack2)
|
vPack1 && vPack2)
|
||||||
cloneChildren(vPack1, vPack2);
|
cloneChildren(vPack1, vPack2);
|
||||||
|
else if (auto [gPack1, gPack2] = std::tuple{getMutable<TypeFunctionGenericTypePack>(tp), getMutable<TypeFunctionGenericTypePack>(tftp)};
|
||||||
|
gPack1 && gPack2)
|
||||||
|
cloneChildren(gPack1, gPack2);
|
||||||
else
|
else
|
||||||
LUAU_ASSERT(!"Unknown pair?"); // First and argument should always represent the same types
|
LUAU_ASSERT(!"Unknown pair?"); // First and argument should always represent the same types
|
||||||
}
|
}
|
||||||
|
@ -2283,6 +2373,14 @@ private:
|
||||||
|
|
||||||
void cloneChildren(TypeFunctionFunctionType* f1, TypeFunctionFunctionType* f2)
|
void cloneChildren(TypeFunctionFunctionType* f1, TypeFunctionFunctionType* f2)
|
||||||
{
|
{
|
||||||
|
f2->generics.reserve(f1->generics.size());
|
||||||
|
for (auto ty : f1->generics)
|
||||||
|
f2->generics.push_back(shallowClone(ty));
|
||||||
|
|
||||||
|
f2->genericPacks.reserve(f1->genericPacks.size());
|
||||||
|
for (auto tp : f1->genericPacks)
|
||||||
|
f2->genericPacks.push_back(shallowClone(tp));
|
||||||
|
|
||||||
f2->argTypes = shallowClone(f1->argTypes);
|
f2->argTypes = shallowClone(f1->argTypes);
|
||||||
f2->retTypes = shallowClone(f1->retTypes);
|
f2->retTypes = shallowClone(f1->retTypes);
|
||||||
}
|
}
|
||||||
|
@ -2292,16 +2390,29 @@ private:
|
||||||
// noop.
|
// noop.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void cloneChildren(TypeFunctionGenericType* g1, TypeFunctionGenericType* g2)
|
||||||
|
{
|
||||||
|
// noop.
|
||||||
|
}
|
||||||
|
|
||||||
void cloneChildren(TypeFunctionTypePack* t1, TypeFunctionTypePack* t2)
|
void cloneChildren(TypeFunctionTypePack* t1, TypeFunctionTypePack* t2)
|
||||||
{
|
{
|
||||||
for (TypeFunctionTypeId& ty : t1->head)
|
for (TypeFunctionTypeId& ty : t1->head)
|
||||||
t2->head.push_back(shallowClone(ty));
|
t2->head.push_back(shallowClone(ty));
|
||||||
|
|
||||||
|
if (t1->tail)
|
||||||
|
t2->tail = shallowClone(*t1->tail);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cloneChildren(TypeFunctionVariadicTypePack* v1, TypeFunctionVariadicTypePack* v2)
|
void cloneChildren(TypeFunctionVariadicTypePack* v1, TypeFunctionVariadicTypePack* v2)
|
||||||
{
|
{
|
||||||
v2->type = shallowClone(v1->type);
|
v2->type = shallowClone(v1->type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void cloneChildren(TypeFunctionGenericTypePack* g1, TypeFunctionGenericTypePack* g2)
|
||||||
|
{
|
||||||
|
// noop.
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
TypeFunctionTypeId deepClone(NotNull<TypeFunctionRuntime> runtime, TypeFunctionTypeId ty)
|
TypeFunctionTypeId deepClone(NotNull<TypeFunctionRuntime> runtime, TypeFunctionTypeId ty)
|
||||||
|
|
|
@ -20,8 +20,6 @@
|
||||||
// currently, controls serialization, deserialization, and `type.copy`
|
// currently, controls serialization, deserialization, and `type.copy`
|
||||||
LUAU_DYNAMIC_FASTINTVARIABLE(LuauTypeFunctionSerdeIterationLimit, 100'000);
|
LUAU_DYNAMIC_FASTINTVARIABLE(LuauTypeFunctionSerdeIterationLimit, 100'000);
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunThreadBuffer)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -160,26 +158,10 @@ private:
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::String));
|
||||||
break;
|
break;
|
||||||
case PrimitiveType::Thread:
|
case PrimitiveType::Thread:
|
||||||
if (FFlag::LuauUserTypeFunThreadBuffer)
|
|
||||||
{
|
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Thread));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Thread));
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
std::string error = format("Argument of primitive type %s is not currently serializable by type functions", toString(ty).c_str());
|
|
||||||
state->errors.push_back(error);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case PrimitiveType::Buffer:
|
case PrimitiveType::Buffer:
|
||||||
if (FFlag::LuauUserTypeFunThreadBuffer)
|
|
||||||
{
|
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Buffer));
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionPrimitiveType(TypeFunctionPrimitiveType::Buffer));
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
std::string error = format("Argument of primitive type %s is not currently serializable by type functions", toString(ty).c_str());
|
|
||||||
state->errors.push_back(error);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
case PrimitiveType::Function:
|
case PrimitiveType::Function:
|
||||||
case PrimitiveType::Table:
|
case PrimitiveType::Table:
|
||||||
|
@ -221,13 +203,22 @@ private:
|
||||||
else if (auto f = get<FunctionType>(ty))
|
else if (auto f = get<FunctionType>(ty))
|
||||||
{
|
{
|
||||||
TypeFunctionTypePackId emptyTypePack = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{});
|
TypeFunctionTypePackId emptyTypePack = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{});
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionFunctionType{emptyTypePack, emptyTypePack});
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionFunctionType{{}, {}, emptyTypePack, emptyTypePack});
|
||||||
}
|
}
|
||||||
else if (auto c = get<ClassType>(ty))
|
else if (auto c = get<ClassType>(ty))
|
||||||
{
|
{
|
||||||
state->classesSerialized[c->name] = ty;
|
state->classesSerialized[c->name] = ty;
|
||||||
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionClassType{{}, std::nullopt, std::nullopt, std::nullopt, c->name});
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionClassType{{}, std::nullopt, std::nullopt, std::nullopt, c->name});
|
||||||
}
|
}
|
||||||
|
else if (auto g = get<GenericType>(ty))
|
||||||
|
{
|
||||||
|
Name name = g->name;
|
||||||
|
|
||||||
|
if (!g->explicitName)
|
||||||
|
name = format("g%d", g->index);
|
||||||
|
|
||||||
|
target = typeFunctionRuntime->typeArena.allocate(TypeFunctionGenericType{g->explicitName, false, name});
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
std::string error = format("Argument of type %s is not currently serializable by type functions", toString(ty).c_str());
|
std::string error = format("Argument of type %s is not currently serializable by type functions", toString(ty).c_str());
|
||||||
|
@ -252,6 +243,15 @@ private:
|
||||||
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{{}});
|
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionTypePack{{}});
|
||||||
else if (auto vPack = get<VariadicTypePack>(tp))
|
else if (auto vPack = get<VariadicTypePack>(tp))
|
||||||
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionVariadicTypePack{});
|
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionVariadicTypePack{});
|
||||||
|
else if (auto gPack = get<GenericTypePack>(tp))
|
||||||
|
{
|
||||||
|
Name name = gPack->name;
|
||||||
|
|
||||||
|
if (!gPack->explicitName)
|
||||||
|
name = format("g%d", gPack->index);
|
||||||
|
|
||||||
|
target = typeFunctionRuntime->typePackArena.allocate(TypeFunctionGenericTypePack{gPack->explicitName, name});
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
std::string error = format("Argument of type pack %s is not currently serializable by type functions", toString(tp).c_str());
|
std::string error = format("Argument of type pack %s is not currently serializable by type functions", toString(tp).c_str());
|
||||||
|
@ -289,6 +289,8 @@ private:
|
||||||
serializeChildren(f1, f2);
|
serializeChildren(f1, f2);
|
||||||
else if (auto [c1, c2] = std::tuple{get<ClassType>(ty), getMutable<TypeFunctionClassType>(tfti)}; c1 && c2)
|
else if (auto [c1, c2] = std::tuple{get<ClassType>(ty), getMutable<TypeFunctionClassType>(tfti)}; c1 && c2)
|
||||||
serializeChildren(c1, c2);
|
serializeChildren(c1, c2);
|
||||||
|
else if (auto [g1, g2] = std::tuple{get<GenericType>(ty), getMutable<TypeFunctionGenericType>(tfti)}; g1 && g2)
|
||||||
|
serializeChildren(g1, g2);
|
||||||
else
|
else
|
||||||
{ // Either this or ty and tfti do not represent the same type
|
{ // Either this or ty and tfti do not represent the same type
|
||||||
std::string error = format("Argument of type %s is not currently serializable by type functions", toString(ty).c_str());
|
std::string error = format("Argument of type %s is not currently serializable by type functions", toString(ty).c_str());
|
||||||
|
@ -302,6 +304,8 @@ private:
|
||||||
serializeChildren(tPack1, tPack2);
|
serializeChildren(tPack1, tPack2);
|
||||||
else if (auto [vPack1, vPack2] = std::tuple{get<VariadicTypePack>(tp), getMutable<TypeFunctionVariadicTypePack>(tftp)}; vPack1 && vPack2)
|
else if (auto [vPack1, vPack2] = std::tuple{get<VariadicTypePack>(tp), getMutable<TypeFunctionVariadicTypePack>(tftp)}; vPack1 && vPack2)
|
||||||
serializeChildren(vPack1, vPack2);
|
serializeChildren(vPack1, vPack2);
|
||||||
|
else if (auto [gPack1, gPack2] = std::tuple{get<GenericTypePack>(tp), getMutable<TypeFunctionGenericTypePack>(tftp)}; gPack1 && gPack2)
|
||||||
|
serializeChildren(gPack1, gPack2);
|
||||||
else
|
else
|
||||||
{ // Either this or ty and tfti do not represent the same type
|
{ // Either this or ty and tfti do not represent the same type
|
||||||
std::string error = format("Argument of type pack %s is not currently serializable by type functions", toString(tp).c_str());
|
std::string error = format("Argument of type pack %s is not currently serializable by type functions", toString(tp).c_str());
|
||||||
|
@ -391,6 +395,14 @@ private:
|
||||||
|
|
||||||
void serializeChildren(const FunctionType* f1, TypeFunctionFunctionType* f2)
|
void serializeChildren(const FunctionType* f1, TypeFunctionFunctionType* f2)
|
||||||
{
|
{
|
||||||
|
f2->generics.reserve(f1->generics.size());
|
||||||
|
for (auto ty : f1->generics)
|
||||||
|
f2->generics.push_back(shallowSerialize(ty));
|
||||||
|
|
||||||
|
f2->genericPacks.reserve(f1->genericPacks.size());
|
||||||
|
for (auto tp : f1->genericPacks)
|
||||||
|
f2->genericPacks.push_back(shallowSerialize(tp));
|
||||||
|
|
||||||
f2->argTypes = shallowSerialize(f1->argTypes);
|
f2->argTypes = shallowSerialize(f1->argTypes);
|
||||||
f2->retTypes = shallowSerialize(f1->retTypes);
|
f2->retTypes = shallowSerialize(f1->retTypes);
|
||||||
}
|
}
|
||||||
|
@ -420,6 +432,11 @@ private:
|
||||||
c2->parent = shallowSerialize(*c1->parent);
|
c2->parent = shallowSerialize(*c1->parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void serializeChildren(const GenericType* g1, TypeFunctionGenericType* g2)
|
||||||
|
{
|
||||||
|
// noop.
|
||||||
|
}
|
||||||
|
|
||||||
void serializeChildren(const TypePack* t1, TypeFunctionTypePack* t2)
|
void serializeChildren(const TypePack* t1, TypeFunctionTypePack* t2)
|
||||||
{
|
{
|
||||||
for (const TypeId& ty : t1->head)
|
for (const TypeId& ty : t1->head)
|
||||||
|
@ -433,6 +450,25 @@ private:
|
||||||
{
|
{
|
||||||
v2->type = shallowSerialize(v1->ty);
|
v2->type = shallowSerialize(v1->ty);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void serializeChildren(const GenericTypePack* v1, TypeFunctionGenericTypePack* v2)
|
||||||
|
{
|
||||||
|
// noop.
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
struct SerializedGeneric
|
||||||
|
{
|
||||||
|
bool isNamed = false;
|
||||||
|
std::string name;
|
||||||
|
T type = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SerializedFunctionScope
|
||||||
|
{
|
||||||
|
size_t oldQueueSize = 0;
|
||||||
|
TypeFunctionFunctionType* function = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Complete inverse of TypeFunctionSerializer
|
// Complete inverse of TypeFunctionSerializer
|
||||||
|
@ -453,6 +489,15 @@ class TypeFunctionDeserializer
|
||||||
// second must be PrimitiveType; else there should be an error
|
// second must be PrimitiveType; else there should be an error
|
||||||
std::vector<std::tuple<TypeFunctionKind, Kind>> queue;
|
std::vector<std::tuple<TypeFunctionKind, Kind>> queue;
|
||||||
|
|
||||||
|
// Generic types and packs currently in scope
|
||||||
|
// Generics are resolved by name even if runtime generic type pointers are different
|
||||||
|
// Multiple names mapping to the same generic can be in scope for nested generic functions
|
||||||
|
std::vector<SerializedGeneric<TypeId>> genericTypes;
|
||||||
|
std::vector<SerializedGeneric<TypePackId>> genericPacks;
|
||||||
|
|
||||||
|
// To track when generics go out of scope, we have a list of queue positions at which a specific function has introduced generics
|
||||||
|
std::vector<SerializedFunctionScope> functionScopes;
|
||||||
|
|
||||||
SeenTypes types; // Mapping of TypeFunctionTypeIds that have been shallow deserialized to TypeIds
|
SeenTypes types; // Mapping of TypeFunctionTypeIds that have been shallow deserialized to TypeIds
|
||||||
SeenTypePacks packs; // Mapping of TypeFunctionTypePackIds that have been shallow deserialized to TypePackIds
|
SeenTypePacks packs; // Mapping of TypeFunctionTypePackIds that have been shallow deserialized to TypePackIds
|
||||||
|
|
||||||
|
@ -464,7 +509,9 @@ public:
|
||||||
, typeFunctionRuntime(state->ctx->typeFunctionRuntime)
|
, typeFunctionRuntime(state->ctx->typeFunctionRuntime)
|
||||||
, queue({})
|
, queue({})
|
||||||
, types({})
|
, types({})
|
||||||
, packs({}){};
|
, packs({})
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
TypeId deserialize(TypeFunctionTypeId ty)
|
TypeId deserialize(TypeFunctionTypeId ty)
|
||||||
{
|
{
|
||||||
|
@ -518,6 +565,13 @@ private:
|
||||||
queue.pop_back();
|
queue.pop_back();
|
||||||
|
|
||||||
deserializeChildren(tfti, ty);
|
deserializeChildren(tfti, ty);
|
||||||
|
|
||||||
|
// If we have completed working on all children of a function, remove the generic parameters from scope
|
||||||
|
if (!functionScopes.empty() && queue.size() == functionScopes.back().oldQueueSize && state->errors.empty())
|
||||||
|
{
|
||||||
|
closeFunctionScope(functionScopes.back().function);
|
||||||
|
functionScopes.pop_back();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -550,6 +604,21 @@ private:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void closeFunctionScope(TypeFunctionFunctionType* f)
|
||||||
|
{
|
||||||
|
if (!f->generics.empty())
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(genericTypes.size() >= f->generics.size());
|
||||||
|
genericTypes.erase(genericTypes.begin() + int(genericTypes.size() - f->generics.size()), genericTypes.end());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!f->genericPacks.empty())
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(genericPacks.size() >= f->genericPacks.size());
|
||||||
|
genericPacks.erase(genericPacks.begin() + int(genericPacks.size() - f->genericPacks.size()), genericPacks.end());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
TypeId shallowDeserialize(TypeFunctionTypeId ty)
|
TypeId shallowDeserialize(TypeFunctionTypeId ty)
|
||||||
{
|
{
|
||||||
if (auto it = find(ty))
|
if (auto it = find(ty))
|
||||||
|
@ -574,16 +643,10 @@ private:
|
||||||
target = state->ctx->builtins->stringType;
|
target = state->ctx->builtins->stringType;
|
||||||
break;
|
break;
|
||||||
case TypeFunctionPrimitiveType::Type::Thread:
|
case TypeFunctionPrimitiveType::Type::Thread:
|
||||||
if (FFlag::LuauUserTypeFunThreadBuffer)
|
|
||||||
target = state->ctx->builtins->threadType;
|
target = state->ctx->builtins->threadType;
|
||||||
else
|
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
|
||||||
break;
|
break;
|
||||||
case TypeFunctionPrimitiveType::Type::Buffer:
|
case TypeFunctionPrimitiveType::Type::Buffer:
|
||||||
if (FFlag::LuauUserTypeFunThreadBuffer)
|
|
||||||
target = state->ctx->builtins->bufferType;
|
target = state->ctx->builtins->bufferType;
|
||||||
else
|
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
|
@ -629,6 +692,33 @@ private:
|
||||||
else
|
else
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious class type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious class type is being deserialized");
|
||||||
}
|
}
|
||||||
|
else if (auto g = get<TypeFunctionGenericType>(ty))
|
||||||
|
{
|
||||||
|
if (g->isPack)
|
||||||
|
{
|
||||||
|
state->errors.push_back(format("Generic type pack '%s...' cannot be placed in a type position", g->name.c_str()));
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
auto it = std::find_if(
|
||||||
|
genericTypes.rbegin(),
|
||||||
|
genericTypes.rend(),
|
||||||
|
[&](const SerializedGeneric<TypeId>& el)
|
||||||
|
{
|
||||||
|
return g->isNamed == el.isNamed && g->name == el.name;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (it == genericTypes.rend())
|
||||||
|
{
|
||||||
|
state->errors.push_back(format("Generic type '%s' is not in a scope of the active generic function", g->name.c_str()));
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
target = it->type;
|
||||||
|
}
|
||||||
|
}
|
||||||
else
|
else
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
|
|
||||||
|
@ -645,11 +735,36 @@ private:
|
||||||
// Create a shallow deserialization
|
// Create a shallow deserialization
|
||||||
TypePackId target = {};
|
TypePackId target = {};
|
||||||
if (auto tPack = get<TypeFunctionTypePack>(tp))
|
if (auto tPack = get<TypeFunctionTypePack>(tp))
|
||||||
|
{
|
||||||
target = state->ctx->arena->addTypePack(TypePack{});
|
target = state->ctx->arena->addTypePack(TypePack{});
|
||||||
|
}
|
||||||
else if (auto vPack = get<TypeFunctionVariadicTypePack>(tp))
|
else if (auto vPack = get<TypeFunctionVariadicTypePack>(tp))
|
||||||
|
{
|
||||||
target = state->ctx->arena->addTypePack(VariadicTypePack{});
|
target = state->ctx->arena->addTypePack(VariadicTypePack{});
|
||||||
|
}
|
||||||
|
else if (auto gPack = get<TypeFunctionGenericTypePack>(tp))
|
||||||
|
{
|
||||||
|
auto it = std::find_if(
|
||||||
|
genericPacks.rbegin(),
|
||||||
|
genericPacks.rend(),
|
||||||
|
[&](const SerializedGeneric<TypePackId>& el)
|
||||||
|
{
|
||||||
|
return gPack->isNamed == el.isNamed && gPack->name == el.name;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (it == genericPacks.rend())
|
||||||
|
{
|
||||||
|
state->errors.push_back(format("Generic type pack '%s...' is not in a scope of the active generic function", gPack->name.c_str()));
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
target = it->type;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
|
}
|
||||||
|
|
||||||
packs[tp] = target;
|
packs[tp] = target;
|
||||||
queue.emplace_back(tp, target);
|
queue.emplace_back(tp, target);
|
||||||
|
@ -684,6 +799,8 @@ private:
|
||||||
deserializeChildren(f2, f1);
|
deserializeChildren(f2, f1);
|
||||||
else if (auto [c1, c2] = std::tuple{getMutable<ClassType>(ty), getMutable<TypeFunctionClassType>(tfti)}; c1 && c2)
|
else if (auto [c1, c2] = std::tuple{getMutable<ClassType>(ty), getMutable<TypeFunctionClassType>(tfti)}; c1 && c2)
|
||||||
deserializeChildren(c2, c1);
|
deserializeChildren(c2, c1);
|
||||||
|
else if (auto [g1, g2] = std::tuple{getMutable<GenericType>(ty), getMutable<TypeFunctionGenericType>(tfti)}; g1 && g2)
|
||||||
|
deserializeChildren(g2, g1);
|
||||||
else
|
else
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
}
|
}
|
||||||
|
@ -695,6 +812,8 @@ private:
|
||||||
else if (auto [vPack1, vPack2] = std::tuple{getMutable<VariadicTypePack>(tp), getMutable<TypeFunctionVariadicTypePack>(tftp)};
|
else if (auto [vPack1, vPack2] = std::tuple{getMutable<VariadicTypePack>(tp), getMutable<TypeFunctionVariadicTypePack>(tftp)};
|
||||||
vPack1 && vPack2)
|
vPack1 && vPack2)
|
||||||
deserializeChildren(vPack2, vPack1);
|
deserializeChildren(vPack2, vPack1);
|
||||||
|
else if (auto [gPack1, gPack2] = std::tuple{getMutable<GenericTypePack>(tp), getMutable<TypeFunctionGenericTypePack>(tftp)}; gPack1 && gPack2)
|
||||||
|
deserializeChildren(gPack2, gPack1);
|
||||||
else
|
else
|
||||||
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
state->ctx->ice->ice("Deserializing user defined type function arguments: mysterious type is being deserialized");
|
||||||
}
|
}
|
||||||
|
@ -778,6 +897,60 @@ private:
|
||||||
|
|
||||||
void deserializeChildren(TypeFunctionFunctionType* f2, FunctionType* f1)
|
void deserializeChildren(TypeFunctionFunctionType* f2, FunctionType* f1)
|
||||||
{
|
{
|
||||||
|
functionScopes.push_back({queue.size(), f2});
|
||||||
|
|
||||||
|
std::set<std::pair<bool, std::string>> genericNames;
|
||||||
|
|
||||||
|
// Introduce generic function parameters into scope
|
||||||
|
for (auto ty : f2->generics)
|
||||||
|
{
|
||||||
|
auto gty = get<TypeFunctionGenericType>(ty);
|
||||||
|
LUAU_ASSERT(gty && !gty->isPack);
|
||||||
|
|
||||||
|
std::pair<bool, std::string> nameKey = std::make_pair(gty->isNamed, gty->name);
|
||||||
|
|
||||||
|
// Duplicates are not allowed
|
||||||
|
if (genericNames.find(nameKey) != genericNames.end())
|
||||||
|
{
|
||||||
|
state->errors.push_back(format("Duplicate type parameter '%s'", gty->name.c_str()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
genericNames.insert(nameKey);
|
||||||
|
|
||||||
|
TypeId mapping = state->ctx->arena->addTV(Type(gty->isNamed ? GenericType{state->ctx->scope.get(), gty->name} : GenericType{}));
|
||||||
|
genericTypes.push_back({gty->isNamed, gty->name, mapping});
|
||||||
|
}
|
||||||
|
|
||||||
|
for (auto tp : f2->genericPacks)
|
||||||
|
{
|
||||||
|
auto gtp = get<TypeFunctionGenericTypePack>(tp);
|
||||||
|
LUAU_ASSERT(gtp);
|
||||||
|
|
||||||
|
std::pair<bool, std::string> nameKey = std::make_pair(gtp->isNamed, gtp->name);
|
||||||
|
|
||||||
|
// Duplicates are not allowed
|
||||||
|
if (genericNames.find(nameKey) != genericNames.end())
|
||||||
|
{
|
||||||
|
state->errors.push_back(format("Duplicate type parameter '%s'", gtp->name.c_str()));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
genericNames.insert(nameKey);
|
||||||
|
|
||||||
|
TypePackId mapping =
|
||||||
|
state->ctx->arena->addTypePack(TypePackVar(gtp->isNamed ? GenericTypePack{state->ctx->scope.get(), gtp->name} : GenericTypePack{}));
|
||||||
|
genericPacks.push_back({gtp->isNamed, gtp->name, mapping});
|
||||||
|
}
|
||||||
|
|
||||||
|
f1->generics.reserve(f2->generics.size());
|
||||||
|
for (auto ty : f2->generics)
|
||||||
|
f1->generics.push_back(shallowDeserialize(ty));
|
||||||
|
|
||||||
|
f1->genericPacks.reserve(f2->genericPacks.size());
|
||||||
|
for (auto tp : f2->genericPacks)
|
||||||
|
f1->genericPacks.push_back(shallowDeserialize(tp));
|
||||||
|
|
||||||
if (f2->argTypes)
|
if (f2->argTypes)
|
||||||
f1->argTypes = shallowDeserialize(f2->argTypes);
|
f1->argTypes = shallowDeserialize(f2->argTypes);
|
||||||
|
|
||||||
|
@ -790,6 +963,11 @@ private:
|
||||||
// noop.
|
// noop.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void deserializeChildren(TypeFunctionGenericType* g2, GenericType* g1)
|
||||||
|
{
|
||||||
|
// noop.
|
||||||
|
}
|
||||||
|
|
||||||
void deserializeChildren(TypeFunctionTypePack* t2, TypePack* t1)
|
void deserializeChildren(TypeFunctionTypePack* t2, TypePack* t1)
|
||||||
{
|
{
|
||||||
for (TypeFunctionTypeId& ty : t2->head)
|
for (TypeFunctionTypeId& ty : t2->head)
|
||||||
|
@ -803,6 +981,11 @@ private:
|
||||||
{
|
{
|
||||||
v1->ty = shallowDeserialize(v2->type);
|
v1->ty = shallowDeserialize(v2->type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void deserializeChildren(TypeFunctionGenericTypePack* v2, GenericTypePack* v1)
|
||||||
|
{
|
||||||
|
// noop.
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
TypeFunctionTypeId serialize(TypeId ty, TypeFunctionRuntimeBuilderState* state)
|
TypeFunctionTypeId serialize(TypeId ty, TypeFunctionRuntimeBuilderState* state)
|
||||||
|
|
|
@ -33,6 +33,8 @@ LUAU_FASTFLAG(LuauKnowsTheDataModel3)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeDuringUnification)
|
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeDuringUnification)
|
||||||
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauOldSolverCreatesChildScopePointers)
|
LUAU_FASTFLAGVARIABLE(LuauOldSolverCreatesChildScopePointers)
|
||||||
|
LUAU_FASTFLAG(LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -761,8 +763,12 @@ ControlFlow TypeChecker::check(const ScopePtr& scope, const AstStatRepeat& state
|
||||||
|
|
||||||
struct Demoter : Substitution
|
struct Demoter : Substitution
|
||||||
{
|
{
|
||||||
Demoter(TypeArena* arena)
|
TypeArena* arena = nullptr;
|
||||||
|
NotNull<BuiltinTypes> builtins;
|
||||||
|
Demoter(TypeArena* arena, NotNull<BuiltinTypes> builtins)
|
||||||
: Substitution(TxnLog::empty(), arena)
|
: Substitution(TxnLog::empty(), arena)
|
||||||
|
, arena(arena)
|
||||||
|
, builtins(builtins)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -788,7 +794,8 @@ struct Demoter : Substitution
|
||||||
{
|
{
|
||||||
auto ftv = get<FreeType>(ty);
|
auto ftv = get<FreeType>(ty);
|
||||||
LUAU_ASSERT(ftv);
|
LUAU_ASSERT(ftv);
|
||||||
return addType(FreeType{demotedLevel(ftv->level)});
|
return FFlag::LuauFreeTypesMustHaveBounds ? arena->freshType(builtins, demotedLevel(ftv->level))
|
||||||
|
: addType(FreeType{demotedLevel(ftv->level)});
|
||||||
}
|
}
|
||||||
|
|
||||||
TypePackId clean(TypePackId tp) override
|
TypePackId clean(TypePackId tp) override
|
||||||
|
@ -835,7 +842,7 @@ ControlFlow TypeChecker::check(const ScopePtr& scope, const AstStatReturn& retur
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Demoter demoter{¤tModule->internalTypes};
|
Demoter demoter{¤tModule->internalTypes, builtinTypes};
|
||||||
demoter.demote(expectedTypes);
|
demoter.demote(expectedTypes);
|
||||||
|
|
||||||
TypePackId retPack = checkExprList(scope, return_.location, return_.list, false, {}, expectedTypes).type;
|
TypePackId retPack = checkExprList(scope, return_.location, return_.list, false, {}, expectedTypes).type;
|
||||||
|
@ -4408,7 +4415,7 @@ std::vector<std::optional<TypeId>> TypeChecker::getExpectedTypesForCall(const st
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Demoter demoter{¤tModule->internalTypes};
|
Demoter demoter{¤tModule->internalTypes, builtinTypes};
|
||||||
demoter.demote(expectedTypes);
|
demoter.demote(expectedTypes);
|
||||||
|
|
||||||
return expectedTypes;
|
return expectedTypes;
|
||||||
|
@ -4506,10 +4513,10 @@ std::unique_ptr<WithPredicate<TypePackId>> TypeChecker::checkCallOverload(
|
||||||
|
|
||||||
// When this function type has magic functions and did return something, we select that overload instead.
|
// When this function type has magic functions and did return something, we select that overload instead.
|
||||||
// TODO: pass in a Unifier object to the magic functions? This will allow the magic functions to cooperate with overload resolution.
|
// TODO: pass in a Unifier object to the magic functions? This will allow the magic functions to cooperate with overload resolution.
|
||||||
if (ftv->magicFunction)
|
if (ftv->magic)
|
||||||
{
|
{
|
||||||
// TODO: We're passing in the wrong TypePackId. Should be argPack, but a unit test fails otherwise. CLI-40458
|
// TODO: We're passing in the wrong TypePackId. Should be argPack, but a unit test fails otherwise. CLI-40458
|
||||||
if (std::optional<WithPredicate<TypePackId>> ret = ftv->magicFunction(*this, scope, expr, argListResult))
|
if (std::optional<WithPredicate<TypePackId>> ret = ftv->magic->handleOldSolver(*this, scope, expr, argListResult))
|
||||||
return std::make_unique<WithPredicate<TypePackId>>(std::move(*ret));
|
return std::make_unique<WithPredicate<TypePackId>>(std::move(*ret));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5273,7 +5280,8 @@ TypeId TypeChecker::freshType(const ScopePtr& scope)
|
||||||
|
|
||||||
TypeId TypeChecker::freshType(TypeLevel level)
|
TypeId TypeChecker::freshType(TypeLevel level)
|
||||||
{
|
{
|
||||||
return currentModule->internalTypes.addType(Type(FreeType(level)));
|
return FFlag::LuauFreeTypesMustHaveBounds ? currentModule->internalTypes.freshType(builtinTypes, level)
|
||||||
|
: currentModule->internalTypes.addType(Type(FreeType(level)));
|
||||||
}
|
}
|
||||||
|
|
||||||
TypeId TypeChecker::singletonType(bool value)
|
TypeId TypeChecker::singletonType(bool value)
|
||||||
|
@ -5718,6 +5726,12 @@ TypeId TypeChecker::resolveTypeWorker(const ScopePtr& scope, const AstType& anno
|
||||||
}
|
}
|
||||||
else if (const auto& un = annotation.as<AstTypeUnion>())
|
else if (const auto& un = annotation.as<AstTypeUnion>())
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
|
{
|
||||||
|
if (un->types.size == 1)
|
||||||
|
return resolveType(scope, *un->types.data[0]);
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<TypeId> types;
|
std::vector<TypeId> types;
|
||||||
for (AstType* ann : un->types)
|
for (AstType* ann : un->types)
|
||||||
types.push_back(resolveType(scope, *ann));
|
types.push_back(resolveType(scope, *ann));
|
||||||
|
@ -5726,12 +5740,22 @@ TypeId TypeChecker::resolveTypeWorker(const ScopePtr& scope, const AstType& anno
|
||||||
}
|
}
|
||||||
else if (const auto& un = annotation.as<AstTypeIntersection>())
|
else if (const auto& un = annotation.as<AstTypeIntersection>())
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauPreserveUnionIntersectionNodeForLeadingTokenSingleType)
|
||||||
|
{
|
||||||
|
if (un->types.size == 1)
|
||||||
|
return resolveType(scope, *un->types.data[0]);
|
||||||
|
}
|
||||||
|
|
||||||
std::vector<TypeId> types;
|
std::vector<TypeId> types;
|
||||||
for (AstType* ann : un->types)
|
for (AstType* ann : un->types)
|
||||||
types.push_back(resolveType(scope, *ann));
|
types.push_back(resolveType(scope, *ann));
|
||||||
|
|
||||||
return addType(IntersectionType{types});
|
return addType(IntersectionType{types});
|
||||||
}
|
}
|
||||||
|
else if (const auto& g = annotation.as<AstTypeGroup>())
|
||||||
|
{
|
||||||
|
return resolveType(scope, *g->type);
|
||||||
|
}
|
||||||
else if (const auto& tsb = annotation.as<AstTypeSingletonBool>())
|
else if (const auto& tsb = annotation.as<AstTypeSingletonBool>())
|
||||||
{
|
{
|
||||||
return singletonType(tsb->value);
|
return singletonType(tsb->value);
|
||||||
|
@ -5889,8 +5913,8 @@ GenericTypeDefinitions TypeChecker::createGenericTypes(
|
||||||
const ScopePtr& scope,
|
const ScopePtr& scope,
|
||||||
std::optional<TypeLevel> levelOpt,
|
std::optional<TypeLevel> levelOpt,
|
||||||
const AstNode& node,
|
const AstNode& node,
|
||||||
const AstArray<AstGenericType>& genericNames,
|
const AstArray<AstGenericType*>& genericNames,
|
||||||
const AstArray<AstGenericTypePack>& genericPackNames,
|
const AstArray<AstGenericTypePack*>& genericPackNames,
|
||||||
bool useCache
|
bool useCache
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
|
@ -5900,14 +5924,14 @@ GenericTypeDefinitions TypeChecker::createGenericTypes(
|
||||||
|
|
||||||
std::vector<GenericTypeDefinition> generics;
|
std::vector<GenericTypeDefinition> generics;
|
||||||
|
|
||||||
for (const AstGenericType& generic : genericNames)
|
for (const AstGenericType* generic : genericNames)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> defaultValue;
|
std::optional<TypeId> defaultValue;
|
||||||
|
|
||||||
if (generic.defaultValue)
|
if (generic->defaultValue)
|
||||||
defaultValue = resolveType(scope, *generic.defaultValue);
|
defaultValue = resolveType(scope, *generic->defaultValue);
|
||||||
|
|
||||||
Name n = generic.name.value;
|
Name n = generic->name.value;
|
||||||
|
|
||||||
// These generics are the only thing that will ever be added to scope, so we can be certain that
|
// These generics are the only thing that will ever be added to scope, so we can be certain that
|
||||||
// a collision can only occur when two generic types have the same name.
|
// a collision can only occur when two generic types have the same name.
|
||||||
|
@ -5936,14 +5960,14 @@ GenericTypeDefinitions TypeChecker::createGenericTypes(
|
||||||
|
|
||||||
std::vector<GenericTypePackDefinition> genericPacks;
|
std::vector<GenericTypePackDefinition> genericPacks;
|
||||||
|
|
||||||
for (const AstGenericTypePack& genericPack : genericPackNames)
|
for (const AstGenericTypePack* genericPack : genericPackNames)
|
||||||
{
|
{
|
||||||
std::optional<TypePackId> defaultValue;
|
std::optional<TypePackId> defaultValue;
|
||||||
|
|
||||||
if (genericPack.defaultValue)
|
if (genericPack->defaultValue)
|
||||||
defaultValue = resolveTypePack(scope, *genericPack.defaultValue);
|
defaultValue = resolveTypePack(scope, *genericPack->defaultValue);
|
||||||
|
|
||||||
Name n = genericPack.name.value;
|
Name n = genericPack->name.value;
|
||||||
|
|
||||||
// These generics are the only thing that will ever be added to scope, so we can be certain that
|
// These generics are the only thing that will ever be added to scope, so we can be certain that
|
||||||
// a collision can only occur when two generic types have the same name.
|
// a collision can only occur when two generic types have the same name.
|
||||||
|
|
|
@ -5,12 +5,15 @@
|
||||||
#include "Luau/Normalize.h"
|
#include "Luau/Normalize.h"
|
||||||
#include "Luau/Scope.h"
|
#include "Luau/Scope.h"
|
||||||
#include "Luau/ToString.h"
|
#include "Luau/ToString.h"
|
||||||
|
#include "Luau/Type.h"
|
||||||
#include "Luau/TypeInfer.h"
|
#include "Luau/TypeInfer.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauSolverV2);
|
LUAU_FASTFLAG(LuauSolverV2);
|
||||||
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete);
|
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete);
|
||||||
|
LUAU_FASTFLAG(LuauTrackInteriorFreeTypesOnScope);
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -318,9 +321,11 @@ TypePack extendTypePack(
|
||||||
{
|
{
|
||||||
FreeType ft{ftp->scope, builtinTypes->neverType, builtinTypes->unknownType};
|
FreeType ft{ftp->scope, builtinTypes->neverType, builtinTypes->unknownType};
|
||||||
t = arena.addType(ft);
|
t = arena.addType(ft);
|
||||||
|
if (FFlag::LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
trackInteriorFreeType(ftp->scope, t);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
t = arena.freshType(ftp->scope);
|
t = FFlag::LuauFreeTypesMustHaveBounds ? arena.freshType(builtinTypes, ftp->scope) : arena.freshType_DEPRECATED(ftp->scope);
|
||||||
}
|
}
|
||||||
|
|
||||||
newPack.head.push_back(t);
|
newPack.head.push_back(t);
|
||||||
|
@ -533,7 +538,7 @@ std::vector<TypeId> findBlockedArgTypesIn(AstExprCall* expr, NotNull<DenseHashMa
|
||||||
{
|
{
|
||||||
std::vector<TypeId> toBlock;
|
std::vector<TypeId> toBlock;
|
||||||
BlockedTypeInLiteralVisitor v{astTypes, NotNull{&toBlock}};
|
BlockedTypeInLiteralVisitor v{astTypes, NotNull{&toBlock}};
|
||||||
for (auto arg: expr->args)
|
for (auto arg : expr->args)
|
||||||
{
|
{
|
||||||
if (isLiteral(arg) || arg->is<AstExprGroup>())
|
if (isLiteral(arg) || arg->is<AstExprGroup>())
|
||||||
{
|
{
|
||||||
|
@ -543,5 +548,21 @@ std::vector<TypeId> findBlockedArgTypesIn(AstExprCall* expr, NotNull<DenseHashMa
|
||||||
return toBlock;
|
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
|
} // namespace Luau
|
||||||
|
|
|
@ -22,6 +22,7 @@ LUAU_FASTFLAGVARIABLE(LuauTransitiveSubtyping)
|
||||||
LUAU_FASTFLAG(LuauSolverV2)
|
LUAU_FASTFLAG(LuauSolverV2)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauFixIndexerSubtypingOrdering)
|
LUAU_FASTFLAGVARIABLE(LuauFixIndexerSubtypingOrdering)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUnifierRecursionOnRestart)
|
LUAU_FASTFLAGVARIABLE(LuauUnifierRecursionOnRestart)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -1648,7 +1649,7 @@ void Unifier::tryUnify_(TypePackId subTp, TypePackId superTp, bool isFunctionCal
|
||||||
if (FFlag::LuauSolverV2)
|
if (FFlag::LuauSolverV2)
|
||||||
return freshType(NotNull{types}, builtinTypes, scope);
|
return freshType(NotNull{types}, builtinTypes, scope);
|
||||||
else
|
else
|
||||||
return types->freshType(scope, level);
|
return FFlag::LuauFreeTypesMustHaveBounds ? types->freshType(builtinTypes, scope, level) : types->freshType_DEPRECATED(scope, level);
|
||||||
};
|
};
|
||||||
|
|
||||||
const TypePackId emptyTp = types->addTypePack(TypePack{{}, std::nullopt});
|
const TypePackId emptyTp = types->addTypePack(TypePack{{}, std::nullopt});
|
||||||
|
|
|
@ -45,4 +45,4 @@ private:
|
||||||
size_t offset;
|
size_t offset;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
} // namespace Luau
|
||||||
|
|
|
@ -120,20 +120,6 @@ struct AstTypeList
|
||||||
|
|
||||||
using AstArgumentName = std::pair<AstName, Location>; // TODO: remove and replace when we get a common struct for this pair instead of AstName
|
using AstArgumentName = std::pair<AstName, Location>; // TODO: remove and replace when we get a common struct for this pair instead of AstName
|
||||||
|
|
||||||
struct AstGenericType
|
|
||||||
{
|
|
||||||
AstName name;
|
|
||||||
Location location;
|
|
||||||
AstType* defaultValue = nullptr;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct AstGenericTypePack
|
|
||||||
{
|
|
||||||
AstName name;
|
|
||||||
Location location;
|
|
||||||
AstTypePack* defaultValue = nullptr;
|
|
||||||
};
|
|
||||||
|
|
||||||
extern int gAstRttiIndex;
|
extern int gAstRttiIndex;
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
|
@ -253,6 +239,32 @@ public:
|
||||||
bool hasSemicolon;
|
bool hasSemicolon;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class AstGenericType : public AstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_RTTI(AstGenericType)
|
||||||
|
|
||||||
|
explicit AstGenericType(const Location& location, AstName name, AstType* defaultValue = nullptr);
|
||||||
|
|
||||||
|
void visit(AstVisitor* visitor) override;
|
||||||
|
|
||||||
|
AstName name;
|
||||||
|
AstType* defaultValue = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
class AstGenericTypePack : public AstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_RTTI(AstGenericTypePack)
|
||||||
|
|
||||||
|
explicit AstGenericTypePack(const Location& location, AstName name, AstTypePack* defaultValue = nullptr);
|
||||||
|
|
||||||
|
void visit(AstVisitor* visitor) override;
|
||||||
|
|
||||||
|
AstName name;
|
||||||
|
AstTypePack* defaultValue = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
class AstExprGroup : public AstExpr
|
class AstExprGroup : public AstExpr
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
@ -424,8 +436,8 @@ public:
|
||||||
AstExprFunction(
|
AstExprFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
AstLocal* self,
|
AstLocal* self,
|
||||||
const AstArray<AstLocal*>& args,
|
const AstArray<AstLocal*>& args,
|
||||||
bool vararg,
|
bool vararg,
|
||||||
|
@ -443,8 +455,8 @@ public:
|
||||||
bool hasNativeAttribute() const;
|
bool hasNativeAttribute() const;
|
||||||
|
|
||||||
AstArray<AstAttr*> attributes;
|
AstArray<AstAttr*> attributes;
|
||||||
AstArray<AstGenericType> generics;
|
AstArray<AstGenericType*> generics;
|
||||||
AstArray<AstGenericTypePack> genericPacks;
|
AstArray<AstGenericTypePack*> genericPacks;
|
||||||
AstLocal* self;
|
AstLocal* self;
|
||||||
AstArray<AstLocal*> args;
|
AstArray<AstLocal*> args;
|
||||||
std::optional<AstTypeList> returnAnnotation;
|
std::optional<AstTypeList> returnAnnotation;
|
||||||
|
@ -857,8 +869,8 @@ public:
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstName& name,
|
const AstName& name,
|
||||||
const Location& nameLocation,
|
const Location& nameLocation,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
AstType* type,
|
AstType* type,
|
||||||
bool exported
|
bool exported
|
||||||
);
|
);
|
||||||
|
@ -867,8 +879,8 @@ public:
|
||||||
|
|
||||||
AstName name;
|
AstName name;
|
||||||
Location nameLocation;
|
Location nameLocation;
|
||||||
AstArray<AstGenericType> generics;
|
AstArray<AstGenericType*> generics;
|
||||||
AstArray<AstGenericTypePack> genericPacks;
|
AstArray<AstGenericTypePack*> genericPacks;
|
||||||
AstType* type;
|
AstType* type;
|
||||||
bool exported;
|
bool exported;
|
||||||
};
|
};
|
||||||
|
@ -911,8 +923,8 @@ public:
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstName& name,
|
const AstName& name,
|
||||||
const Location& nameLocation,
|
const Location& nameLocation,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& params,
|
const AstTypeList& params,
|
||||||
const AstArray<AstArgumentName>& paramNames,
|
const AstArray<AstArgumentName>& paramNames,
|
||||||
bool vararg,
|
bool vararg,
|
||||||
|
@ -925,8 +937,8 @@ public:
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
const AstName& name,
|
const AstName& name,
|
||||||
const Location& nameLocation,
|
const Location& nameLocation,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& params,
|
const AstTypeList& params,
|
||||||
const AstArray<AstArgumentName>& paramNames,
|
const AstArray<AstArgumentName>& paramNames,
|
||||||
bool vararg,
|
bool vararg,
|
||||||
|
@ -942,8 +954,8 @@ public:
|
||||||
AstArray<AstAttr*> attributes;
|
AstArray<AstAttr*> attributes;
|
||||||
AstName name;
|
AstName name;
|
||||||
Location nameLocation;
|
Location nameLocation;
|
||||||
AstArray<AstGenericType> generics;
|
AstArray<AstGenericType*> generics;
|
||||||
AstArray<AstGenericTypePack> genericPacks;
|
AstArray<AstGenericTypePack*> genericPacks;
|
||||||
AstTypeList params;
|
AstTypeList params;
|
||||||
AstArray<AstArgumentName> paramNames;
|
AstArray<AstArgumentName> paramNames;
|
||||||
bool vararg = false;
|
bool vararg = false;
|
||||||
|
@ -1074,8 +1086,8 @@ public:
|
||||||
|
|
||||||
AstTypeFunction(
|
AstTypeFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& argTypes,
|
const AstTypeList& argTypes,
|
||||||
const AstArray<std::optional<AstArgumentName>>& argNames,
|
const AstArray<std::optional<AstArgumentName>>& argNames,
|
||||||
const AstTypeList& returnTypes
|
const AstTypeList& returnTypes
|
||||||
|
@ -1084,8 +1096,8 @@ public:
|
||||||
AstTypeFunction(
|
AstTypeFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& argTypes,
|
const AstTypeList& argTypes,
|
||||||
const AstArray<std::optional<AstArgumentName>>& argNames,
|
const AstArray<std::optional<AstArgumentName>>& argNames,
|
||||||
const AstTypeList& returnTypes
|
const AstTypeList& returnTypes
|
||||||
|
@ -1096,8 +1108,8 @@ public:
|
||||||
bool isCheckedFunction() const;
|
bool isCheckedFunction() const;
|
||||||
|
|
||||||
AstArray<AstAttr*> attributes;
|
AstArray<AstAttr*> attributes;
|
||||||
AstArray<AstGenericType> generics;
|
AstArray<AstGenericType*> generics;
|
||||||
AstArray<AstGenericTypePack> genericPacks;
|
AstArray<AstGenericTypePack*> genericPacks;
|
||||||
AstTypeList argTypes;
|
AstTypeList argTypes;
|
||||||
AstArray<std::optional<AstArgumentName>> argNames;
|
AstArray<std::optional<AstArgumentName>> argNames;
|
||||||
AstTypeList returnTypes;
|
AstTypeList returnTypes;
|
||||||
|
@ -1204,6 +1216,18 @@ public:
|
||||||
const AstArray<char> value;
|
const AstArray<char> value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class AstTypeGroup : public AstType
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_RTTI(AstTypeGroup)
|
||||||
|
|
||||||
|
explicit AstTypeGroup(const Location& location, AstType* type);
|
||||||
|
|
||||||
|
void visit(AstVisitor* visitor) override;
|
||||||
|
|
||||||
|
AstType* type;
|
||||||
|
};
|
||||||
|
|
||||||
class AstTypePack : public AstNode
|
class AstTypePack : public AstNode
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
@ -1264,6 +1288,16 @@ public:
|
||||||
return visit(static_cast<AstNode*>(node));
|
return visit(static_cast<AstNode*>(node));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual bool visit(class AstGenericType* node)
|
||||||
|
{
|
||||||
|
return visit(static_cast<AstNode*>(node));
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool visit(class AstGenericTypePack* node)
|
||||||
|
{
|
||||||
|
return visit(static_cast<AstNode*>(node));
|
||||||
|
}
|
||||||
|
|
||||||
virtual bool visit(class AstExpr* node)
|
virtual bool visit(class AstExpr* node)
|
||||||
{
|
{
|
||||||
return visit(static_cast<AstNode*>(node));
|
return visit(static_cast<AstNode*>(node));
|
||||||
|
@ -1470,6 +1504,10 @@ public:
|
||||||
{
|
{
|
||||||
return visit(static_cast<AstType*>(node));
|
return visit(static_cast<AstType*>(node));
|
||||||
}
|
}
|
||||||
|
virtual bool visit(class AstTypeGroup* node)
|
||||||
|
{
|
||||||
|
return visit(static_cast<AstType*>(node));
|
||||||
|
}
|
||||||
virtual bool visit(class AstTypeError* node)
|
virtual bool visit(class AstTypeError* node)
|
||||||
{
|
{
|
||||||
return visit(static_cast<AstType*>(node));
|
return visit(static_cast<AstType*>(node));
|
||||||
|
|
344
Ast/include/Luau/Cst.h
Normal file
344
Ast/include/Luau/Cst.h
Normal file
|
@ -0,0 +1,344 @@
|
||||||
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "Luau/Location.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace Luau
|
||||||
|
{
|
||||||
|
|
||||||
|
extern int gCstRttiIndex;
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
struct CstRtti
|
||||||
|
{
|
||||||
|
static const int value;
|
||||||
|
};
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
const int CstRtti<T>::value = ++gCstRttiIndex;
|
||||||
|
|
||||||
|
#define LUAU_CST_RTTI(Class) \
|
||||||
|
static int CstClassIndex() \
|
||||||
|
{ \
|
||||||
|
return CstRtti<Class>::value; \
|
||||||
|
}
|
||||||
|
|
||||||
|
class CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
explicit CstNode(int classIndex)
|
||||||
|
: classIndex(classIndex)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename T>
|
||||||
|
bool is() const
|
||||||
|
{
|
||||||
|
return classIndex == T::CstClassIndex();
|
||||||
|
}
|
||||||
|
template<typename T>
|
||||||
|
T* as()
|
||||||
|
{
|
||||||
|
return classIndex == T::CstClassIndex() ? static_cast<T*>(this) : nullptr;
|
||||||
|
}
|
||||||
|
template<typename T>
|
||||||
|
const T* as() const
|
||||||
|
{
|
||||||
|
return classIndex == T::CstClassIndex() ? static_cast<const T*>(this) : nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
const int classIndex;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprConstantNumber : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprConstantNumber)
|
||||||
|
|
||||||
|
explicit CstExprConstantNumber(const AstArray<char>& value);
|
||||||
|
|
||||||
|
AstArray<char> value;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprConstantString : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprConstantNumber)
|
||||||
|
|
||||||
|
enum QuoteStyle
|
||||||
|
{
|
||||||
|
QuotedSingle,
|
||||||
|
QuotedDouble,
|
||||||
|
QuotedRaw,
|
||||||
|
QuotedInterp,
|
||||||
|
};
|
||||||
|
|
||||||
|
CstExprConstantString(AstArray<char> sourceString, QuoteStyle quoteStyle, unsigned int blockDepth);
|
||||||
|
|
||||||
|
AstArray<char> sourceString;
|
||||||
|
QuoteStyle quoteStyle;
|
||||||
|
unsigned int blockDepth;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprCall : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprCall)
|
||||||
|
|
||||||
|
CstExprCall(std::optional<Position> openParens, std::optional<Position> closeParens, AstArray<Position> commaPositions);
|
||||||
|
|
||||||
|
std::optional<Position> openParens;
|
||||||
|
std::optional<Position> closeParens;
|
||||||
|
AstArray<Position> commaPositions;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprIndexExpr : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprIndexExpr)
|
||||||
|
|
||||||
|
CstExprIndexExpr(Position openBracketPosition, Position closeBracketPosition);
|
||||||
|
|
||||||
|
Position openBracketPosition;
|
||||||
|
Position closeBracketPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprTable : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprTable)
|
||||||
|
|
||||||
|
enum Separator
|
||||||
|
{
|
||||||
|
Comma,
|
||||||
|
Semicolon,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Item
|
||||||
|
{
|
||||||
|
std::optional<Position> indexerOpenPosition; // '[', only if Kind == General
|
||||||
|
std::optional<Position> indexerClosePosition; // ']', only if Kind == General
|
||||||
|
std::optional<Position> equalsPosition; // only if Kind != List
|
||||||
|
std::optional<Separator> separator; // may be missing for last Item
|
||||||
|
std::optional<Position> separatorPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
explicit CstExprTable(const AstArray<Item>& items);
|
||||||
|
|
||||||
|
AstArray<Item> items;
|
||||||
|
};
|
||||||
|
|
||||||
|
// TODO: Shared between unary and binary, should we split?
|
||||||
|
class CstExprOp : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprOp)
|
||||||
|
|
||||||
|
explicit CstExprOp(Position opPosition);
|
||||||
|
|
||||||
|
Position opPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprTypeAssertion : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprTypeAssertion)
|
||||||
|
|
||||||
|
explicit CstExprTypeAssertion(Position opPosition);
|
||||||
|
|
||||||
|
Position opPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprIfElse : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprIfElse)
|
||||||
|
|
||||||
|
CstExprIfElse(Position thenPosition, Position elsePosition, bool isElseIf);
|
||||||
|
|
||||||
|
Position thenPosition;
|
||||||
|
Position elsePosition;
|
||||||
|
bool isElseIf;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstExprInterpString : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstExprInterpString)
|
||||||
|
|
||||||
|
explicit CstExprInterpString(AstArray<AstArray<char>> sourceStrings, AstArray<Position> stringPositions);
|
||||||
|
|
||||||
|
AstArray<AstArray<char>> sourceStrings;
|
||||||
|
AstArray<Position> stringPositions;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatDo : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatDo)
|
||||||
|
|
||||||
|
explicit CstStatDo(Position endPosition);
|
||||||
|
|
||||||
|
Position endPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatRepeat : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatRepeat)
|
||||||
|
|
||||||
|
explicit CstStatRepeat(Position untilPosition);
|
||||||
|
|
||||||
|
Position untilPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatReturn : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatReturn)
|
||||||
|
|
||||||
|
explicit CstStatReturn(AstArray<Position> commaPositions);
|
||||||
|
|
||||||
|
AstArray<Position> commaPositions;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatLocal : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatLocal)
|
||||||
|
|
||||||
|
CstStatLocal(AstArray<Position> varsCommaPositions, AstArray<Position> valuesCommaPositions);
|
||||||
|
|
||||||
|
AstArray<Position> varsCommaPositions;
|
||||||
|
AstArray<Position> valuesCommaPositions;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatFor : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatFor)
|
||||||
|
|
||||||
|
CstStatFor(Position equalsPosition, Position endCommaPosition, std::optional<Position> stepCommaPosition);
|
||||||
|
|
||||||
|
Position equalsPosition;
|
||||||
|
Position endCommaPosition;
|
||||||
|
std::optional<Position> stepCommaPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatForIn : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatForIn)
|
||||||
|
|
||||||
|
CstStatForIn(AstArray<Position> varsCommaPositions, AstArray<Position> valuesCommaPositions);
|
||||||
|
|
||||||
|
AstArray<Position> varsCommaPositions;
|
||||||
|
AstArray<Position> valuesCommaPositions;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatAssign : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatAssign)
|
||||||
|
|
||||||
|
CstStatAssign(AstArray<Position> varsCommaPositions, Position equalsPosition, AstArray<Position> valuesCommaPositions);
|
||||||
|
|
||||||
|
AstArray<Position> varsCommaPositions;
|
||||||
|
Position equalsPosition;
|
||||||
|
AstArray<Position> valuesCommaPositions;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatCompoundAssign : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatCompoundAssign)
|
||||||
|
|
||||||
|
explicit CstStatCompoundAssign(Position opPosition);
|
||||||
|
|
||||||
|
Position opPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstStatLocalFunction : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstStatLocalFunction)
|
||||||
|
|
||||||
|
explicit CstStatLocalFunction(Position functionKeywordPosition);
|
||||||
|
|
||||||
|
Position functionKeywordPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstTypeReference : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstTypeReference)
|
||||||
|
|
||||||
|
CstTypeReference(
|
||||||
|
std::optional<Position> prefixPointPosition,
|
||||||
|
Position openParametersPosition,
|
||||||
|
AstArray<Position> parametersCommaPositions,
|
||||||
|
Position closeParametersPosition
|
||||||
|
);
|
||||||
|
|
||||||
|
std::optional<Position> prefixPointPosition;
|
||||||
|
Position openParametersPosition;
|
||||||
|
AstArray<Position> parametersCommaPositions;
|
||||||
|
Position closeParametersPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstTypeTable : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstTypeTable)
|
||||||
|
|
||||||
|
struct Item
|
||||||
|
{
|
||||||
|
enum struct Kind
|
||||||
|
{
|
||||||
|
Indexer,
|
||||||
|
Property,
|
||||||
|
StringProperty,
|
||||||
|
};
|
||||||
|
|
||||||
|
Kind kind;
|
||||||
|
Position indexerOpenPosition; // '[', only if Kind != Property
|
||||||
|
Position indexerClosePosition; // ']' only if Kind != Property
|
||||||
|
Position colonPosition;
|
||||||
|
std::optional<CstExprTable::Separator> separator; // may be missing for last Item
|
||||||
|
std::optional<Position> separatorPosition;
|
||||||
|
|
||||||
|
CstExprConstantString* stringInfo = nullptr; // only if Kind == StringProperty
|
||||||
|
};
|
||||||
|
|
||||||
|
CstTypeTable(AstArray<Item> items, bool isArray);
|
||||||
|
|
||||||
|
AstArray<Item> items;
|
||||||
|
bool isArray = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstTypeTypeof : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstTypeTypeof)
|
||||||
|
|
||||||
|
CstTypeTypeof(Position openPosition, Position closePosition);
|
||||||
|
|
||||||
|
Position openPosition;
|
||||||
|
Position closePosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
class CstTypeSingletonString : public CstNode
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LUAU_CST_RTTI(CstTypeSingletonString)
|
||||||
|
|
||||||
|
CstTypeSingletonString(AstArray<char> sourceString, CstExprConstantString::QuoteStyle quoteStyle, unsigned int blockDepth);
|
||||||
|
|
||||||
|
AstArray<char> sourceString;
|
||||||
|
CstExprConstantString::QuoteStyle quoteStyle;
|
||||||
|
unsigned int blockDepth;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Luau
|
|
@ -87,6 +87,12 @@ struct Lexeme
|
||||||
Reserved_END
|
Reserved_END
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum struct QuoteStyle
|
||||||
|
{
|
||||||
|
Single,
|
||||||
|
Double,
|
||||||
|
};
|
||||||
|
|
||||||
Type type;
|
Type type;
|
||||||
Location location;
|
Location location;
|
||||||
|
|
||||||
|
@ -111,6 +117,8 @@ public:
|
||||||
Lexeme(const Location& location, Type type, const char* name);
|
Lexeme(const Location& location, Type type, const char* name);
|
||||||
|
|
||||||
unsigned int getLength() const;
|
unsigned int getLength() const;
|
||||||
|
unsigned int getBlockDepth() const;
|
||||||
|
QuoteStyle getQuoteStyle() const;
|
||||||
|
|
||||||
std::string toString() const;
|
std::string toString() const;
|
||||||
};
|
};
|
||||||
|
|
|
@ -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;
|
return this->column == rhs.column && this->line == rhs.line;
|
||||||
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 == 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);
|
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 encloses(const Location& l) const;
|
||||||
bool overlaps(const Location& l) const;
|
bool overlaps(const Location& l) const;
|
||||||
|
|
|
@ -29,6 +29,8 @@ struct ParseOptions
|
||||||
bool allowDeclarationSyntax = false;
|
bool allowDeclarationSyntax = false;
|
||||||
bool captureComments = false;
|
bool captureComments = false;
|
||||||
std::optional<FragmentParseResumeSettings> parseFragment = std::nullopt;
|
std::optional<FragmentParseResumeSettings> parseFragment = std::nullopt;
|
||||||
|
bool storeCstData = false;
|
||||||
|
bool noErrorLimit = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Luau
|
} // namespace Luau
|
||||||
|
|
|
@ -10,6 +10,7 @@ namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
class AstStatBlock;
|
class AstStatBlock;
|
||||||
|
class CstNode;
|
||||||
|
|
||||||
class ParseError : public std::exception
|
class ParseError : public std::exception
|
||||||
{
|
{
|
||||||
|
@ -55,6 +56,8 @@ struct Comment
|
||||||
Location location;
|
Location location;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
using CstNodeMap = DenseHashMap<AstNode*, CstNode*>;
|
||||||
|
|
||||||
struct ParseResult
|
struct ParseResult
|
||||||
{
|
{
|
||||||
AstStatBlock* root;
|
AstStatBlock* root;
|
||||||
|
@ -64,6 +67,8 @@ struct ParseResult
|
||||||
std::vector<ParseError> errors;
|
std::vector<ParseError> errors;
|
||||||
|
|
||||||
std::vector<Comment> commentLocations;
|
std::vector<Comment> commentLocations;
|
||||||
|
|
||||||
|
CstNodeMap cstNodeMap{nullptr};
|
||||||
};
|
};
|
||||||
|
|
||||||
static constexpr const char* kParseNameError = "%error-id%";
|
static constexpr const char* kParseNameError = "%error-id%";
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "Luau/StringUtils.h"
|
#include "Luau/StringUtils.h"
|
||||||
#include "Luau/DenseHash.h"
|
#include "Luau/DenseHash.h"
|
||||||
#include "Luau/Common.h"
|
#include "Luau/Common.h"
|
||||||
|
#include "Luau/Cst.h"
|
||||||
|
|
||||||
#include <initializer_list>
|
#include <initializer_list>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
@ -116,7 +117,7 @@ private:
|
||||||
AstStat* parseFor();
|
AstStat* parseFor();
|
||||||
|
|
||||||
// funcname ::= Name {`.' Name} [`:' Name]
|
// funcname ::= Name {`.' Name} [`:' Name]
|
||||||
AstExpr* parseFunctionName(Location start, bool& hasself, AstName& debugname);
|
AstExpr* parseFunctionName(Location start_DEPRECATED, bool& hasself, AstName& debugname);
|
||||||
|
|
||||||
// function funcname funcbody
|
// function funcname funcbody
|
||||||
LUAU_FORCEINLINE AstStat* parseFunctionStat(const AstArray<AstAttr*>& attributes = {nullptr, 0});
|
LUAU_FORCEINLINE AstStat* parseFunctionStat(const AstArray<AstAttr*>& attributes = {nullptr, 0});
|
||||||
|
@ -173,14 +174,18 @@ private:
|
||||||
);
|
);
|
||||||
|
|
||||||
// explist ::= {exp `,'} exp
|
// explist ::= {exp `,'} exp
|
||||||
void parseExprList(TempVector<AstExpr*>& result);
|
void parseExprList(TempVector<AstExpr*>& result, TempVector<Position>* commaPositions = nullptr);
|
||||||
|
|
||||||
// binding ::= Name [`:` Type]
|
// binding ::= Name [`:` Type]
|
||||||
Binding parseBinding();
|
Binding parseBinding();
|
||||||
|
|
||||||
// bindinglist ::= (binding | `...') {`,' bindinglist}
|
// bindinglist ::= (binding | `...') {`,' bindinglist}
|
||||||
// Returns the location of the vararg ..., or std::nullopt if the function is not vararg.
|
// Returns the location of the vararg ..., or std::nullopt if the function is not vararg.
|
||||||
std::tuple<bool, Location, AstTypePack*> parseBindingList(TempVector<Binding>& result, bool allowDot3 = false);
|
std::tuple<bool, Location, AstTypePack*> parseBindingList(
|
||||||
|
TempVector<Binding>& result,
|
||||||
|
bool allowDot3 = false,
|
||||||
|
TempVector<Position>* commaPositions = nullptr
|
||||||
|
);
|
||||||
|
|
||||||
AstType* parseOptionalType();
|
AstType* parseOptionalType();
|
||||||
|
|
||||||
|
@ -201,14 +206,24 @@ private:
|
||||||
std::optional<AstTypeList> parseOptionalReturnType();
|
std::optional<AstTypeList> parseOptionalReturnType();
|
||||||
std::pair<Location, AstTypeList> parseReturnType();
|
std::pair<Location, AstTypeList> parseReturnType();
|
||||||
|
|
||||||
AstTableIndexer* parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation);
|
struct TableIndexerResult
|
||||||
|
{
|
||||||
|
AstTableIndexer* node;
|
||||||
|
Position indexerOpenPosition;
|
||||||
|
Position indexerClosePosition;
|
||||||
|
Position colonPosition;
|
||||||
|
};
|
||||||
|
|
||||||
|
TableIndexerResult parseTableIndexer(AstTableAccess access, std::optional<Location> accessLocation);
|
||||||
|
// Remove with FFlagLuauStoreCSTData
|
||||||
|
AstTableIndexer* parseTableIndexer_DEPRECATED(AstTableAccess access, std::optional<Location> accessLocation);
|
||||||
|
|
||||||
AstTypeOrPack parseFunctionType(bool allowPack, const AstArray<AstAttr*>& attributes);
|
AstTypeOrPack parseFunctionType(bool allowPack, const AstArray<AstAttr*>& attributes);
|
||||||
AstType* parseFunctionTypeTail(
|
AstType* parseFunctionTypeTail(
|
||||||
const Lexeme& begin,
|
const Lexeme& begin,
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
AstArray<AstGenericType> generics,
|
AstArray<AstGenericType*> generics,
|
||||||
AstArray<AstGenericTypePack> genericPacks,
|
AstArray<AstGenericTypePack*> genericPacks,
|
||||||
AstArray<AstType*> params,
|
AstArray<AstType*> params,
|
||||||
AstArray<std::optional<AstArgumentName>> paramNames,
|
AstArray<std::optional<AstArgumentName>> paramNames,
|
||||||
AstTypePack* varargAnnotation
|
AstTypePack* varargAnnotation
|
||||||
|
@ -259,6 +274,8 @@ private:
|
||||||
// args ::= `(' [explist] `)' | tableconstructor | String
|
// args ::= `(' [explist] `)' | tableconstructor | String
|
||||||
AstExpr* parseFunctionArgs(AstExpr* func, bool self);
|
AstExpr* parseFunctionArgs(AstExpr* func, bool self);
|
||||||
|
|
||||||
|
std::optional<CstExprTable::Separator> tableSeparator();
|
||||||
|
|
||||||
// tableconstructor ::= `{' [fieldlist] `}'
|
// tableconstructor ::= `{' [fieldlist] `}'
|
||||||
// fieldlist ::= field {fieldsep field} [fieldsep]
|
// fieldlist ::= field {fieldsep field} [fieldsep]
|
||||||
// field ::= `[' exp `]' `=' exp | Name `=' exp | exp
|
// field ::= `[' exp `]' `=' exp | Name `=' exp | exp
|
||||||
|
@ -277,12 +294,16 @@ private:
|
||||||
Name parseIndexName(const char* context, const Position& previous);
|
Name parseIndexName(const char* context, const Position& previous);
|
||||||
|
|
||||||
// `<' namelist `>'
|
// `<' namelist `>'
|
||||||
std::pair<AstArray<AstGenericType>, AstArray<AstGenericTypePack>> parseGenericTypeList(bool withDefaultValues);
|
std::pair<AstArray<AstGenericType*>, AstArray<AstGenericTypePack*>> parseGenericTypeList(bool withDefaultValues);
|
||||||
|
|
||||||
// `<' Type[, ...] `>'
|
// `<' Type[, ...] `>'
|
||||||
AstArray<AstTypeOrPack> parseTypeParams();
|
AstArray<AstTypeOrPack> parseTypeParams(
|
||||||
|
Position* openingPosition = nullptr,
|
||||||
|
TempVector<Position>* commaPositions = nullptr,
|
||||||
|
Position* closingPosition = nullptr
|
||||||
|
);
|
||||||
|
|
||||||
std::optional<AstArray<char>> parseCharArray();
|
std::optional<AstArray<char>> parseCharArray(AstArray<char>* originalString = nullptr);
|
||||||
AstExpr* parseString();
|
AstExpr* parseString();
|
||||||
AstExpr* parseNumber();
|
AstExpr* parseNumber();
|
||||||
|
|
||||||
|
@ -292,6 +313,9 @@ private:
|
||||||
|
|
||||||
void restoreLocals(unsigned int offset);
|
void restoreLocals(unsigned int offset);
|
||||||
|
|
||||||
|
/// Returns string quote style and block depth
|
||||||
|
std::pair<CstExprConstantString::QuoteStyle, unsigned int> extractStringDetails();
|
||||||
|
|
||||||
// check that parser is at lexeme/symbol, move to next lexeme/symbol on success, report failure and continue on failure
|
// check that parser is at lexeme/symbol, move to next lexeme/symbol on success, report failure and continue on failure
|
||||||
bool expectAndConsume(char value, const char* context = nullptr);
|
bool expectAndConsume(char value, const char* context = nullptr);
|
||||||
bool expectAndConsume(Lexeme::Type type, const char* context = nullptr);
|
bool expectAndConsume(Lexeme::Type type, const char* context = nullptr);
|
||||||
|
@ -435,6 +459,7 @@ private:
|
||||||
std::vector<AstAttr*> scratchAttr;
|
std::vector<AstAttr*> scratchAttr;
|
||||||
std::vector<AstStat*> scratchStat;
|
std::vector<AstStat*> scratchStat;
|
||||||
std::vector<AstArray<char>> scratchString;
|
std::vector<AstArray<char>> scratchString;
|
||||||
|
std::vector<AstArray<char>> scratchString2;
|
||||||
std::vector<AstExpr*> scratchExpr;
|
std::vector<AstExpr*> scratchExpr;
|
||||||
std::vector<AstExpr*> scratchExprAux;
|
std::vector<AstExpr*> scratchExprAux;
|
||||||
std::vector<AstName> scratchName;
|
std::vector<AstName> scratchName;
|
||||||
|
@ -442,15 +467,20 @@ private:
|
||||||
std::vector<Binding> scratchBinding;
|
std::vector<Binding> scratchBinding;
|
||||||
std::vector<AstLocal*> scratchLocal;
|
std::vector<AstLocal*> scratchLocal;
|
||||||
std::vector<AstTableProp> scratchTableTypeProps;
|
std::vector<AstTableProp> scratchTableTypeProps;
|
||||||
|
std::vector<CstTypeTable::Item> scratchCstTableTypeProps;
|
||||||
std::vector<AstType*> scratchType;
|
std::vector<AstType*> scratchType;
|
||||||
std::vector<AstTypeOrPack> scratchTypeOrPack;
|
std::vector<AstTypeOrPack> scratchTypeOrPack;
|
||||||
std::vector<AstDeclaredClassProp> scratchDeclaredClassProps;
|
std::vector<AstDeclaredClassProp> scratchDeclaredClassProps;
|
||||||
std::vector<AstExprTable::Item> scratchItem;
|
std::vector<AstExprTable::Item> scratchItem;
|
||||||
|
std::vector<CstExprTable::Item> scratchCstItem;
|
||||||
std::vector<AstArgumentName> scratchArgName;
|
std::vector<AstArgumentName> scratchArgName;
|
||||||
std::vector<AstGenericType> scratchGenericTypes;
|
std::vector<AstGenericType*> scratchGenericTypes;
|
||||||
std::vector<AstGenericTypePack> scratchGenericTypePacks;
|
std::vector<AstGenericTypePack*> scratchGenericTypePacks;
|
||||||
std::vector<std::optional<AstArgumentName>> scratchOptArgName;
|
std::vector<std::optional<AstArgumentName>> scratchOptArgName;
|
||||||
|
std::vector<Position> scratchPosition;
|
||||||
std::string scratchData;
|
std::string scratchData;
|
||||||
|
|
||||||
|
CstNodeMap cstNodeMap;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Luau
|
} // namespace Luau
|
||||||
|
|
|
@ -63,4 +63,4 @@ void* Allocator::allocate(size_t size)
|
||||||
return page->data;
|
return page->data;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
} // namespace Luau
|
||||||
|
|
|
@ -28,6 +28,38 @@ void AstAttr::visit(AstVisitor* visitor)
|
||||||
|
|
||||||
int gAstRttiIndex = 0;
|
int gAstRttiIndex = 0;
|
||||||
|
|
||||||
|
AstGenericType::AstGenericType(const Location& location, AstName name, AstType* defaultValue)
|
||||||
|
: AstNode(ClassIndex(), location)
|
||||||
|
, name(name)
|
||||||
|
, defaultValue(defaultValue)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void AstGenericType::visit(AstVisitor* visitor)
|
||||||
|
{
|
||||||
|
if (visitor->visit(this))
|
||||||
|
{
|
||||||
|
if (defaultValue)
|
||||||
|
defaultValue->visit(visitor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
AstGenericTypePack::AstGenericTypePack(const Location& location, AstName name, AstTypePack* defaultValue)
|
||||||
|
: AstNode(ClassIndex(), location)
|
||||||
|
, name(name)
|
||||||
|
, defaultValue(defaultValue)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void AstGenericTypePack::visit(AstVisitor* visitor)
|
||||||
|
{
|
||||||
|
if (visitor->visit(this))
|
||||||
|
{
|
||||||
|
if (defaultValue)
|
||||||
|
defaultValue->visit(visitor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
AstExprGroup::AstExprGroup(const Location& location, AstExpr* expr)
|
AstExprGroup::AstExprGroup(const Location& location, AstExpr* expr)
|
||||||
: AstExpr(ClassIndex(), location)
|
: AstExpr(ClassIndex(), location)
|
||||||
, expr(expr)
|
, expr(expr)
|
||||||
|
@ -185,8 +217,8 @@ void AstExprIndexExpr::visit(AstVisitor* visitor)
|
||||||
AstExprFunction::AstExprFunction(
|
AstExprFunction::AstExprFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
AstLocal* self,
|
AstLocal* self,
|
||||||
const AstArray<AstLocal*>& args,
|
const AstArray<AstLocal*>& args,
|
||||||
bool vararg,
|
bool vararg,
|
||||||
|
@ -721,8 +753,8 @@ AstStatTypeAlias::AstStatTypeAlias(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstName& name,
|
const AstName& name,
|
||||||
const Location& nameLocation,
|
const Location& nameLocation,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
AstType* type,
|
AstType* type,
|
||||||
bool exported
|
bool exported
|
||||||
)
|
)
|
||||||
|
@ -740,16 +772,14 @@ void AstStatTypeAlias::visit(AstVisitor* visitor)
|
||||||
{
|
{
|
||||||
if (visitor->visit(this))
|
if (visitor->visit(this))
|
||||||
{
|
{
|
||||||
for (const AstGenericType& el : generics)
|
for (AstGenericType* el : generics)
|
||||||
{
|
{
|
||||||
if (el.defaultValue)
|
el->visit(visitor);
|
||||||
el.defaultValue->visit(visitor);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const AstGenericTypePack& el : genericPacks)
|
for (AstGenericTypePack* el : genericPacks)
|
||||||
{
|
{
|
||||||
if (el.defaultValue)
|
el->visit(visitor);
|
||||||
el.defaultValue->visit(visitor);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type->visit(visitor);
|
type->visit(visitor);
|
||||||
|
@ -795,8 +825,8 @@ AstStatDeclareFunction::AstStatDeclareFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstName& name,
|
const AstName& name,
|
||||||
const Location& nameLocation,
|
const Location& nameLocation,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& params,
|
const AstTypeList& params,
|
||||||
const AstArray<AstArgumentName>& paramNames,
|
const AstArray<AstArgumentName>& paramNames,
|
||||||
bool vararg,
|
bool vararg,
|
||||||
|
@ -822,8 +852,8 @@ AstStatDeclareFunction::AstStatDeclareFunction(
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
const AstName& name,
|
const AstName& name,
|
||||||
const Location& nameLocation,
|
const Location& nameLocation,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& params,
|
const AstTypeList& params,
|
||||||
const AstArray<AstArgumentName>& paramNames,
|
const AstArray<AstArgumentName>& paramNames,
|
||||||
bool vararg,
|
bool vararg,
|
||||||
|
@ -970,8 +1000,8 @@ void AstTypeTable::visit(AstVisitor* visitor)
|
||||||
|
|
||||||
AstTypeFunction::AstTypeFunction(
|
AstTypeFunction::AstTypeFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& argTypes,
|
const AstTypeList& argTypes,
|
||||||
const AstArray<std::optional<AstArgumentName>>& argNames,
|
const AstArray<std::optional<AstArgumentName>>& argNames,
|
||||||
const AstTypeList& returnTypes
|
const AstTypeList& returnTypes
|
||||||
|
@ -990,8 +1020,8 @@ AstTypeFunction::AstTypeFunction(
|
||||||
AstTypeFunction::AstTypeFunction(
|
AstTypeFunction::AstTypeFunction(
|
||||||
const Location& location,
|
const Location& location,
|
||||||
const AstArray<AstAttr*>& attributes,
|
const AstArray<AstAttr*>& attributes,
|
||||||
const AstArray<AstGenericType>& generics,
|
const AstArray<AstGenericType*>& generics,
|
||||||
const AstArray<AstGenericTypePack>& genericPacks,
|
const AstArray<AstGenericTypePack*>& genericPacks,
|
||||||
const AstTypeList& argTypes,
|
const AstTypeList& argTypes,
|
||||||
const AstArray<std::optional<AstArgumentName>>& argNames,
|
const AstArray<std::optional<AstArgumentName>>& argNames,
|
||||||
const AstTypeList& returnTypes
|
const AstTypeList& returnTypes
|
||||||
|
@ -1091,6 +1121,18 @@ void AstTypeSingletonString::visit(AstVisitor* visitor)
|
||||||
visitor->visit(this);
|
visitor->visit(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
AstTypeGroup::AstTypeGroup(const Location& location, AstType* type)
|
||||||
|
: AstType(ClassIndex(), location)
|
||||||
|
, type(type)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void AstTypeGroup::visit(AstVisitor* visitor)
|
||||||
|
{
|
||||||
|
if (visitor->visit(this))
|
||||||
|
type->visit(visitor);
|
||||||
|
}
|
||||||
|
|
||||||
AstTypeError::AstTypeError(const Location& location, const AstArray<AstType*>& types, bool isMissing, unsigned messageIndex)
|
AstTypeError::AstTypeError(const Location& location, const AstArray<AstType*>& types, bool isMissing, unsigned messageIndex)
|
||||||
: AstType(ClassIndex(), location)
|
: AstType(ClassIndex(), location)
|
||||||
, types(types)
|
, types(types)
|
||||||
|
@ -1151,10 +1193,7 @@ void AstTypePackGeneric::visit(AstVisitor* visitor)
|
||||||
|
|
||||||
bool isLValue(const AstExpr* expr)
|
bool isLValue(const AstExpr* expr)
|
||||||
{
|
{
|
||||||
return expr->is<AstExprLocal>()
|
return expr->is<AstExprLocal>() || expr->is<AstExprGlobal>() || expr->is<AstExprIndexName>() || expr->is<AstExprIndexExpr>();
|
||||||
|| expr->is<AstExprGlobal>()
|
|
||||||
|| expr->is<AstExprIndexName>()
|
|
||||||
|| expr->is<AstExprIndexExpr>();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
AstName getIdentifier(AstExpr* node)
|
AstName getIdentifier(AstExpr* node)
|
||||||
|
|
175
Ast/src/Cst.cpp
Normal file
175
Ast/src/Cst.cpp
Normal file
|
@ -0,0 +1,175 @@
|
||||||
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
|
#include "Luau/Ast.h"
|
||||||
|
#include "Luau/Cst.h"
|
||||||
|
#include "Luau/Common.h"
|
||||||
|
|
||||||
|
namespace Luau
|
||||||
|
{
|
||||||
|
|
||||||
|
int gCstRttiIndex = 0;
|
||||||
|
|
||||||
|
CstExprConstantNumber::CstExprConstantNumber(const AstArray<char>& value)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, value(value)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprConstantString::CstExprConstantString(AstArray<char> sourceString, QuoteStyle quoteStyle, unsigned int blockDepth)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, sourceString(sourceString)
|
||||||
|
, quoteStyle(quoteStyle)
|
||||||
|
, blockDepth(blockDepth)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(blockDepth == 0 || quoteStyle == QuoteStyle::QuotedRaw);
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprCall::CstExprCall(std::optional<Position> openParens, std::optional<Position> closeParens, AstArray<Position> commaPositions)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, openParens(openParens)
|
||||||
|
, closeParens(closeParens)
|
||||||
|
, commaPositions(commaPositions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprIndexExpr::CstExprIndexExpr(Position openBracketPosition, Position closeBracketPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, openBracketPosition(openBracketPosition)
|
||||||
|
, closeBracketPosition(closeBracketPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprTable::CstExprTable(const AstArray<Item>& items)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, items(items)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprOp::CstExprOp(Position opPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, opPosition(opPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprTypeAssertion::CstExprTypeAssertion(Position opPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, opPosition(opPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprIfElse::CstExprIfElse(Position thenPosition, Position elsePosition, bool isElseIf)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, thenPosition(thenPosition)
|
||||||
|
, elsePosition(elsePosition)
|
||||||
|
, isElseIf(isElseIf)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstExprInterpString::CstExprInterpString(AstArray<AstArray<char>> sourceStrings, AstArray<Position> stringPositions)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, sourceStrings(sourceStrings)
|
||||||
|
, stringPositions(stringPositions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatDo::CstStatDo(Position endPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, endPosition(endPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatRepeat::CstStatRepeat(Position untilPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, untilPosition(untilPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatReturn::CstStatReturn(AstArray<Position> commaPositions)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, commaPositions(commaPositions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatLocal::CstStatLocal(AstArray<Position> varsCommaPositions, AstArray<Position> valuesCommaPositions)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, varsCommaPositions(varsCommaPositions)
|
||||||
|
, valuesCommaPositions(valuesCommaPositions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatFor::CstStatFor(Position equalsPosition, Position endCommaPosition, std::optional<Position> stepCommaPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, equalsPosition(equalsPosition)
|
||||||
|
, endCommaPosition(endCommaPosition)
|
||||||
|
, stepCommaPosition(stepCommaPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatForIn::CstStatForIn(AstArray<Position> varsCommaPositions, AstArray<Position> valuesCommaPositions)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, varsCommaPositions(varsCommaPositions)
|
||||||
|
, valuesCommaPositions(valuesCommaPositions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatAssign::CstStatAssign(
|
||||||
|
AstArray<Position> varsCommaPositions,
|
||||||
|
Position equalsPosition,
|
||||||
|
AstArray<Position> valuesCommaPositions
|
||||||
|
)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, varsCommaPositions(varsCommaPositions)
|
||||||
|
, equalsPosition(equalsPosition)
|
||||||
|
, valuesCommaPositions(valuesCommaPositions)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatCompoundAssign::CstStatCompoundAssign(Position opPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, opPosition(opPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstStatLocalFunction::CstStatLocalFunction(Position functionKeywordPosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, functionKeywordPosition(functionKeywordPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstTypeReference::CstTypeReference(
|
||||||
|
std::optional<Position> prefixPointPosition,
|
||||||
|
Position openParametersPosition,
|
||||||
|
AstArray<Position> parametersCommaPositions,
|
||||||
|
Position closeParametersPosition
|
||||||
|
)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, prefixPointPosition(prefixPointPosition)
|
||||||
|
, openParametersPosition(openParametersPosition)
|
||||||
|
, parametersCommaPositions(parametersCommaPositions)
|
||||||
|
, closeParametersPosition(closeParametersPosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstTypeTable::CstTypeTable(AstArray<Item> items, bool isArray)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, items(items)
|
||||||
|
, isArray(isArray)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstTypeTypeof::CstTypeTypeof(Position openPosition, Position closePosition)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, openPosition(openPosition)
|
||||||
|
, closePosition(closePosition)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
CstTypeSingletonString::CstTypeSingletonString(AstArray<char> sourceString, CstExprConstantString::QuoteStyle quoteStyle, unsigned int blockDepth)
|
||||||
|
: CstNode(CstClassIndex())
|
||||||
|
, sourceString(sourceString)
|
||||||
|
, quoteStyle(quoteStyle)
|
||||||
|
, blockDepth(blockDepth)
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(quoteStyle != CstExprConstantString::QuotedInterp);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Luau
|
|
@ -9,6 +9,8 @@
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LexerResumesFromPosition2)
|
LUAU_FASTFLAGVARIABLE(LexerResumesFromPosition2)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LexerFixInterpStringStart)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -304,6 +306,38 @@ static char unescape(char ch)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
unsigned int Lexeme::getBlockDepth() const
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(type == Lexeme::RawString || type == Lexeme::BlockComment);
|
||||||
|
|
||||||
|
// If we have a well-formed string, we are guaranteed to see 2 `]` characters after the end of the string contents
|
||||||
|
LUAU_ASSERT(*(data + length) == ']');
|
||||||
|
unsigned int depth = 0;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
depth++;
|
||||||
|
} while (*(data + length + depth) != ']');
|
||||||
|
|
||||||
|
return depth - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Lexeme::QuoteStyle Lexeme::getQuoteStyle() const
|
||||||
|
{
|
||||||
|
LUAU_ASSERT(type == Lexeme::QuotedString);
|
||||||
|
|
||||||
|
// If we have a well-formed string, we are guaranteed to see a closing delimiter after the string
|
||||||
|
LUAU_ASSERT(data);
|
||||||
|
|
||||||
|
char quote = *(data + length);
|
||||||
|
if (quote == '\'')
|
||||||
|
return Lexeme::QuoteStyle::Single;
|
||||||
|
else if (quote == '"')
|
||||||
|
return Lexeme::QuoteStyle::Double;
|
||||||
|
|
||||||
|
LUAU_ASSERT(!"Unknown quote style");
|
||||||
|
return Lexeme::QuoteStyle::Double; // unreachable, but required due to compiler warning
|
||||||
|
}
|
||||||
|
|
||||||
Lexer::Lexer(const char* buffer, size_t bufferSize, AstNameTable& names, Position startPosition)
|
Lexer::Lexer(const char* buffer, size_t bufferSize, AstNameTable& names, Position startPosition)
|
||||||
: buffer(buffer)
|
: buffer(buffer)
|
||||||
, bufferSize(bufferSize)
|
, bufferSize(bufferSize)
|
||||||
|
@ -759,7 +793,7 @@ Lexeme Lexer::readNext()
|
||||||
return Lexeme(Location(start, 1), '}');
|
return Lexeme(Location(start, 1), '}');
|
||||||
}
|
}
|
||||||
|
|
||||||
return readInterpolatedStringSection(position(), Lexeme::InterpStringMid, Lexeme::InterpStringEnd);
|
return readInterpolatedStringSection(FFlag::LexerFixInterpStringStart ? start : position(), Lexeme::InterpStringMid, Lexeme::InterpStringEnd);
|
||||||
}
|
}
|
||||||
|
|
||||||
case '=':
|
case '=':
|
||||||
|
|
|
@ -4,42 +4,6 @@
|
||||||
namespace Luau
|
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)
|
void Position::shift(const Position& start, const Position& oldEnd, const Position& newEnd)
|
||||||
{
|
{
|
||||||
if (*this >= start)
|
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
|
bool Location::encloses(const Location& l) const
|
||||||
{
|
{
|
||||||
return begin <= l.begin && end >= l.end;
|
return begin <= l.begin && end >= l.end;
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -10,7 +10,7 @@
|
||||||
std::optional<std::string> getCurrentWorkingDirectory();
|
std::optional<std::string> getCurrentWorkingDirectory();
|
||||||
|
|
||||||
std::string normalizePath(std::string_view path);
|
std::string normalizePath(std::string_view path);
|
||||||
std::string resolvePath(std::string_view relativePath, std::string_view baseFilePath);
|
std::optional<std::string> resolvePath(std::string_view relativePath, std::string_view baseFilePath);
|
||||||
|
|
||||||
std::optional<std::string> readFile(const std::string& name);
|
std::optional<std::string> readFile(const std::string& name);
|
||||||
std::optional<std::string> readStdin();
|
std::optional<std::string> readStdin();
|
||||||
|
@ -23,7 +23,7 @@ bool isDirectory(const std::string& path);
|
||||||
bool traverseDirectory(const std::string& path, const std::function<void(const std::string& name)>& callback);
|
bool traverseDirectory(const std::string& path, const std::function<void(const std::string& name)>& callback);
|
||||||
|
|
||||||
std::vector<std::string_view> splitPath(std::string_view path);
|
std::vector<std::string_view> splitPath(std::string_view path);
|
||||||
std::string joinPaths(const std::string& lhs, const std::string& rhs);
|
std::string joinPaths(std::string_view lhs, std::string_view rhs);
|
||||||
std::optional<std::string> getParentPath(const std::string& path);
|
std::optional<std::string> getParentPath(std::string_view path);
|
||||||
|
|
||||||
std::vector<std::string> getSourceFiles(int argc, char** argv);
|
std::vector<std::string> getSourceFiles(int argc, char** argv);
|
|
@ -7,9 +7,9 @@
|
||||||
#include "Luau/TypeAttach.h"
|
#include "Luau/TypeAttach.h"
|
||||||
#include "Luau/Transpiler.h"
|
#include "Luau/Transpiler.h"
|
||||||
|
|
||||||
#include "FileUtils.h"
|
#include "Luau/FileUtils.h"
|
||||||
#include "Flags.h"
|
#include "Luau/Flags.h"
|
||||||
#include "Require.h"
|
#include "Luau/Require.h"
|
||||||
|
|
||||||
#include <condition_variable>
|
#include <condition_variable>
|
||||||
#include <functional>
|
#include <functional>
|
|
@ -8,7 +8,7 @@
|
||||||
#include "Luau/ParseOptions.h"
|
#include "Luau/ParseOptions.h"
|
||||||
#include "Luau/ToString.h"
|
#include "Luau/ToString.h"
|
||||||
|
|
||||||
#include "FileUtils.h"
|
#include "Luau/FileUtils.h"
|
||||||
|
|
||||||
static void displayHelp(const char* argv0)
|
static void displayHelp(const char* argv0)
|
||||||
{
|
{
|
|
@ -7,8 +7,8 @@
|
||||||
#include "Luau/BytecodeBuilder.h"
|
#include "Luau/BytecodeBuilder.h"
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/BytecodeSummary.h"
|
#include "Luau/BytecodeSummary.h"
|
||||||
#include "FileUtils.h"
|
#include "Luau/FileUtils.h"
|
||||||
#include "Flags.h"
|
#include "Luau/Flags.h"
|
||||||
|
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
|
@ -8,8 +8,8 @@
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/TimeTrace.h"
|
#include "Luau/TimeTrace.h"
|
||||||
|
|
||||||
#include "FileUtils.h"
|
#include "Luau/FileUtils.h"
|
||||||
#include "Flags.h"
|
#include "Luau/Flags.h"
|
||||||
|
|
||||||
#include <memory>
|
#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.setDumpFlags(Luau::BytecodeBuilder::Dump_Source | Luau::BytecodeBuilder::Dump_Remarks);
|
||||||
bcb.setDumpSource(*source);
|
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(
|
bcb.setDumpFlags(
|
||||||
Luau::BytecodeBuilder::Dump_Code | Luau::BytecodeBuilder::Dump_Source | Luau::BytecodeBuilder::Dump_Locals |
|
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
|
// 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"
|
#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
|
// 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"
|
#include "Luau/Common.h"
|
||||||
|
|
||||||
|
@ -20,6 +20,7 @@
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <string_view>
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
static std::wstring fromUtf8(const std::string& path)
|
static std::wstring fromUtf8(const std::string& path)
|
||||||
|
@ -90,108 +91,76 @@ std::optional<std::string> getCurrentWorkingDirectory()
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns the normal/canonical form of a path (e.g. "../subfolder/../module.luau" -> "../module.luau")
|
|
||||||
std::string normalizePath(std::string_view path)
|
std::string normalizePath(std::string_view path)
|
||||||
{
|
{
|
||||||
return resolvePath(path, "");
|
const std::vector<std::string_view> components = splitPath(path);
|
||||||
}
|
std::vector<std::string_view> normalizedComponents;
|
||||||
|
|
||||||
// Takes a path that is relative to the file at baseFilePath and returns the path explicitly rebased onto baseFilePath.
|
const bool isAbsolute = isAbsolutePath(path);
|
||||||
// For absolute paths, baseFilePath will be ignored, and this function will resolve the path to a canonical path:
|
|
||||||
// (e.g. "/Users/.././Users/johndoe" -> "/Users/johndoe").
|
|
||||||
std::string resolvePath(std::string_view path, std::string_view baseFilePath)
|
|
||||||
{
|
|
||||||
std::vector<std::string_view> pathComponents;
|
|
||||||
std::vector<std::string_view> baseFilePathComponents;
|
|
||||||
|
|
||||||
// Dependent on whether the final resolved path is absolute or relative
|
// 1. Normalize path components
|
||||||
// - if relative (when path and baseFilePath are both relative), resolvedPathPrefix remains empty
|
const size_t startIndex = isAbsolute ? 1 : 0;
|
||||||
// - if absolute (if either path or baseFilePath are absolute), resolvedPathPrefix is "C:\", "/", etc.
|
for (size_t i = startIndex; i < components.size(); i++)
|
||||||
std::string resolvedPathPrefix;
|
|
||||||
bool isResolvedPathRelative = false;
|
|
||||||
|
|
||||||
if (isAbsolutePath(path))
|
|
||||||
{
|
|
||||||
// path is absolute, we use path's prefix and ignore baseFilePath
|
|
||||||
size_t afterPrefix = path.find_first_of("\\/") + 1;
|
|
||||||
resolvedPathPrefix = path.substr(0, afterPrefix);
|
|
||||||
pathComponents = splitPath(path.substr(afterPrefix));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
size_t afterPrefix = baseFilePath.find_first_of("\\/") + 1;
|
|
||||||
baseFilePathComponents = splitPath(baseFilePath.substr(afterPrefix));
|
|
||||||
if (isAbsolutePath(baseFilePath))
|
|
||||||
{
|
|
||||||
// path is relative and baseFilePath is absolute, we use baseFilePath's prefix
|
|
||||||
resolvedPathPrefix = baseFilePath.substr(0, afterPrefix);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// path and baseFilePath are both relative, we do not set a prefix (resolved path will be relative)
|
|
||||||
isResolvedPathRelative = true;
|
|
||||||
}
|
|
||||||
pathComponents = splitPath(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove filename from components
|
|
||||||
if (!baseFilePathComponents.empty())
|
|
||||||
baseFilePathComponents.pop_back();
|
|
||||||
|
|
||||||
// Resolve the path by applying pathComponents to baseFilePathComponents
|
|
||||||
int numPrependedParents = 0;
|
|
||||||
for (std::string_view component : pathComponents)
|
|
||||||
{
|
{
|
||||||
|
std::string_view component = components[i];
|
||||||
if (component == "..")
|
if (component == "..")
|
||||||
{
|
{
|
||||||
if (baseFilePathComponents.empty())
|
if (normalizedComponents.empty())
|
||||||
{
|
{
|
||||||
if (isResolvedPathRelative)
|
if (!isAbsolute)
|
||||||
numPrependedParents++; // "../" will later be added to the beginning of the resolved path
|
|
||||||
}
|
|
||||||
else if (baseFilePathComponents.back() != "..")
|
|
||||||
{
|
{
|
||||||
baseFilePathComponents.pop_back(); // Resolve cases like "folder/subfolder/../../file" to "file"
|
normalizedComponents.emplace_back("..");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (component != "." && !component.empty())
|
else if (normalizedComponents.back() == "..")
|
||||||
{
|
{
|
||||||
baseFilePathComponents.push_back(component);
|
normalizedComponents.emplace_back("..");
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create resolved path prefix for relative paths
|
|
||||||
if (isResolvedPathRelative)
|
|
||||||
{
|
|
||||||
if (numPrependedParents > 0)
|
|
||||||
{
|
|
||||||
resolvedPathPrefix.reserve(numPrependedParents * 3);
|
|
||||||
for (int i = 0; i < numPrependedParents; i++)
|
|
||||||
{
|
|
||||||
resolvedPathPrefix += "../";
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
resolvedPathPrefix = "./";
|
normalizedComponents.pop_back();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (!component.empty() && component != ".")
|
||||||
|
{
|
||||||
|
normalizedComponents.emplace_back(component);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Join baseFilePathComponents to form the resolved path
|
std::string normalizedPath;
|
||||||
std::string resolvedPath = resolvedPathPrefix;
|
|
||||||
for (auto iter = baseFilePathComponents.begin(); iter != baseFilePathComponents.end(); ++iter)
|
|
||||||
{
|
|
||||||
if (iter != baseFilePathComponents.begin())
|
|
||||||
resolvedPath += "/";
|
|
||||||
|
|
||||||
resolvedPath += *iter;
|
// 2. Add correct prefix to formatted path
|
||||||
}
|
if (isAbsolute)
|
||||||
if (resolvedPath.size() > resolvedPathPrefix.size() && resolvedPath.back() == '/')
|
|
||||||
{
|
{
|
||||||
// Remove trailing '/' if present
|
normalizedPath += components[0];
|
||||||
resolvedPath.pop_back();
|
normalizedPath += "/";
|
||||||
}
|
}
|
||||||
return resolvedPath;
|
else if (normalizedComponents.empty() || normalizedComponents[0] != "..")
|
||||||
|
{
|
||||||
|
normalizedPath += "./";
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Join path components to form the normalized path
|
||||||
|
for (auto iter = normalizedComponents.begin(); iter != normalizedComponents.end(); ++iter)
|
||||||
|
{
|
||||||
|
if (iter != normalizedComponents.begin())
|
||||||
|
normalizedPath += "/";
|
||||||
|
|
||||||
|
normalizedPath += *iter;
|
||||||
|
}
|
||||||
|
if (normalizedPath.size() >= 2 && normalizedPath[normalizedPath.size() - 1] == '.' && normalizedPath[normalizedPath.size() - 2] == '.')
|
||||||
|
normalizedPath += "/";
|
||||||
|
|
||||||
|
return normalizedPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<std::string> resolvePath(std::string_view path, std::string_view baseFilePath)
|
||||||
|
{
|
||||||
|
std::optional<std::string> baseFilePathParent = getParentPath(baseFilePath);
|
||||||
|
if (!baseFilePathParent)
|
||||||
|
return std::nullopt;
|
||||||
|
|
||||||
|
return normalizePath(joinPaths(*baseFilePathParent, path));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool hasFileExtension(std::string_view name, const std::vector<std::string>& extensions)
|
bool hasFileExtension(std::string_view name, const std::vector<std::string>& extensions)
|
||||||
|
@ -416,16 +385,16 @@ std::vector<std::string_view> splitPath(std::string_view path)
|
||||||
return components;
|
return components;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string joinPaths(const std::string& lhs, const std::string& rhs)
|
std::string joinPaths(std::string_view lhs, std::string_view rhs)
|
||||||
{
|
{
|
||||||
std::string result = lhs;
|
std::string result = std::string(lhs);
|
||||||
if (!result.empty() && result.back() != '/' && result.back() != '\\')
|
if (!result.empty() && result.back() != '/' && result.back() != '\\')
|
||||||
result += '/';
|
result += '/';
|
||||||
result += rhs;
|
result += rhs;
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::optional<std::string> getParentPath(const std::string& path)
|
std::optional<std::string> getParentPath(std::string_view path)
|
||||||
{
|
{
|
||||||
if (path == "" || path == "." || path == "/")
|
if (path == "" || path == "." || path == "/")
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
|
@ -441,7 +410,7 @@ std::optional<std::string> getParentPath(const std::string& path)
|
||||||
return "/";
|
return "/";
|
||||||
|
|
||||||
if (slash != std::string::npos)
|
if (slash != std::string::npos)
|
||||||
return path.substr(0, slash);
|
return std::string(path.substr(0, slash));
|
||||||
|
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
@ -471,10 +440,12 @@ std::vector<std::string> getSourceFiles(int argc, char** argv)
|
||||||
if (argv[i][0] == '-' && argv[i][1] != '\0')
|
if (argv[i][0] == '-' && argv[i][1] != '\0')
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (isDirectory(argv[i]))
|
std::string normalized = normalizePath(argv[i]);
|
||||||
|
|
||||||
|
if (isDirectory(normalized))
|
||||||
{
|
{
|
||||||
traverseDirectory(
|
traverseDirectory(
|
||||||
argv[i],
|
normalized,
|
||||||
[&](const std::string& name)
|
[&](const std::string& name)
|
||||||
{
|
{
|
||||||
std::string ext = getExtension(name);
|
std::string ext = getExtension(name);
|
||||||
|
@ -486,7 +457,7 @@ std::vector<std::string> getSourceFiles(int argc, char** argv)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
files.push_back(argv[i]);
|
files.push_back(normalized);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/Transpiler.h"
|
#include "Luau/Transpiler.h"
|
||||||
|
|
||||||
#include "FileUtils.h"
|
#include "Luau/FileUtils.h"
|
||||||
|
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <stdio.h>
|
#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
|
// 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 "Luau/Common.h"
|
||||||
#include "lua.h"
|
#include "lua.h"
|
||||||
|
@ -10,11 +10,11 @@
|
||||||
#include "Luau/Parser.h"
|
#include "Luau/Parser.h"
|
||||||
#include "Luau/TimeTrace.h"
|
#include "Luau/TimeTrace.h"
|
||||||
|
|
||||||
#include "Coverage.h"
|
#include "Luau/Coverage.h"
|
||||||
#include "FileUtils.h"
|
#include "Luau/FileUtils.h"
|
||||||
#include "Flags.h"
|
#include "Luau/Flags.h"
|
||||||
#include "Profiler.h"
|
#include "Luau/Profiler.h"
|
||||||
#include "Require.h"
|
#include "Luau/Require.h"
|
||||||
|
|
||||||
#include "isocline.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
|
// 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)
|
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
|
// 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/Common.h"
|
||||||
#include "Luau/Config.h"
|
#include "Luau/Config.h"
|
||||||
|
|
||||||
|
@ -141,8 +141,17 @@ bool RequireResolver::resolveAndStoreDefaultPaths()
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// resolvePath automatically sanitizes/normalizes the paths
|
// resolvePath automatically sanitizes/normalizes the paths
|
||||||
resolvedRequire.identifier = resolvePath(pathToResolve, identifierContext);
|
std::optional<std::string> identifier = resolvePath(pathToResolve, identifierContext);
|
||||||
resolvedRequire.absolutePath = resolvePath(pathToResolve, *absolutePathContext);
|
std::optional<std::string> absolutePath = resolvePath(pathToResolve, *absolutePathContext);
|
||||||
|
|
||||||
|
if (!identifier || !absolutePath)
|
||||||
|
{
|
||||||
|
errorHandler.reportError("could not resolve require path");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
resolvedRequire.identifier = std::move(*identifier);
|
||||||
|
resolvedRequire.absolutePath = std::move(*absolutePath);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -181,7 +190,7 @@ std::optional<std::string> RequireResolver::getRequiringContextAbsolute()
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Require statement is being executed in a file, must resolve relative to CWD
|
// Require statement is being executed in a file, must resolve relative to CWD
|
||||||
requiringFile = resolvePath(requireContext.getPath(), joinPaths(*cwd, "stdin"));
|
requiringFile = normalizePath(joinPaths(*cwd, requireContext.getPath()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
std::replace(requiringFile.begin(), requiringFile.end(), '\\', '/');
|
std::replace(requiringFile.begin(), requiringFile.end(), '\\', '/');
|
||||||
|
@ -190,7 +199,7 @@ std::optional<std::string> RequireResolver::getRequiringContextAbsolute()
|
||||||
|
|
||||||
std::string RequireResolver::getRequiringContextRelative()
|
std::string RequireResolver::getRequiringContextRelative()
|
||||||
{
|
{
|
||||||
return requireContext.isStdin() ? "" : requireContext.getPath();
|
return requireContext.isStdin() ? "./" : requireContext.getPath();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool RequireResolver::substituteAliasIfPresent(std::string& path)
|
bool RequireResolver::substituteAliasIfPresent(std::string& path)
|
|
@ -68,11 +68,12 @@ include(Sources.cmake)
|
||||||
target_include_directories(Luau.Common INTERFACE Common/include)
|
target_include_directories(Luau.Common INTERFACE Common/include)
|
||||||
|
|
||||||
target_compile_features(Luau.CLI.lib PUBLIC cxx_std_17)
|
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_compile_features(Luau.Ast PUBLIC cxx_std_17)
|
||||||
target_include_directories(Luau.Ast PUBLIC Ast/include)
|
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_compile_features(Luau.Compiler PUBLIC cxx_std_17)
|
||||||
target_include_directories(Luau.Compiler PUBLIC Compiler/include)
|
target_include_directories(Luau.Compiler PUBLIC Compiler/include)
|
||||||
|
|
|
@ -160,6 +160,7 @@ public:
|
||||||
void vmaxsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
void vmaxsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
||||||
void vminsd(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 vcmpltsd(OperandX64 dst, OperandX64 src1, OperandX64 src2);
|
||||||
|
|
||||||
void vblendvpd(RegisterX64 dst, RegisterX64 src1, OperandX64 mask, RegisterX64 src3);
|
void vblendvpd(RegisterX64 dst, RegisterX64 src1, OperandX64 mask, RegisterX64 src3);
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "Luau/CodeGen.h"
|
#include "Luau/CodeGenOptions.h"
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,10 @@
|
||||||
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <algorithm>
|
#include "Luau/CodeGenCommon.h"
|
||||||
|
#include "Luau/CodeGenOptions.h"
|
||||||
|
#include "Luau/LoweringStats.h"
|
||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
@ -12,25 +15,11 @@
|
||||||
|
|
||||||
struct lua_State;
|
struct lua_State;
|
||||||
|
|
||||||
#if defined(__x86_64__) || defined(_M_X64)
|
|
||||||
#define CODEGEN_TARGET_X64
|
|
||||||
#elif defined(__aarch64__) || defined(_M_ARM64)
|
|
||||||
#define CODEGEN_TARGET_A64
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
namespace CodeGen
|
namespace CodeGen
|
||||||
{
|
{
|
||||||
|
|
||||||
enum CodeGenFlags
|
|
||||||
{
|
|
||||||
// Only run native codegen for modules that have been marked with --!native
|
|
||||||
CodeGen_OnlyNativeModules = 1 << 0,
|
|
||||||
// Run native codegen for functions that the compiler considers not profitable
|
|
||||||
CodeGen_ColdFunctions = 1 << 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
// These enum values can be reported through telemetry.
|
// These enum values can be reported through telemetry.
|
||||||
// To ensure consistency, changes should be additive.
|
// To ensure consistency, changes should be additive.
|
||||||
enum class CodeGenCompilationResult
|
enum class CodeGenCompilationResult
|
||||||
|
@ -72,106 +61,6 @@ struct CompilationResult
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
struct IrBuilder;
|
|
||||||
struct IrOp;
|
|
||||||
|
|
||||||
using HostVectorOperationBytecodeType = uint8_t (*)(const char* member, size_t memberLength);
|
|
||||||
using HostVectorAccessHandler = bool (*)(IrBuilder& builder, const char* member, size_t memberLength, int resultReg, int sourceReg, int pcpos);
|
|
||||||
using HostVectorNamecallHandler =
|
|
||||||
bool (*)(IrBuilder& builder, const char* member, size_t memberLength, int argResReg, int sourceReg, int params, int results, int pcpos);
|
|
||||||
|
|
||||||
enum class HostMetamethod
|
|
||||||
{
|
|
||||||
Add,
|
|
||||||
Sub,
|
|
||||||
Mul,
|
|
||||||
Div,
|
|
||||||
Idiv,
|
|
||||||
Mod,
|
|
||||||
Pow,
|
|
||||||
Minus,
|
|
||||||
Equal,
|
|
||||||
LessThan,
|
|
||||||
LessEqual,
|
|
||||||
Length,
|
|
||||||
Concat,
|
|
||||||
};
|
|
||||||
|
|
||||||
using HostUserdataOperationBytecodeType = uint8_t (*)(uint8_t type, const char* member, size_t memberLength);
|
|
||||||
using HostUserdataMetamethodBytecodeType = uint8_t (*)(uint8_t lhsTy, uint8_t rhsTy, HostMetamethod method);
|
|
||||||
using HostUserdataAccessHandler =
|
|
||||||
bool (*)(IrBuilder& builder, uint8_t type, const char* member, size_t memberLength, int resultReg, int sourceReg, int pcpos);
|
|
||||||
using HostUserdataMetamethodHandler =
|
|
||||||
bool (*)(IrBuilder& builder, uint8_t lhsTy, uint8_t rhsTy, int resultReg, IrOp lhs, IrOp rhs, HostMetamethod method, int pcpos);
|
|
||||||
using HostUserdataNamecallHandler = bool (*)(
|
|
||||||
IrBuilder& builder,
|
|
||||||
uint8_t type,
|
|
||||||
const char* member,
|
|
||||||
size_t memberLength,
|
|
||||||
int argResReg,
|
|
||||||
int sourceReg,
|
|
||||||
int params,
|
|
||||||
int results,
|
|
||||||
int pcpos
|
|
||||||
);
|
|
||||||
|
|
||||||
struct HostIrHooks
|
|
||||||
{
|
|
||||||
// Suggest result type of a vector field access
|
|
||||||
HostVectorOperationBytecodeType vectorAccessBytecodeType = nullptr;
|
|
||||||
|
|
||||||
// Suggest result type of a vector function namecall
|
|
||||||
HostVectorOperationBytecodeType vectorNamecallBytecodeType = nullptr;
|
|
||||||
|
|
||||||
// Handle vector value field access
|
|
||||||
// 'sourceReg' is guaranteed to be a vector
|
|
||||||
// Guards should take a VM exit to 'pcpos'
|
|
||||||
HostVectorAccessHandler vectorAccess = nullptr;
|
|
||||||
|
|
||||||
// Handle namecall performed on a vector value
|
|
||||||
// 'sourceReg' (self argument) is guaranteed to be a vector
|
|
||||||
// All other arguments can be of any type
|
|
||||||
// Guards should take a VM exit to 'pcpos'
|
|
||||||
HostVectorNamecallHandler vectorNamecall = nullptr;
|
|
||||||
|
|
||||||
// Suggest result type of a userdata field access
|
|
||||||
HostUserdataOperationBytecodeType userdataAccessBytecodeType = nullptr;
|
|
||||||
|
|
||||||
// Suggest result type of a metamethod call
|
|
||||||
HostUserdataMetamethodBytecodeType userdataMetamethodBytecodeType = nullptr;
|
|
||||||
|
|
||||||
// Suggest result type of a userdata namecall
|
|
||||||
HostUserdataOperationBytecodeType userdataNamecallBytecodeType = nullptr;
|
|
||||||
|
|
||||||
// Handle userdata value field access
|
|
||||||
// 'sourceReg' is guaranteed to be a userdata, but tag has to be checked
|
|
||||||
// Write to 'resultReg' might invalidate 'sourceReg'
|
|
||||||
// Guards should take a VM exit to 'pcpos'
|
|
||||||
HostUserdataAccessHandler userdataAccess = nullptr;
|
|
||||||
|
|
||||||
// Handle metamethod operation on a userdata value
|
|
||||||
// 'lhs' and 'rhs' operands can be VM registers of constants
|
|
||||||
// Operand types have to be checked and userdata operand tags have to be checked
|
|
||||||
// Write to 'resultReg' might invalidate source operands
|
|
||||||
// Guards should take a VM exit to 'pcpos'
|
|
||||||
HostUserdataMetamethodHandler userdataMetamethod = nullptr;
|
|
||||||
|
|
||||||
// Handle namecall performed on a userdata value
|
|
||||||
// 'sourceReg' (self argument) is guaranteed to be a userdata, but tag has to be checked
|
|
||||||
// All other arguments can be of any type
|
|
||||||
// Guards should take a VM exit to 'pcpos'
|
|
||||||
HostUserdataNamecallHandler userdataNamecall = nullptr;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct CompilationOptions
|
|
||||||
{
|
|
||||||
unsigned int flags = 0;
|
|
||||||
HostIrHooks hooks;
|
|
||||||
|
|
||||||
// null-terminated array of userdata types names that might have custom lowering
|
|
||||||
const char* const* userdataTypes = nullptr;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct CompilationStats
|
struct CompilationStats
|
||||||
{
|
{
|
||||||
size_t bytecodeSizeBytes = 0;
|
size_t bytecodeSizeBytes = 0;
|
||||||
|
@ -184,8 +73,6 @@ struct CompilationStats
|
||||||
uint32_t functionsBound = 0;
|
uint32_t functionsBound = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
using AllocationCallback = void(void* context, void* oldPointer, size_t oldSize, void* newPointer, size_t newSize);
|
|
||||||
|
|
||||||
bool isSupported();
|
bool isSupported();
|
||||||
|
|
||||||
class SharedCodeGenContext;
|
class SharedCodeGenContext;
|
||||||
|
@ -249,153 +136,6 @@ CompilationResult compile(const ModuleId& moduleId, lua_State* L, int idx, unsig
|
||||||
CompilationResult compile(lua_State* L, int idx, const CompilationOptions& options, CompilationStats* stats = nullptr);
|
CompilationResult compile(lua_State* L, int idx, const CompilationOptions& options, CompilationStats* stats = nullptr);
|
||||||
CompilationResult compile(const ModuleId& moduleId, lua_State* L, int idx, const CompilationOptions& options, CompilationStats* stats = nullptr);
|
CompilationResult compile(const ModuleId& moduleId, lua_State* L, int idx, const CompilationOptions& options, CompilationStats* stats = nullptr);
|
||||||
|
|
||||||
using AnnotatorFn = void (*)(void* context, std::string& result, int fid, int instpos);
|
|
||||||
|
|
||||||
// Output "#" before IR blocks and instructions
|
|
||||||
enum class IncludeIrPrefix
|
|
||||||
{
|
|
||||||
No,
|
|
||||||
Yes
|
|
||||||
};
|
|
||||||
|
|
||||||
// Output user count and last use information of blocks and instructions
|
|
||||||
enum class IncludeUseInfo
|
|
||||||
{
|
|
||||||
No,
|
|
||||||
Yes
|
|
||||||
};
|
|
||||||
|
|
||||||
// Output CFG informations like block predecessors, successors and etc
|
|
||||||
enum class IncludeCfgInfo
|
|
||||||
{
|
|
||||||
No,
|
|
||||||
Yes
|
|
||||||
};
|
|
||||||
|
|
||||||
// Output VM register live in/out information for blocks
|
|
||||||
enum class IncludeRegFlowInfo
|
|
||||||
{
|
|
||||||
No,
|
|
||||||
Yes
|
|
||||||
};
|
|
||||||
|
|
||||||
struct AssemblyOptions
|
|
||||||
{
|
|
||||||
enum Target
|
|
||||||
{
|
|
||||||
Host,
|
|
||||||
A64,
|
|
||||||
A64_NoFeatures,
|
|
||||||
X64_Windows,
|
|
||||||
X64_SystemV,
|
|
||||||
};
|
|
||||||
|
|
||||||
Target target = Host;
|
|
||||||
|
|
||||||
CompilationOptions compilationOptions;
|
|
||||||
|
|
||||||
bool outputBinary = false;
|
|
||||||
|
|
||||||
bool includeAssembly = false;
|
|
||||||
bool includeIr = false;
|
|
||||||
bool includeOutlinedCode = false;
|
|
||||||
bool includeIrTypes = false;
|
|
||||||
|
|
||||||
IncludeIrPrefix includeIrPrefix = IncludeIrPrefix::Yes;
|
|
||||||
IncludeUseInfo includeUseInfo = IncludeUseInfo::Yes;
|
|
||||||
IncludeCfgInfo includeCfgInfo = IncludeCfgInfo::Yes;
|
|
||||||
IncludeRegFlowInfo includeRegFlowInfo = IncludeRegFlowInfo::Yes;
|
|
||||||
|
|
||||||
// Optional annotator function can be provided to describe each instruction, it takes function id and sequential instruction id
|
|
||||||
AnnotatorFn annotator = nullptr;
|
|
||||||
void* annotatorContext = nullptr;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct BlockLinearizationStats
|
|
||||||
{
|
|
||||||
unsigned int constPropInstructionCount = 0;
|
|
||||||
double timeSeconds = 0.0;
|
|
||||||
|
|
||||||
BlockLinearizationStats& operator+=(const BlockLinearizationStats& that)
|
|
||||||
{
|
|
||||||
this->constPropInstructionCount += that.constPropInstructionCount;
|
|
||||||
this->timeSeconds += that.timeSeconds;
|
|
||||||
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
BlockLinearizationStats operator+(const BlockLinearizationStats& other) const
|
|
||||||
{
|
|
||||||
BlockLinearizationStats result(*this);
|
|
||||||
result += other;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
enum FunctionStatsFlags
|
|
||||||
{
|
|
||||||
// Enable stats collection per function
|
|
||||||
FunctionStats_Enable = 1 << 0,
|
|
||||||
// Compute function bytecode summary
|
|
||||||
FunctionStats_BytecodeSummary = 1 << 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
struct FunctionStats
|
|
||||||
{
|
|
||||||
std::string name;
|
|
||||||
int line = -1;
|
|
||||||
unsigned bcodeCount = 0;
|
|
||||||
unsigned irCount = 0;
|
|
||||||
unsigned asmCount = 0;
|
|
||||||
unsigned asmSize = 0;
|
|
||||||
std::vector<std::vector<unsigned>> bytecodeSummary;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct LoweringStats
|
|
||||||
{
|
|
||||||
unsigned totalFunctions = 0;
|
|
||||||
unsigned skippedFunctions = 0;
|
|
||||||
int spillsToSlot = 0;
|
|
||||||
int spillsToRestore = 0;
|
|
||||||
unsigned maxSpillSlotsUsed = 0;
|
|
||||||
unsigned blocksPreOpt = 0;
|
|
||||||
unsigned blocksPostOpt = 0;
|
|
||||||
unsigned maxBlockInstructions = 0;
|
|
||||||
|
|
||||||
int regAllocErrors = 0;
|
|
||||||
int loweringErrors = 0;
|
|
||||||
|
|
||||||
BlockLinearizationStats blockLinearizationStats;
|
|
||||||
|
|
||||||
unsigned functionStatsFlags = 0;
|
|
||||||
std::vector<FunctionStats> functions;
|
|
||||||
|
|
||||||
LoweringStats operator+(const LoweringStats& other) const
|
|
||||||
{
|
|
||||||
LoweringStats result(*this);
|
|
||||||
result += other;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
LoweringStats& operator+=(const LoweringStats& that)
|
|
||||||
{
|
|
||||||
this->totalFunctions += that.totalFunctions;
|
|
||||||
this->skippedFunctions += that.skippedFunctions;
|
|
||||||
this->spillsToSlot += that.spillsToSlot;
|
|
||||||
this->spillsToRestore += that.spillsToRestore;
|
|
||||||
this->maxSpillSlotsUsed = std::max(this->maxSpillSlotsUsed, that.maxSpillSlotsUsed);
|
|
||||||
this->blocksPreOpt += that.blocksPreOpt;
|
|
||||||
this->blocksPostOpt += that.blocksPostOpt;
|
|
||||||
this->maxBlockInstructions = std::max(this->maxBlockInstructions, that.maxBlockInstructions);
|
|
||||||
this->regAllocErrors += that.regAllocErrors;
|
|
||||||
this->loweringErrors += that.loweringErrors;
|
|
||||||
this->blockLinearizationStats += that.blockLinearizationStats;
|
|
||||||
if (this->functionStatsFlags & FunctionStats_Enable)
|
|
||||||
this->functions.insert(this->functions.end(), that.functions.begin(), that.functions.end());
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Generates assembly for target function and all inner functions
|
// Generates assembly for target function and all inner functions
|
||||||
std::string getAssembly(lua_State* L, int idx, AssemblyOptions options = {}, LoweringStats* stats = nullptr);
|
std::string getAssembly(lua_State* L, int idx, AssemblyOptions options = {}, LoweringStats* stats = nullptr);
|
||||||
|
|
||||||
|
|
|
@ -10,3 +10,9 @@
|
||||||
#else
|
#else
|
||||||
#define CODEGEN_ASSERT(expr) (void)sizeof(!!(expr))
|
#define CODEGEN_ASSERT(expr) (void)sizeof(!!(expr))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if defined(__x86_64__) || defined(_M_X64)
|
||||||
|
#define CODEGEN_TARGET_X64
|
||||||
|
#elif defined(__aarch64__) || defined(_M_ARM64)
|
||||||
|
#define CODEGEN_TARGET_A64
|
||||||
|
#endif
|
||||||
|
|
188
CodeGen/include/Luau/CodeGenOptions.h
Normal file
188
CodeGen/include/Luau/CodeGenOptions.h
Normal file
|
@ -0,0 +1,188 @@
|
||||||
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
namespace Luau
|
||||||
|
{
|
||||||
|
namespace CodeGen
|
||||||
|
{
|
||||||
|
|
||||||
|
enum CodeGenFlags
|
||||||
|
{
|
||||||
|
// Only run native codegen for modules that have been marked with --!native
|
||||||
|
CodeGen_OnlyNativeModules = 1 << 0,
|
||||||
|
// Run native codegen for functions that the compiler considers not profitable
|
||||||
|
CodeGen_ColdFunctions = 1 << 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
using AllocationCallback = void(void* context, void* oldPointer, size_t oldSize, void* newPointer, size_t newSize);
|
||||||
|
|
||||||
|
struct IrBuilder;
|
||||||
|
struct IrOp;
|
||||||
|
|
||||||
|
using HostVectorOperationBytecodeType = uint8_t (*)(const char* member, size_t memberLength);
|
||||||
|
using HostVectorAccessHandler = bool (*)(IrBuilder& builder, const char* member, size_t memberLength, int resultReg, int sourceReg, int pcpos);
|
||||||
|
using HostVectorNamecallHandler =
|
||||||
|
bool (*)(IrBuilder& builder, const char* member, size_t memberLength, int argResReg, int sourceReg, int params, int results, int pcpos);
|
||||||
|
|
||||||
|
enum class HostMetamethod
|
||||||
|
{
|
||||||
|
Add,
|
||||||
|
Sub,
|
||||||
|
Mul,
|
||||||
|
Div,
|
||||||
|
Idiv,
|
||||||
|
Mod,
|
||||||
|
Pow,
|
||||||
|
Minus,
|
||||||
|
Equal,
|
||||||
|
LessThan,
|
||||||
|
LessEqual,
|
||||||
|
Length,
|
||||||
|
Concat,
|
||||||
|
};
|
||||||
|
|
||||||
|
using HostUserdataOperationBytecodeType = uint8_t (*)(uint8_t type, const char* member, size_t memberLength);
|
||||||
|
using HostUserdataMetamethodBytecodeType = uint8_t (*)(uint8_t lhsTy, uint8_t rhsTy, HostMetamethod method);
|
||||||
|
using HostUserdataAccessHandler =
|
||||||
|
bool (*)(IrBuilder& builder, uint8_t type, const char* member, size_t memberLength, int resultReg, int sourceReg, int pcpos);
|
||||||
|
using HostUserdataMetamethodHandler =
|
||||||
|
bool (*)(IrBuilder& builder, uint8_t lhsTy, uint8_t rhsTy, int resultReg, IrOp lhs, IrOp rhs, HostMetamethod method, int pcpos);
|
||||||
|
using HostUserdataNamecallHandler = bool (*)(
|
||||||
|
IrBuilder& builder,
|
||||||
|
uint8_t type,
|
||||||
|
const char* member,
|
||||||
|
size_t memberLength,
|
||||||
|
int argResReg,
|
||||||
|
int sourceReg,
|
||||||
|
int params,
|
||||||
|
int results,
|
||||||
|
int pcpos
|
||||||
|
);
|
||||||
|
|
||||||
|
struct HostIrHooks
|
||||||
|
{
|
||||||
|
// Suggest result type of a vector field access
|
||||||
|
HostVectorOperationBytecodeType vectorAccessBytecodeType = nullptr;
|
||||||
|
|
||||||
|
// Suggest result type of a vector function namecall
|
||||||
|
HostVectorOperationBytecodeType vectorNamecallBytecodeType = nullptr;
|
||||||
|
|
||||||
|
// Handle vector value field access
|
||||||
|
// 'sourceReg' is guaranteed to be a vector
|
||||||
|
// Guards should take a VM exit to 'pcpos'
|
||||||
|
HostVectorAccessHandler vectorAccess = nullptr;
|
||||||
|
|
||||||
|
// Handle namecall performed on a vector value
|
||||||
|
// 'sourceReg' (self argument) is guaranteed to be a vector
|
||||||
|
// All other arguments can be of any type
|
||||||
|
// Guards should take a VM exit to 'pcpos'
|
||||||
|
HostVectorNamecallHandler vectorNamecall = nullptr;
|
||||||
|
|
||||||
|
// Suggest result type of a userdata field access
|
||||||
|
HostUserdataOperationBytecodeType userdataAccessBytecodeType = nullptr;
|
||||||
|
|
||||||
|
// Suggest result type of a metamethod call
|
||||||
|
HostUserdataMetamethodBytecodeType userdataMetamethodBytecodeType = nullptr;
|
||||||
|
|
||||||
|
// Suggest result type of a userdata namecall
|
||||||
|
HostUserdataOperationBytecodeType userdataNamecallBytecodeType = nullptr;
|
||||||
|
|
||||||
|
// Handle userdata value field access
|
||||||
|
// 'sourceReg' is guaranteed to be a userdata, but tag has to be checked
|
||||||
|
// Write to 'resultReg' might invalidate 'sourceReg'
|
||||||
|
// Guards should take a VM exit to 'pcpos'
|
||||||
|
HostUserdataAccessHandler userdataAccess = nullptr;
|
||||||
|
|
||||||
|
// Handle metamethod operation on a userdata value
|
||||||
|
// 'lhs' and 'rhs' operands can be VM registers of constants
|
||||||
|
// Operand types have to be checked and userdata operand tags have to be checked
|
||||||
|
// Write to 'resultReg' might invalidate source operands
|
||||||
|
// Guards should take a VM exit to 'pcpos'
|
||||||
|
HostUserdataMetamethodHandler userdataMetamethod = nullptr;
|
||||||
|
|
||||||
|
// Handle namecall performed on a userdata value
|
||||||
|
// 'sourceReg' (self argument) is guaranteed to be a userdata, but tag has to be checked
|
||||||
|
// All other arguments can be of any type
|
||||||
|
// Guards should take a VM exit to 'pcpos'
|
||||||
|
HostUserdataNamecallHandler userdataNamecall = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct CompilationOptions
|
||||||
|
{
|
||||||
|
unsigned int flags = 0;
|
||||||
|
HostIrHooks hooks;
|
||||||
|
|
||||||
|
// null-terminated array of userdata types names that might have custom lowering
|
||||||
|
const char* const* userdataTypes = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
using AnnotatorFn = void (*)(void* context, std::string& result, int fid, int instpos);
|
||||||
|
|
||||||
|
// Output "#" before IR blocks and instructions
|
||||||
|
enum class IncludeIrPrefix
|
||||||
|
{
|
||||||
|
No,
|
||||||
|
Yes
|
||||||
|
};
|
||||||
|
|
||||||
|
// Output user count and last use information of blocks and instructions
|
||||||
|
enum class IncludeUseInfo
|
||||||
|
{
|
||||||
|
No,
|
||||||
|
Yes
|
||||||
|
};
|
||||||
|
|
||||||
|
// Output CFG informations like block predecessors, successors and etc
|
||||||
|
enum class IncludeCfgInfo
|
||||||
|
{
|
||||||
|
No,
|
||||||
|
Yes
|
||||||
|
};
|
||||||
|
|
||||||
|
// Output VM register live in/out information for blocks
|
||||||
|
enum class IncludeRegFlowInfo
|
||||||
|
{
|
||||||
|
No,
|
||||||
|
Yes
|
||||||
|
};
|
||||||
|
|
||||||
|
struct AssemblyOptions
|
||||||
|
{
|
||||||
|
enum Target
|
||||||
|
{
|
||||||
|
Host,
|
||||||
|
A64,
|
||||||
|
A64_NoFeatures,
|
||||||
|
X64_Windows,
|
||||||
|
X64_SystemV,
|
||||||
|
};
|
||||||
|
|
||||||
|
Target target = Host;
|
||||||
|
|
||||||
|
CompilationOptions compilationOptions;
|
||||||
|
|
||||||
|
bool outputBinary = false;
|
||||||
|
|
||||||
|
bool includeAssembly = false;
|
||||||
|
bool includeIr = false;
|
||||||
|
bool includeOutlinedCode = false;
|
||||||
|
bool includeIrTypes = false;
|
||||||
|
|
||||||
|
IncludeIrPrefix includeIrPrefix = IncludeIrPrefix::Yes;
|
||||||
|
IncludeUseInfo includeUseInfo = IncludeUseInfo::Yes;
|
||||||
|
IncludeCfgInfo includeCfgInfo = IncludeCfgInfo::Yes;
|
||||||
|
IncludeRegFlowInfo includeRegFlowInfo = IncludeRegFlowInfo::Yes;
|
||||||
|
|
||||||
|
// Optional annotator function can be provided to describe each instruction, it takes function id and sequential instruction id
|
||||||
|
AnnotatorFn annotator = nullptr;
|
||||||
|
void* annotatorContext = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace CodeGen
|
||||||
|
} // namespace Luau
|
|
@ -20,6 +20,8 @@ namespace Luau
|
||||||
namespace CodeGen
|
namespace CodeGen
|
||||||
{
|
{
|
||||||
|
|
||||||
|
struct LoweringStats;
|
||||||
|
|
||||||
// IR extensions to LuauBuiltinFunction enum (these only exist inside IR, and start from 256 to avoid collisions)
|
// IR extensions to LuauBuiltinFunction enum (these only exist inside IR, and start from 256 to avoid collisions)
|
||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
|
@ -67,18 +69,18 @@ enum class IrCmd : uint8_t
|
||||||
LOAD_ENV,
|
LOAD_ENV,
|
||||||
|
|
||||||
// Get pointer (TValue) to table array at index
|
// Get pointer (TValue) to table array at index
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: int
|
// B: int
|
||||||
GET_ARR_ADDR,
|
GET_ARR_ADDR,
|
||||||
|
|
||||||
// Get pointer (LuaNode) to table node element at the active cached slot index
|
// Get pointer (LuaNode) to table node element at the active cached slot index
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: unsigned int (pcpos)
|
// B: unsigned int (pcpos)
|
||||||
// C: Kn
|
// C: Kn
|
||||||
GET_SLOT_NODE_ADDR,
|
GET_SLOT_NODE_ADDR,
|
||||||
|
|
||||||
// Get pointer (LuaNode) to table node element at the main position of the specified key hash
|
// Get pointer (LuaNode) to table node element at the main position of the specified key hash
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: unsigned int (hash)
|
// B: unsigned int (hash)
|
||||||
GET_HASH_NODE_ADDR,
|
GET_HASH_NODE_ADDR,
|
||||||
|
|
||||||
|
@ -185,6 +187,11 @@ enum class IrCmd : uint8_t
|
||||||
// A: double
|
// A: double
|
||||||
SIGN_NUM,
|
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
|
// Add/Sub/Mul/Div/Idiv two vectors
|
||||||
// A, B: TValue
|
// A, B: TValue
|
||||||
ADD_VEC,
|
ADD_VEC,
|
||||||
|
@ -268,7 +275,7 @@ enum class IrCmd : uint8_t
|
||||||
JUMP_SLOT_MATCH,
|
JUMP_SLOT_MATCH,
|
||||||
|
|
||||||
// Get table length
|
// Get table length
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
TABLE_LEN,
|
TABLE_LEN,
|
||||||
|
|
||||||
// Get string length
|
// Get string length
|
||||||
|
@ -281,11 +288,11 @@ enum class IrCmd : uint8_t
|
||||||
NEW_TABLE,
|
NEW_TABLE,
|
||||||
|
|
||||||
// Duplicate a table
|
// Duplicate a table
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
DUP_TABLE,
|
DUP_TABLE,
|
||||||
|
|
||||||
// Insert an integer key into a table and return the pointer to inserted value (TValue)
|
// Insert an integer key into a table and return the pointer to inserted value (TValue)
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: int (key)
|
// B: int (key)
|
||||||
TABLE_SETNUM,
|
TABLE_SETNUM,
|
||||||
|
|
||||||
|
@ -425,13 +432,13 @@ enum class IrCmd : uint8_t
|
||||||
CHECK_TRUTHY,
|
CHECK_TRUTHY,
|
||||||
|
|
||||||
// Guard against readonly table
|
// Guard against readonly table
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: block/vmexit/undef
|
// B: block/vmexit/undef
|
||||||
// When undef is specified instead of a block, execution is aborted on check failure
|
// When undef is specified instead of a block, execution is aborted on check failure
|
||||||
CHECK_READONLY,
|
CHECK_READONLY,
|
||||||
|
|
||||||
// Guard against table having a metatable
|
// Guard against table having a metatable
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: block/vmexit/undef
|
// B: block/vmexit/undef
|
||||||
// When undef is specified instead of a block, execution is aborted on check failure
|
// When undef is specified instead of a block, execution is aborted on check failure
|
||||||
CHECK_NO_METATABLE,
|
CHECK_NO_METATABLE,
|
||||||
|
@ -442,7 +449,7 @@ enum class IrCmd : uint8_t
|
||||||
CHECK_SAFE_ENV,
|
CHECK_SAFE_ENV,
|
||||||
|
|
||||||
// Guard against index overflowing the table array size
|
// Guard against index overflowing the table array size
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: int (index)
|
// B: int (index)
|
||||||
// C: block/vmexit/undef
|
// C: block/vmexit/undef
|
||||||
// When undef is specified instead of a block, execution is aborted on check failure
|
// When undef is specified instead of a block, execution is aborted on check failure
|
||||||
|
@ -498,11 +505,11 @@ enum class IrCmd : uint8_t
|
||||||
BARRIER_OBJ,
|
BARRIER_OBJ,
|
||||||
|
|
||||||
// Handle GC write barrier (backwards) for a write into a table
|
// Handle GC write barrier (backwards) for a write into a table
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
BARRIER_TABLE_BACK,
|
BARRIER_TABLE_BACK,
|
||||||
|
|
||||||
// Handle GC write barrier (forward) for a write into a table
|
// Handle GC write barrier (forward) for a write into a table
|
||||||
// A: pointer (Table)
|
// A: pointer (LuaTable)
|
||||||
// B: Rn (TValue that was written to the object)
|
// B: Rn (TValue that was written to the object)
|
||||||
// C: tag/undef (tag of the value that was written)
|
// C: tag/undef (tag of the value that was written)
|
||||||
BARRIER_TABLE_FORWARD,
|
BARRIER_TABLE_FORWARD,
|
||||||
|
@ -1044,6 +1051,8 @@ struct IrFunction
|
||||||
|
|
||||||
CfgInfo cfg;
|
CfgInfo cfg;
|
||||||
|
|
||||||
|
LoweringStats* stats = nullptr;
|
||||||
|
|
||||||
IrBlock& blockOp(IrOp op)
|
IrBlock& blockOp(IrOp op)
|
||||||
{
|
{
|
||||||
CODEGEN_ASSERT(op.kind == IrOpKind::Block);
|
CODEGEN_ASSERT(op.kind == IrOpKind::Block);
|
||||||
|
|
|
@ -2,11 +2,13 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "Luau/IrData.h"
|
#include "Luau/IrData.h"
|
||||||
#include "Luau/CodeGen.h"
|
#include "Luau/CodeGenOptions.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
struct Proto;
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
namespace CodeGen
|
namespace CodeGen
|
||||||
|
@ -23,6 +25,7 @@ struct IrToStringContext
|
||||||
const std::vector<IrBlock>& blocks;
|
const std::vector<IrBlock>& blocks;
|
||||||
const std::vector<IrConst>& constants;
|
const std::vector<IrConst>& constants;
|
||||||
const CfgInfo& cfg;
|
const CfgInfo& cfg;
|
||||||
|
Proto* proto = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
void toString(IrToStringContext& ctx, const IrInst& inst, uint32_t index);
|
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::SQRT_NUM:
|
||||||
case IrCmd::ABS_NUM:
|
case IrCmd::ABS_NUM:
|
||||||
case IrCmd::SIGN_NUM:
|
case IrCmd::SIGN_NUM:
|
||||||
|
case IrCmd::SELECT_NUM:
|
||||||
case IrCmd::ADD_VEC:
|
case IrCmd::ADD_VEC:
|
||||||
case IrCmd::SUB_VEC:
|
case IrCmd::SUB_VEC:
|
||||||
case IrCmd::MUL_VEC:
|
case IrCmd::MUL_VEC:
|
||||||
|
|
103
CodeGen/include/Luau/LoweringStats.h
Normal file
103
CodeGen/include/Luau/LoweringStats.h
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
namespace Luau
|
||||||
|
{
|
||||||
|
namespace CodeGen
|
||||||
|
{
|
||||||
|
|
||||||
|
struct BlockLinearizationStats
|
||||||
|
{
|
||||||
|
unsigned int constPropInstructionCount = 0;
|
||||||
|
double timeSeconds = 0.0;
|
||||||
|
|
||||||
|
BlockLinearizationStats& operator+=(const BlockLinearizationStats& that)
|
||||||
|
{
|
||||||
|
this->constPropInstructionCount += that.constPropInstructionCount;
|
||||||
|
this->timeSeconds += that.timeSeconds;
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
BlockLinearizationStats operator+(const BlockLinearizationStats& other) const
|
||||||
|
{
|
||||||
|
BlockLinearizationStats result(*this);
|
||||||
|
result += other;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
enum FunctionStatsFlags
|
||||||
|
{
|
||||||
|
// Enable stats collection per function
|
||||||
|
FunctionStats_Enable = 1 << 0,
|
||||||
|
// Compute function bytecode summary
|
||||||
|
FunctionStats_BytecodeSummary = 1 << 1,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct FunctionStats
|
||||||
|
{
|
||||||
|
std::string name;
|
||||||
|
int line = -1;
|
||||||
|
unsigned bcodeCount = 0;
|
||||||
|
unsigned irCount = 0;
|
||||||
|
unsigned asmCount = 0;
|
||||||
|
unsigned asmSize = 0;
|
||||||
|
std::vector<std::vector<unsigned>> bytecodeSummary;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct LoweringStats
|
||||||
|
{
|
||||||
|
unsigned totalFunctions = 0;
|
||||||
|
unsigned skippedFunctions = 0;
|
||||||
|
int spillsToSlot = 0;
|
||||||
|
int spillsToRestore = 0;
|
||||||
|
unsigned maxSpillSlotsUsed = 0;
|
||||||
|
unsigned blocksPreOpt = 0;
|
||||||
|
unsigned blocksPostOpt = 0;
|
||||||
|
unsigned maxBlockInstructions = 0;
|
||||||
|
|
||||||
|
int regAllocErrors = 0;
|
||||||
|
int loweringErrors = 0;
|
||||||
|
|
||||||
|
BlockLinearizationStats blockLinearizationStats;
|
||||||
|
|
||||||
|
unsigned functionStatsFlags = 0;
|
||||||
|
std::vector<FunctionStats> functions;
|
||||||
|
|
||||||
|
LoweringStats operator+(const LoweringStats& other) const
|
||||||
|
{
|
||||||
|
LoweringStats result(*this);
|
||||||
|
result += other;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
LoweringStats& operator+=(const LoweringStats& that)
|
||||||
|
{
|
||||||
|
this->totalFunctions += that.totalFunctions;
|
||||||
|
this->skippedFunctions += that.skippedFunctions;
|
||||||
|
this->spillsToSlot += that.spillsToSlot;
|
||||||
|
this->spillsToRestore += that.spillsToRestore;
|
||||||
|
this->maxSpillSlotsUsed = std::max(this->maxSpillSlotsUsed, that.maxSpillSlotsUsed);
|
||||||
|
this->blocksPreOpt += that.blocksPreOpt;
|
||||||
|
this->blocksPostOpt += that.blocksPostOpt;
|
||||||
|
this->maxBlockInstructions = std::max(this->maxBlockInstructions, that.maxBlockInstructions);
|
||||||
|
|
||||||
|
this->regAllocErrors += that.regAllocErrors;
|
||||||
|
this->loweringErrors += that.loweringErrors;
|
||||||
|
|
||||||
|
this->blockLinearizationStats += that.blockLinearizationStats;
|
||||||
|
|
||||||
|
if (this->functionStatsFlags & FunctionStats_Enable)
|
||||||
|
this->functions.insert(this->functions.end(), that.functions.begin(), that.functions.end());
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace CodeGen
|
||||||
|
} // namespace Luau
|
|
@ -927,6 +927,11 @@ void AssemblyBuilderX64::vminsd(OperandX64 dst, OperandX64 src1, OperandX64 src2
|
||||||
placeAvx("vminsd", dst, src1, src2, 0x5d, false, AVX_0F, AVX_F2);
|
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)
|
void AssemblyBuilderX64::vcmpltsd(OperandX64 dst, OperandX64 src1, OperandX64 src2)
|
||||||
{
|
{
|
||||||
placeAvx("vcmpltsd", dst, src1, src2, 0x01, 0xc2, false, AVX_0F, AVX_F2);
|
placeAvx("vcmpltsd", dst, src1, src2, 0x01, 0xc2, false, AVX_0F, AVX_F2);
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue