mirror of
https://github.com/luau-lang/luau.git
synced 2025-04-03 18:30:54 +01:00
Merge remote-tracking branch 'origin/master' into patch-2
This commit is contained in:
commit
ceed23d2b9
338 changed files with 13549 additions and 3674 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
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -53,7 +53,7 @@ LUAU_EQSAT_NODE_SET(Intersection);
|
||||||
|
|
||||||
LUAU_EQSAT_NODE_ARRAY(Negation, 1);
|
LUAU_EQSAT_NODE_ARRAY(Negation, 1);
|
||||||
|
|
||||||
LUAU_EQSAT_NODE_ATOM_WITH_VECTOR(TTypeFun, const TypeFunction*);
|
LUAU_EQSAT_NODE_ATOM_WITH_VECTOR(TTypeFun, std::shared_ptr<const TypeFunctionInstanceType>);
|
||||||
|
|
||||||
LUAU_EQSAT_UNIT(TNoRefine);
|
LUAU_EQSAT_UNIT(TNoRefine);
|
||||||
LUAU_EQSAT_UNIT(Invalid);
|
LUAU_EQSAT_UNIT(Invalid);
|
||||||
|
@ -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);
|
||||||
|
|
||||||
|
@ -218,6 +225,7 @@ struct Simplifier
|
||||||
void simplifyUnion(Id id);
|
void simplifyUnion(Id id);
|
||||||
void uninhabitedIntersection(Id id);
|
void uninhabitedIntersection(Id id);
|
||||||
void intersectWithNegatedClass(Id id);
|
void intersectWithNegatedClass(Id id);
|
||||||
|
void intersectWithNegatedAtom(Id id);
|
||||||
void intersectWithNoRefine(Id id);
|
void intersectWithNoRefine(Id id);
|
||||||
void cyclicIntersectionOfUnion(Id id);
|
void cyclicIntersectionOfUnion(Id id);
|
||||||
void cyclicUnionOfIntersection(Id id);
|
void cyclicUnionOfIntersection(Id id);
|
||||||
|
@ -228,6 +236,7 @@ struct Simplifier
|
||||||
void unneededTableModification(Id id);
|
void unneededTableModification(Id id);
|
||||||
void builtinTypeFunctions(Id id);
|
void builtinTypeFunctions(Id id);
|
||||||
void iffyTypeFunctions(Id id);
|
void iffyTypeFunctions(Id id);
|
||||||
|
void strictMetamethods(Id id);
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename Tag>
|
template<typename Tag>
|
||||||
|
@ -293,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;
|
||||||
|
@ -329,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};
|
||||||
|
@ -341,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,6 +16,8 @@
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
|
||||||
|
LUAU_FASTFLAG(LuauIncrementalAutocompleteCommentDetection)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -55,6 +57,7 @@ struct SourceModule
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
bool isWithinComment(const std::vector<Comment>& commentLocations, Position pos);
|
||||||
bool isWithinComment(const SourceModule& sourceModule, Position pos);
|
bool isWithinComment(const SourceModule& sourceModule, Position pos);
|
||||||
bool isWithinComment(const ParseResult& result, Position pos);
|
bool isWithinComment(const ParseResult& result, Position pos);
|
||||||
|
|
||||||
|
@ -136,6 +139,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;
|
||||||
|
|
||||||
|
|
|
@ -15,6 +15,7 @@ struct TypeCheckLimits;
|
||||||
|
|
||||||
void checkNonStrict(
|
void checkNonStrict(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> ice,
|
NotNull<InternalErrorReporter> ice,
|
||||||
NotNull<UnifierSharedState> unifierState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
|
|
|
@ -1,6 +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/EqSatSimplification.h"
|
||||||
#include "Luau/NotNull.h"
|
#include "Luau/NotNull.h"
|
||||||
#include "Luau/Set.h"
|
#include "Luau/Set.h"
|
||||||
#include "Luau/TypeFwd.h"
|
#include "Luau/TypeFwd.h"
|
||||||
|
@ -21,8 +22,22 @@ struct Scope;
|
||||||
|
|
||||||
using ModulePtr = std::shared_ptr<Module>;
|
using ModulePtr = std::shared_ptr<Module>;
|
||||||
|
|
||||||
bool isSubtype(TypeId subTy, TypeId superTy, NotNull<Scope> scope, NotNull<BuiltinTypes> builtinTypes, InternalErrorReporter& ice);
|
bool isSubtype(
|
||||||
bool isSubtype(TypePackId subTy, TypePackId superTy, NotNull<Scope> scope, NotNull<BuiltinTypes> builtinTypes, InternalErrorReporter& ice);
|
TypeId subTy,
|
||||||
|
TypeId superTy,
|
||||||
|
NotNull<Scope> scope,
|
||||||
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
|
InternalErrorReporter& ice
|
||||||
|
);
|
||||||
|
bool isSubtype(
|
||||||
|
TypePackId subPack,
|
||||||
|
TypePackId superPack,
|
||||||
|
NotNull<Scope> scope,
|
||||||
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
|
InternalErrorReporter& ice
|
||||||
|
);
|
||||||
|
|
||||||
class TypeIds
|
class TypeIds
|
||||||
{
|
{
|
||||||
|
|
|
@ -2,12 +2,13 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "Luau/Ast.h"
|
#include "Luau/Ast.h"
|
||||||
#include "Luau/InsertionOrderedMap.h"
|
#include "Luau/EqSatSimplification.h"
|
||||||
#include "Luau/NotNull.h"
|
|
||||||
#include "Luau/TypeFwd.h"
|
|
||||||
#include "Luau/Location.h"
|
|
||||||
#include "Luau/Error.h"
|
#include "Luau/Error.h"
|
||||||
|
#include "Luau/InsertionOrderedMap.h"
|
||||||
|
#include "Luau/Location.h"
|
||||||
|
#include "Luau/NotNull.h"
|
||||||
#include "Luau/Subtyping.h"
|
#include "Luau/Subtyping.h"
|
||||||
|
#include "Luau/TypeFwd.h"
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -34,6 +35,7 @@ struct OverloadResolver
|
||||||
OverloadResolver(
|
OverloadResolver(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<Scope> scope,
|
NotNull<Scope> scope,
|
||||||
|
@ -44,6 +46,7 @@ struct OverloadResolver
|
||||||
|
|
||||||
NotNull<BuiltinTypes> builtinTypes;
|
NotNull<BuiltinTypes> builtinTypes;
|
||||||
NotNull<TypeArena> arena;
|
NotNull<TypeArena> arena;
|
||||||
|
NotNull<Simplifier> simplifier;
|
||||||
NotNull<Normalizer> normalizer;
|
NotNull<Normalizer> normalizer;
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
||||||
NotNull<Scope> scope;
|
NotNull<Scope> scope;
|
||||||
|
@ -110,6 +113,7 @@ struct SolveResult
|
||||||
SolveResult solveFunctionCall(
|
SolveResult solveFunctionCall(
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> iceReporter,
|
NotNull<InternalErrorReporter> iceReporter,
|
||||||
|
|
|
@ -85,12 +85,18 @@ struct Scope
|
||||||
void inheritAssignments(const ScopePtr& childScope);
|
void inheritAssignments(const ScopePtr& childScope);
|
||||||
void inheritRefinements(const ScopePtr& childScope);
|
void inheritRefinements(const ScopePtr& childScope);
|
||||||
|
|
||||||
|
// Track globals that should emit warnings during type checking.
|
||||||
|
DenseHashSet<std::string> globalsToWarn{""};
|
||||||
|
bool shouldWarnGlobal(std::string name) const;
|
||||||
|
|
||||||
// For mutually recursive type aliases, it's important that
|
// For mutually recursive type aliases, it's important that
|
||||||
// they use the same types for the same names.
|
// they use the same types for the same names.
|
||||||
// For instance, in `type Tree<T> { data: T, children: Forest<T> } type Forest<T> = {Tree<T>}`
|
// For instance, in `type Tree<T> { data: T, children: Forest<T> } type Forest<T> = {Tree<T>}`
|
||||||
// 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
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,13 +1,14 @@
|
||||||
// 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/DenseHash.h"
|
||||||
|
#include "Luau/EqSatSimplification.h"
|
||||||
#include "Luau/Set.h"
|
#include "Luau/Set.h"
|
||||||
|
#include "Luau/TypeCheckLimits.h"
|
||||||
|
#include "Luau/TypeFunction.h"
|
||||||
#include "Luau/TypeFwd.h"
|
#include "Luau/TypeFwd.h"
|
||||||
#include "Luau/TypePairHash.h"
|
#include "Luau/TypePairHash.h"
|
||||||
#include "Luau/TypePath.h"
|
#include "Luau/TypePath.h"
|
||||||
#include "Luau/TypeFunction.h"
|
|
||||||
#include "Luau/TypeCheckLimits.h"
|
|
||||||
#include "Luau/DenseHash.h"
|
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
@ -134,6 +135,7 @@ struct Subtyping
|
||||||
{
|
{
|
||||||
NotNull<BuiltinTypes> builtinTypes;
|
NotNull<BuiltinTypes> builtinTypes;
|
||||||
NotNull<TypeArena> arena;
|
NotNull<TypeArena> arena;
|
||||||
|
NotNull<Simplifier> simplifier;
|
||||||
NotNull<Normalizer> normalizer;
|
NotNull<Normalizer> normalizer;
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
||||||
NotNull<InternalErrorReporter> iceReporter;
|
NotNull<InternalErrorReporter> iceReporter;
|
||||||
|
@ -155,6 +157,7 @@ struct Subtyping
|
||||||
Subtyping(
|
Subtyping(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<TypeArena> typeArena,
|
NotNull<TypeArena> typeArena,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> iceReporter
|
NotNull<InternalErrorReporter> iceReporter
|
||||||
|
|
|
@ -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.
|
||||||
|
@ -608,7 +620,8 @@ struct UserDefinedFunctionData
|
||||||
// References to AST elements are owned by the Module allocator which also stores this type
|
// References to AST elements are owned by the Module allocator which also stores this type
|
||||||
AstStatTypeFunction* definition = nullptr;
|
AstStatTypeFunction* definition = nullptr;
|
||||||
|
|
||||||
DenseHashMap<Name, AstStatTypeFunction*> environment{""};
|
DenseHashMap<Name, std::pair<AstStatTypeFunction*, size_t>> environment{""};
|
||||||
|
DenseHashMap<Name, AstStatTypeFunction*> environment_DEPRECATED{""};
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
@ -2,15 +2,16 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "Luau/Error.h"
|
|
||||||
#include "Luau/NotNull.h"
|
|
||||||
#include "Luau/Common.h"
|
#include "Luau/Common.h"
|
||||||
#include "Luau/TypeUtils.h"
|
#include "Luau/EqSatSimplification.h"
|
||||||
|
#include "Luau/Error.h"
|
||||||
|
#include "Luau/Normalize.h"
|
||||||
|
#include "Luau/NotNull.h"
|
||||||
|
#include "Luau/Subtyping.h"
|
||||||
#include "Luau/Type.h"
|
#include "Luau/Type.h"
|
||||||
#include "Luau/TypeFwd.h"
|
#include "Luau/TypeFwd.h"
|
||||||
#include "Luau/TypeOrPack.h"
|
#include "Luau/TypeOrPack.h"
|
||||||
#include "Luau/Normalize.h"
|
#include "Luau/TypeUtils.h"
|
||||||
#include "Luau/Subtyping.h"
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -60,8 +61,9 @@ struct Reasonings
|
||||||
|
|
||||||
void check(
|
void check(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
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,
|
||||||
|
@ -71,6 +73,7 @@ void check(
|
||||||
struct TypeChecker2
|
struct TypeChecker2
|
||||||
{
|
{
|
||||||
NotNull<BuiltinTypes> builtinTypes;
|
NotNull<BuiltinTypes> builtinTypes;
|
||||||
|
NotNull<Simplifier> simplifier;
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
||||||
DcrLogger* logger;
|
DcrLogger* logger;
|
||||||
const NotNull<TypeCheckLimits> limits;
|
const NotNull<TypeCheckLimits> limits;
|
||||||
|
@ -90,6 +93,7 @@ struct TypeChecker2
|
||||||
|
|
||||||
TypeChecker2(
|
TypeChecker2(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<UnifierSharedState> unifierState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
NotNull<TypeCheckLimits> limits,
|
NotNull<TypeCheckLimits> limits,
|
||||||
|
@ -112,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);
|
||||||
|
@ -156,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);
|
||||||
|
@ -213,6 +217,9 @@ private:
|
||||||
std::vector<TypeError>& errors
|
std::vector<TypeError>& errors
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Avoid duplicate warnings being emitted for the same global variable.
|
||||||
|
DenseHashSet<std::string> warnedGlobals{""};
|
||||||
|
|
||||||
void diagnoseMissingTableKey(UnknownProperty* utk, TypeErrorData& data) const;
|
void diagnoseMissingTableKey(UnknownProperty* utk, TypeErrorData& data) const;
|
||||||
bool isErrorSuppressing(Location loc, TypeId ty);
|
bool isErrorSuppressing(Location loc, TypeId ty);
|
||||||
bool isErrorSuppressing(Location loc1, TypeId ty1, Location loc2, TypeId ty2);
|
bool isErrorSuppressing(Location loc1, TypeId ty1, Location loc2, TypeId ty2);
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "Luau/Constraint.h"
|
#include "Luau/Constraint.h"
|
||||||
|
#include "Luau/EqSatSimplification.h"
|
||||||
#include "Luau/Error.h"
|
#include "Luau/Error.h"
|
||||||
#include "Luau/NotNull.h"
|
#include "Luau/NotNull.h"
|
||||||
#include "Luau/TypeCheckLimits.h"
|
#include "Luau/TypeCheckLimits.h"
|
||||||
|
@ -41,9 +42,15 @@ struct TypeFunctionRuntime
|
||||||
|
|
||||||
StateRef state;
|
StateRef state;
|
||||||
|
|
||||||
|
// Set of functions which have their environment table initialized
|
||||||
|
DenseHashSet<AstStatTypeFunction*> initialized{nullptr};
|
||||||
|
|
||||||
// Evaluation of type functions should only be performed in the absence of parse errors in the source module
|
// Evaluation of type functions should only be performed in the absence of parse errors in the source module
|
||||||
bool allowEvaluation = true;
|
bool allowEvaluation = true;
|
||||||
|
|
||||||
|
// Output created by 'print' function
|
||||||
|
std::vector<std::string> messages;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void prepareState();
|
void prepareState();
|
||||||
};
|
};
|
||||||
|
@ -53,6 +60,7 @@ struct TypeFunctionContext
|
||||||
NotNull<TypeArena> arena;
|
NotNull<TypeArena> arena;
|
||||||
NotNull<BuiltinTypes> builtins;
|
NotNull<BuiltinTypes> builtins;
|
||||||
NotNull<Scope> scope;
|
NotNull<Scope> scope;
|
||||||
|
NotNull<Simplifier> simplifier;
|
||||||
NotNull<Normalizer> normalizer;
|
NotNull<Normalizer> normalizer;
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
||||||
NotNull<InternalErrorReporter> ice;
|
NotNull<InternalErrorReporter> ice;
|
||||||
|
@ -71,6 +79,7 @@ struct TypeFunctionContext
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
NotNull<BuiltinTypes> builtins,
|
NotNull<BuiltinTypes> builtins,
|
||||||
NotNull<Scope> scope,
|
NotNull<Scope> scope,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> ice,
|
NotNull<InternalErrorReporter> ice,
|
||||||
|
@ -79,6 +88,7 @@ struct TypeFunctionContext
|
||||||
: arena(arena)
|
: arena(arena)
|
||||||
, builtins(builtins)
|
, builtins(builtins)
|
||||||
, scope(scope)
|
, scope(scope)
|
||||||
|
, simplifier(simplifier)
|
||||||
, normalizer(normalizer)
|
, normalizer(normalizer)
|
||||||
, typeFunctionRuntime(typeFunctionRuntime)
|
, typeFunctionRuntime(typeFunctionRuntime)
|
||||||
, ice(ice)
|
, ice(ice)
|
||||||
|
@ -91,19 +101,31 @@ struct TypeFunctionContext
|
||||||
NotNull<Constraint> pushConstraint(ConstraintV&& c) const;
|
NotNull<Constraint> pushConstraint(ConstraintV&& c) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum class Reduction
|
||||||
|
{
|
||||||
|
// The type function is either known to be reducible or the determination is blocked.
|
||||||
|
MaybeOk,
|
||||||
|
// The type function is known to be irreducible, but maybe not be erroneous, e.g. when it's over generics or free types.
|
||||||
|
Irreducible,
|
||||||
|
// The type function is known to be irreducible, and is definitely erroneous.
|
||||||
|
Erroneous,
|
||||||
|
};
|
||||||
|
|
||||||
/// Represents a reduction result, which may have successfully reduced the type,
|
/// Represents a reduction result, which may have successfully reduced the type,
|
||||||
/// may have concretely failed to reduce the type, or may simply be stuck
|
/// may have concretely failed to reduce the type, or may simply be stuck
|
||||||
/// without more information.
|
/// without more information.
|
||||||
template<typename Ty>
|
template<typename Ty>
|
||||||
struct TypeFunctionReductionResult
|
struct TypeFunctionReductionResult
|
||||||
{
|
{
|
||||||
|
|
||||||
/// The result of the reduction, if any. If this is nullopt, the type function
|
/// The result of the reduction, if any. If this is nullopt, the type function
|
||||||
/// could not be reduced.
|
/// could not be reduced.
|
||||||
std::optional<Ty> result;
|
std::optional<Ty> result;
|
||||||
/// Whether the result is uninhabited: whether we know, unambiguously and
|
/// Indicates the status of this reduction: is `Reduction::Irreducible` if
|
||||||
/// permanently, whether this type function reduction results in an
|
/// the this result indicates the type function is irreducible, and
|
||||||
/// uninhabitable type. This will trigger an error to be reported.
|
/// `Reduction::Erroneous` if this result indicates the type function is
|
||||||
bool uninhabited;
|
/// erroneous. `Reduction::MaybeOk` otherwise.
|
||||||
|
Reduction reductionStatus;
|
||||||
/// Any types that need to be progressed or mutated before the reduction may
|
/// Any types that need to be progressed or mutated before the reduction may
|
||||||
/// proceed.
|
/// proceed.
|
||||||
std::vector<TypeId> blockedTypes;
|
std::vector<TypeId> blockedTypes;
|
||||||
|
@ -112,6 +134,8 @@ struct TypeFunctionReductionResult
|
||||||
std::vector<TypePackId> blockedPacks;
|
std::vector<TypePackId> blockedPacks;
|
||||||
/// A runtime error message from user-defined type functions
|
/// A runtime error message from user-defined type functions
|
||||||
std::optional<std::string> error;
|
std::optional<std::string> error;
|
||||||
|
/// Messages printed out from user-defined type functions
|
||||||
|
std::vector<std::string> messages;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename T>
|
template<typename T>
|
||||||
|
@ -145,6 +169,7 @@ struct TypePackFunction
|
||||||
struct FunctionGraphReductionResult
|
struct FunctionGraphReductionResult
|
||||||
{
|
{
|
||||||
ErrorVec errors;
|
ErrorVec errors;
|
||||||
|
ErrorVec messages;
|
||||||
DenseHashSet<TypeId> blockedTypes{nullptr};
|
DenseHashSet<TypeId> blockedTypes{nullptr};
|
||||||
DenseHashSet<TypePackId> blockedPacks{nullptr};
|
DenseHashSet<TypePackId> blockedPacks{nullptr};
|
||||||
DenseHashSet<TypeId> reducedTypes{nullptr};
|
DenseHashSet<TypeId> reducedTypes{nullptr};
|
||||||
|
@ -216,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
|
||||||
{
|
{
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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"};
|
||||||
|
@ -150,6 +151,7 @@ static bool checkTypeMatch(TypeId subTy, TypeId superTy, NotNull<Scope> scope, T
|
||||||
{
|
{
|
||||||
InternalErrorReporter iceReporter;
|
InternalErrorReporter iceReporter;
|
||||||
UnifierSharedState unifierState(&iceReporter);
|
UnifierSharedState unifierState(&iceReporter);
|
||||||
|
SimplifierPtr simplifier = newSimplifier(NotNull{typeArena}, builtinTypes);
|
||||||
Normalizer normalizer{typeArena, builtinTypes, NotNull{&unifierState}};
|
Normalizer normalizer{typeArena, builtinTypes, NotNull{&unifierState}};
|
||||||
|
|
||||||
if (FFlag::LuauSolverV2)
|
if (FFlag::LuauSolverV2)
|
||||||
|
@ -162,7 +164,9 @@ static bool checkTypeMatch(TypeId subTy, TypeId superTy, NotNull<Scope> scope, T
|
||||||
unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit;
|
unifierState.counters.recursionLimit = FInt::LuauTypeInferRecursionLimit;
|
||||||
unifierState.counters.iterationLimit = FInt::LuauTypeInferIterationLimit;
|
unifierState.counters.iterationLimit = FInt::LuauTypeInferIterationLimit;
|
||||||
|
|
||||||
Subtyping subtyping{builtinTypes, NotNull{typeArena}, NotNull{&normalizer}, NotNull{&typeFunctionRuntime}, NotNull{&iceReporter}};
|
Subtyping subtyping{
|
||||||
|
builtinTypes, NotNull{typeArena}, NotNull{simplifier.get()}, NotNull{&normalizer}, NotNull{&typeFunctionRuntime}, NotNull{&iceReporter}
|
||||||
|
};
|
||||||
|
|
||||||
return subtyping.isSubtype(subTy, superTy, scope).isSubtype;
|
return subtyping.isSubtype(subTy, superTy, scope).isSubtype;
|
||||||
}
|
}
|
||||||
|
@ -174,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,81 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
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)
|
||||||
|
|
||||||
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 +198,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 +308,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 +328,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 +399,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 +415,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 +447,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 +501,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 +551,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 +595,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 +611,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,7 +629,25 @@ 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);
|
||||||
|
@ -637,6 +655,9 @@ static void dcrMagicFunctionTypeCheckFormat(MagicFunctionTypeCheckContext contex
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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)
|
||||||
{
|
{
|
||||||
std::vector<TypeId> result;
|
std::vector<TypeId> result;
|
||||||
|
@ -697,7 +718,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 +754,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 +787,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 +827,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 +863,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 +921,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 +998,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 +1014,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 +1029,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 +1090,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 +1135,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 +1180,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 +1262,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 +1301,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 +1349,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,7 +1389,74 @@ 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;
|
||||||
|
|
||||||
|
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;
|
TypeArena* arena = context.solver->arena;
|
||||||
|
|
||||||
|
@ -1383,7 +1476,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 +1501,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 +1565,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 +1611,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,13 +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_FASTFLAGVARIABLE(LuauNewSolverVisitErrorExprLvalues)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPrePopulateClasses)
|
LUAU_FASTFLAGVARIABLE(LuauNewSolverPrePopulateClasses)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunExportedAndLocal)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauNewSolverPopulateTableLocations)
|
LUAU_FASTFLAGVARIABLE(LuauNewSolverPopulateTableLocations)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(InferGlobalTypes)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -229,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,
|
||||||
|
@ -299,10 +309,20 @@ 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)
|
||||||
{
|
{
|
||||||
|
@ -720,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
|
||||||
|
@ -751,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}
|
||||||
|
@ -764,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);
|
||||||
|
@ -799,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)
|
||||||
{
|
{
|
||||||
|
@ -822,6 +831,7 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
if (mainTypeFun)
|
if (mainTypeFun)
|
||||||
{
|
{
|
||||||
UserDefinedFunctionData& userFuncData = mainTypeFun->userFuncData;
|
UserDefinedFunctionData& userFuncData = mainTypeFun->userFuncData;
|
||||||
|
size_t level = 0;
|
||||||
|
|
||||||
for (Scope* curr = scope.get(); curr; curr = curr->parent.get())
|
for (Scope* curr = scope.get(); curr; curr = curr->parent.get())
|
||||||
{
|
{
|
||||||
|
@ -831,7 +841,7 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (auto ty = get<TypeFunctionInstanceType>(tf.type); ty && ty->userFuncData.definition)
|
if (auto ty = get<TypeFunctionInstanceType>(tf.type); ty && ty->userFuncData.definition)
|
||||||
userFuncData.environment[name] = ty->userFuncData.definition;
|
userFuncData.environment[name] = std::make_pair(ty->userFuncData.definition, level);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto& [name, tf] : curr->exportedTypeBindings)
|
for (auto& [name, tf] : curr->exportedTypeBindings)
|
||||||
|
@ -840,8 +850,34 @@ void ConstraintGenerator::checkAliases(const ScopePtr& scope, AstStatBlock* bloc
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (auto ty = get<TypeFunctionInstanceType>(tf.type); ty && ty->userFuncData.definition)
|
if (auto ty = get<TypeFunctionInstanceType>(tf.type); ty && ty->userFuncData.definition)
|
||||||
userFuncData.environment[name] = ty->userFuncData.definition;
|
userFuncData.environment[name] = std::make_pair(ty->userFuncData.definition, level);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
level++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (mainTypeFun)
|
||||||
|
{
|
||||||
|
UserDefinedFunctionData& userFuncData = mainTypeFun->userFuncData;
|
||||||
|
|
||||||
|
for (Scope* curr = scope.get(); curr; curr = curr->parent.get())
|
||||||
|
{
|
||||||
|
for (auto& [name, tf] : curr->privateTypeBindings)
|
||||||
|
{
|
||||||
|
if (userFuncData.environment_DEPRECATED.find(name))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (auto ty = get<TypeFunctionInstanceType>(tf.type); ty && ty->userFuncData.definition)
|
||||||
|
userFuncData.environment_DEPRECATED[name] = ty->userFuncData.definition;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (auto& [name, tf] : curr->exportedTypeBindings)
|
||||||
|
{
|
||||||
|
if (userFuncData.environment_DEPRECATED.find(name))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (auto ty = get<TypeFunctionInstanceType>(tf.type); ty && ty->userFuncData.definition)
|
||||||
|
userFuncData.environment_DEPRECATED[name] = ty->userFuncData.definition;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1052,20 +1088,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)
|
||||||
|
@ -1571,20 +1597,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 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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2047,7 +2059,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{});
|
||||||
|
@ -2196,7 +2208,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);
|
||||||
|
@ -2210,7 +2223,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);
|
||||||
|
@ -2372,8 +2386,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();
|
||||||
|
|
||||||
|
@ -2720,8 +2743,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
|
||||||
|
@ -2731,7 +2752,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);
|
||||||
}
|
}
|
||||||
|
@ -2790,6 +2810,14 @@ 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
|
||||||
|
// initially created by ConstraintGenerator::prepopulateGlobalScope.
|
||||||
|
if (auto bt = get<BlockedType>(follow(*annotatedTy)); bt && !bt->getOwner())
|
||||||
|
emplaceType<BoundType>(asMutable(*annotatedTy), rhsType);
|
||||||
|
}
|
||||||
|
|
||||||
addConstraint(scope, global->location, SubtypeConstraint{rhsType, *annotatedTy});
|
addConstraint(scope, global->location, SubtypeConstraint{rhsType, *annotatedTy});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3182,9 +3210,8 @@ 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
|
// If the alias is not generic, we don't need to set up a blocked type and an instantiation constraint
|
||||||
// type and an instantiation constraint.
|
if (alias.has_value() && alias->typeParams.empty() && alias->typePackParams.empty() && !ref->hasParameterList)
|
||||||
if (alias.has_value() && alias->typeParams.empty() && alias->typePackParams.empty())
|
|
||||||
{
|
{
|
||||||
result = alias->type;
|
result = alias->type;
|
||||||
}
|
}
|
||||||
|
@ -3393,6 +3420,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)
|
||||||
{
|
{
|
||||||
|
@ -3403,6 +3436,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)
|
||||||
{
|
{
|
||||||
|
@ -3411,6 +3450,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)
|
||||||
|
@ -3694,6 +3737,26 @@ struct GlobalPrepopulator : AstVisitor
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool visit(AstStatAssign* assign) override
|
||||||
|
{
|
||||||
|
if (FFlag::InferGlobalTypes)
|
||||||
|
{
|
||||||
|
for (const Luau::AstExpr* expr : assign->vars)
|
||||||
|
{
|
||||||
|
if (const AstExprGlobal* g = expr->as<AstExprGlobal>())
|
||||||
|
{
|
||||||
|
if (!globalScope->lookup(g->name))
|
||||||
|
globalScope->globalsToWarn.insert(g->name.value);
|
||||||
|
|
||||||
|
TypeId bt = arena->addType(BlockedType{});
|
||||||
|
globalScope->bindings[g->name] = Binding{bt, g->location};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
bool visit(AstStatFunction* function) override
|
bool visit(AstStatFunction* function) override
|
||||||
{
|
{
|
||||||
if (AstExprGlobal* g = function->name->as<AstExprGlobal>())
|
if (AstExprGlobal* g = function->name->as<AstExprGlobal>())
|
||||||
|
@ -3877,19 +3940,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,10 +31,12 @@ 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(LuauTrackInteriorFreeTypesOnScope)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauAlwaysFillInFunctionCallDiscriminantTypes)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauTrackInteriorFreeTablesOnScope)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -72,7 +74,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;
|
||||||
}
|
}
|
||||||
|
@ -443,7 +445,7 @@ void ConstraintSolver::run()
|
||||||
if (success)
|
if (success)
|
||||||
{
|
{
|
||||||
unblock(c);
|
unblock(c);
|
||||||
unsolvedConstraints.erase(unsolvedConstraints.begin() + i);
|
unsolvedConstraints.erase(unsolvedConstraints.begin() + ptrdiff_t(i));
|
||||||
|
|
||||||
// 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())
|
||||||
|
@ -550,7 +552,7 @@ void ConstraintSolver::finalizeTypeFunctions()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ConstraintSolver::isDone()
|
bool ConstraintSolver::isDone() const
|
||||||
{
|
{
|
||||||
return unsolvedConstraints.empty();
|
return unsolvedConstraints.empty();
|
||||||
}
|
}
|
||||||
|
@ -723,8 +725,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;
|
||||||
}
|
}
|
||||||
|
@ -800,9 +814,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);
|
||||||
|
@ -939,14 +961,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 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
|
||||||
|
@ -959,6 +973,13 @@ bool ConstraintSolver::tryDispatch(const TypeAliasExpansionConstraint& c, NotNul
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 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;
|
||||||
|
}
|
||||||
|
|
||||||
auto [typeArguments, packArguments] = saturateArguments(arena, builtinTypes, *tf, petv->typeArguments, petv->packArguments);
|
auto [typeArguments, packArguments] = saturateArguments(arena, builtinTypes, *tf, petv->typeArguments, petv->packArguments);
|
||||||
|
|
||||||
bool sameTypes = std::equal(
|
bool sameTypes = std::equal(
|
||||||
|
@ -1122,6 +1143,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);
|
||||||
|
@ -1137,6 +1180,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1144,12 +1189,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1219,17 +1268,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)
|
||||||
|
@ -1242,27 +1298,16 @@ 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},
|
||||||
|
simplifier,
|
||||||
normalizer,
|
normalizer,
|
||||||
typeFunctionRuntime,
|
typeFunctionRuntime,
|
||||||
constraint->scope,
|
constraint->scope,
|
||||||
|
@ -1618,7 +1663,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.
|
||||||
|
@ -1650,7 +1695,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.
|
||||||
|
@ -1770,6 +1815,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);
|
||||||
|
|
||||||
|
@ -2048,6 +2097,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);
|
||||||
}
|
}
|
||||||
|
@ -2103,6 +2154,11 @@ bool ConstraintSolver::tryDispatch(const ReduceConstraint& c, NotNull<const Cons
|
||||||
|
|
||||||
if (force || reductionFinished)
|
if (force || reductionFinished)
|
||||||
{
|
{
|
||||||
|
for (auto& message : result.messages)
|
||||||
|
{
|
||||||
|
reportError(std::move(message));
|
||||||
|
}
|
||||||
|
|
||||||
// if we're completely dispatching this constraint, we want to record any uninhabited type functions to unblock.
|
// if we're completely dispatching this constraint, we want to record any uninhabited type functions to unblock.
|
||||||
for (auto error : result.errors)
|
for (auto error : result.errors)
|
||||||
{
|
{
|
||||||
|
@ -2178,6 +2234,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};
|
||||||
|
|
||||||
|
@ -2434,6 +2495,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:
|
||||||
|
@ -2539,10 +2602,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:
|
||||||
|
@ -2773,10 +2843,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");
|
||||||
}
|
}
|
||||||
|
|
|
@ -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,15 +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_FASTFLAGVARIABLE(LuauMathMapDefinition)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitions)
|
LUAU_FASTFLAG(LuauVector2Constructor)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauVectorDefinitionsExtra)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
// TODO: there has to be a better way, like splitting up per library
|
|
||||||
static const std::string kBuiltinDefinitionLuaSrcChecked_DEPRECATED = R"BUILTIN_SRC(
|
static const std::string kBuiltinDefinitionLuaSrcChecked_DEPRECATED = R"BUILTIN_SRC(
|
||||||
|
|
||||||
declare bit32: {
|
declare bit32: {
|
||||||
|
@ -30,227 +28,6 @@ declare bit32: {
|
||||||
byteswap: @checked (n: number) -> number,
|
byteswap: @checked (n: number) -> number,
|
||||||
}
|
}
|
||||||
|
|
||||||
declare math: {
|
|
||||||
frexp: @checked (n: number) -> (number, number),
|
|
||||||
ldexp: @checked (s: number, e: number) -> number,
|
|
||||||
fmod: @checked (x: number, y: number) -> number,
|
|
||||||
modf: @checked (n: number) -> (number, number),
|
|
||||||
pow: @checked (x: number, y: number) -> number,
|
|
||||||
exp: @checked (n: number) -> number,
|
|
||||||
|
|
||||||
ceil: @checked (n: number) -> number,
|
|
||||||
floor: @checked (n: number) -> number,
|
|
||||||
abs: @checked (n: number) -> number,
|
|
||||||
sqrt: @checked (n: number) -> number,
|
|
||||||
|
|
||||||
log: @checked (n: number, base: number?) -> number,
|
|
||||||
log10: @checked (n: number) -> number,
|
|
||||||
|
|
||||||
rad: @checked (n: number) -> number,
|
|
||||||
deg: @checked (n: number) -> number,
|
|
||||||
|
|
||||||
sin: @checked (n: number) -> number,
|
|
||||||
cos: @checked (n: number) -> number,
|
|
||||||
tan: @checked (n: number) -> number,
|
|
||||||
sinh: @checked (n: number) -> number,
|
|
||||||
cosh: @checked (n: number) -> number,
|
|
||||||
tanh: @checked (n: number) -> number,
|
|
||||||
atan: @checked (n: number) -> number,
|
|
||||||
acos: @checked (n: number) -> number,
|
|
||||||
asin: @checked (n: number) -> number,
|
|
||||||
atan2: @checked (y: number, x: number) -> number,
|
|
||||||
|
|
||||||
min: @checked (number, ...number) -> number,
|
|
||||||
max: @checked (number, ...number) -> number,
|
|
||||||
|
|
||||||
pi: number,
|
|
||||||
huge: number,
|
|
||||||
|
|
||||||
randomseed: @checked (seed: number) -> (),
|
|
||||||
random: @checked (number?, number?) -> number,
|
|
||||||
|
|
||||||
sign: @checked (n: number) -> number,
|
|
||||||
clamp: @checked (n: number, min: number, max: number) -> number,
|
|
||||||
noise: @checked (x: number, y: number?, z: number?) -> number,
|
|
||||||
round: @checked (n: number) -> number,
|
|
||||||
}
|
|
||||||
|
|
||||||
type DateTypeArg = {
|
|
||||||
year: number,
|
|
||||||
month: number,
|
|
||||||
day: number,
|
|
||||||
hour: number?,
|
|
||||||
min: number?,
|
|
||||||
sec: number?,
|
|
||||||
isdst: boolean?,
|
|
||||||
}
|
|
||||||
|
|
||||||
type DateTypeResult = {
|
|
||||||
year: number,
|
|
||||||
month: number,
|
|
||||||
wday: number,
|
|
||||||
yday: number,
|
|
||||||
day: number,
|
|
||||||
hour: number,
|
|
||||||
min: number,
|
|
||||||
sec: number,
|
|
||||||
isdst: boolean,
|
|
||||||
}
|
|
||||||
|
|
||||||
declare os: {
|
|
||||||
time: (time: DateTypeArg?) -> number,
|
|
||||||
date: ((formatString: "*t" | "!*t", time: number?) -> DateTypeResult) & ((formatString: string?, time: number?) -> string),
|
|
||||||
difftime: (t2: DateTypeResult | number, t1: DateTypeResult | number) -> number,
|
|
||||||
clock: () -> number,
|
|
||||||
}
|
|
||||||
|
|
||||||
@checked declare function require(target: any): any
|
|
||||||
|
|
||||||
@checked declare function getfenv(target: any): { [string]: any }
|
|
||||||
|
|
||||||
declare _G: any
|
|
||||||
declare _VERSION: string
|
|
||||||
|
|
||||||
declare function gcinfo(): number
|
|
||||||
|
|
||||||
declare function print<T...>(...: T...)
|
|
||||||
|
|
||||||
declare function type<T>(value: T): string
|
|
||||||
declare function typeof<T>(value: T): string
|
|
||||||
|
|
||||||
-- `assert` has a magic function attached that will give more detailed type information
|
|
||||||
declare function assert<T>(value: T, errorMessage: string?): T
|
|
||||||
declare function error<T>(message: T, level: number?): never
|
|
||||||
|
|
||||||
declare function tostring<T>(value: T): string
|
|
||||||
declare function tonumber<T>(value: T, radix: number?): number?
|
|
||||||
|
|
||||||
declare function rawequal<T1, T2>(a: T1, b: T2): boolean
|
|
||||||
declare function rawget<K, V>(tab: {[K]: V}, k: K): V
|
|
||||||
declare function rawset<K, V>(tab: {[K]: V}, k: K, v: V): {[K]: V}
|
|
||||||
declare function rawlen<K, V>(obj: {[K]: V} | string): number
|
|
||||||
|
|
||||||
declare function setfenv<T..., R...>(target: number | (T...) -> R..., env: {[string]: any}): ((T...) -> R...)?
|
|
||||||
|
|
||||||
declare function ipairs<V>(tab: {V}): (({V}, number) -> (number?, V), {V}, number)
|
|
||||||
|
|
||||||
declare function pcall<A..., R...>(f: (A...) -> R..., ...: A...): (boolean, R...)
|
|
||||||
|
|
||||||
-- FIXME: The actual type of `xpcall` is:
|
|
||||||
-- <E, A..., R1..., R2...>(f: (A...) -> R1..., err: (E) -> R2..., A...) -> (true, R1...) | (false, R2...)
|
|
||||||
-- Since we can't represent the return value, we use (boolean, R1...).
|
|
||||||
declare function xpcall<E, A..., R1..., R2...>(f: (A...) -> R1..., err: (E) -> R2..., ...: A...): (boolean, R1...)
|
|
||||||
|
|
||||||
-- `select` has a magic function attached to provide more detailed type information
|
|
||||||
declare function select<A...>(i: string | number, ...: A...): ...any
|
|
||||||
|
|
||||||
-- FIXME: This type is not entirely correct - `loadstring` returns a function or
|
|
||||||
-- (nil, string).
|
|
||||||
declare function loadstring<A...>(src: string, chunkname: string?): (((A...) -> any)?, string?)
|
|
||||||
|
|
||||||
@checked declare function newproxy(mt: boolean?): any
|
|
||||||
|
|
||||||
declare coroutine: {
|
|
||||||
create: <A..., R...>(f: (A...) -> R...) -> thread,
|
|
||||||
resume: <A..., R...>(co: thread, A...) -> (boolean, R...),
|
|
||||||
running: () -> thread,
|
|
||||||
status: @checked (co: thread) -> "dead" | "running" | "normal" | "suspended",
|
|
||||||
wrap: <A..., R...>(f: (A...) -> R...) -> ((A...) -> R...),
|
|
||||||
yield: <A..., R...>(A...) -> R...,
|
|
||||||
isyieldable: () -> boolean,
|
|
||||||
close: @checked (co: thread) -> (boolean, any)
|
|
||||||
}
|
|
||||||
|
|
||||||
declare table: {
|
|
||||||
concat: <V>(t: {V}, sep: string?, i: number?, j: number?) -> string,
|
|
||||||
insert: (<V>(t: {V}, value: V) -> ()) & (<V>(t: {V}, pos: number, value: V) -> ()),
|
|
||||||
maxn: <V>(t: {V}) -> number,
|
|
||||||
remove: <V>(t: {V}, number?) -> V?,
|
|
||||||
sort: <V>(t: {V}, comp: ((V, V) -> boolean)?) -> (),
|
|
||||||
create: <V>(count: number, value: V?) -> {V},
|
|
||||||
find: <V>(haystack: {V}, needle: V, init: number?) -> number?,
|
|
||||||
|
|
||||||
unpack: <V>(list: {V}, i: number?, j: number?) -> ...V,
|
|
||||||
pack: <V>(...V) -> { n: number, [number]: V },
|
|
||||||
|
|
||||||
getn: <V>(t: {V}) -> number,
|
|
||||||
foreach: <K, V>(t: {[K]: V}, f: (K, V) -> ()) -> (),
|
|
||||||
foreachi: <V>({V}, (number, V) -> ()) -> (),
|
|
||||||
|
|
||||||
move: <V>(src: {V}, a: number, b: number, t: number, dst: {V}?) -> {V},
|
|
||||||
clear: <K, V>(table: {[K]: V}) -> (),
|
|
||||||
|
|
||||||
isfrozen: <K, V>(t: {[K]: V}) -> boolean,
|
|
||||||
}
|
|
||||||
|
|
||||||
declare debug: {
|
|
||||||
info: (<R...>(thread: thread, level: number, options: string) -> R...) & (<R...>(level: number, options: string) -> R...) & (<A..., R1..., R2...>(func: (A...) -> R1..., options: string) -> R2...),
|
|
||||||
traceback: ((message: string?, level: number?) -> string) & ((thread: thread, message: string?, level: number?) -> string),
|
|
||||||
}
|
|
||||||
|
|
||||||
declare utf8: {
|
|
||||||
char: @checked (...number) -> string,
|
|
||||||
charpattern: string,
|
|
||||||
codes: @checked (str: string) -> ((string, number) -> (number, number), string, number),
|
|
||||||
codepoint: @checked (str: string, i: number?, j: number?) -> ...number,
|
|
||||||
len: @checked (s: string, i: number?, j: number?) -> (number?, number?),
|
|
||||||
offset: @checked (s: string, n: number?, i: number?) -> number,
|
|
||||||
}
|
|
||||||
|
|
||||||
-- Cannot use `typeof` here because it will produce a polytype when we expect a monotype.
|
|
||||||
declare function unpack<V>(tab: {V}, i: number?, j: number?): ...V
|
|
||||||
|
|
||||||
|
|
||||||
--- Buffer API
|
|
||||||
declare buffer: {
|
|
||||||
create: @checked (size: number) -> buffer,
|
|
||||||
fromstring: @checked (str: string) -> buffer,
|
|
||||||
tostring: @checked (b: buffer) -> string,
|
|
||||||
len: @checked (b: buffer) -> number,
|
|
||||||
copy: @checked (target: buffer, targetOffset: number, source: buffer, sourceOffset: number?, count: number?) -> (),
|
|
||||||
fill: @checked (b: buffer, offset: number, value: number, count: number?) -> (),
|
|
||||||
readi8: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readu8: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readi16: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readu16: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readi32: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readu32: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readf32: @checked (b: buffer, offset: number) -> number,
|
|
||||||
readf64: @checked (b: buffer, offset: number) -> number,
|
|
||||||
writei8: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writeu8: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writei16: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writeu16: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writei32: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writeu32: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writef32: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
writef64: @checked (b: buffer, offset: number, value: number) -> (),
|
|
||||||
readstring: @checked (b: buffer, offset: number, count: number) -> string,
|
|
||||||
writestring: @checked (b: buffer, offset: number, value: string, count: number?) -> (),
|
|
||||||
}
|
|
||||||
|
|
||||||
)BUILTIN_SRC";
|
|
||||||
|
|
||||||
static const std::string kBuiltinDefinitionLuaSrcChecked = R"BUILTIN_SRC(
|
|
||||||
|
|
||||||
declare bit32: {
|
|
||||||
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: {
|
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,
|
||||||
|
@ -422,7 +199,231 @@ declare utf8: {
|
||||||
-- 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
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionBaseSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
@checked declare function require(target: any): any
|
||||||
|
|
||||||
|
@checked declare function getfenv(target: any): { [string]: any }
|
||||||
|
|
||||||
|
declare _G: any
|
||||||
|
declare _VERSION: string
|
||||||
|
|
||||||
|
declare function gcinfo(): number
|
||||||
|
|
||||||
|
declare function print<T...>(...: T...)
|
||||||
|
|
||||||
|
declare function type<T>(value: T): string
|
||||||
|
declare function typeof<T>(value: T): string
|
||||||
|
|
||||||
|
-- `assert` has a magic function attached that will give more detailed type information
|
||||||
|
declare function assert<T>(value: T, errorMessage: string?): T
|
||||||
|
declare function error<T>(message: T, level: number?): never
|
||||||
|
|
||||||
|
declare function tostring<T>(value: T): string
|
||||||
|
declare function tonumber<T>(value: T, radix: number?): number?
|
||||||
|
|
||||||
|
declare function rawequal<T1, T2>(a: T1, b: T2): boolean
|
||||||
|
declare function rawget<K, V>(tab: {[K]: V}, k: K): V
|
||||||
|
declare function rawset<K, V>(tab: {[K]: V}, k: K, v: V): {[K]: V}
|
||||||
|
declare function rawlen<K, V>(obj: {[K]: V} | string): number
|
||||||
|
|
||||||
|
declare function setfenv<T..., R...>(target: number | (T...) -> R..., env: {[string]: any}): ((T...) -> R...)?
|
||||||
|
|
||||||
|
declare function ipairs<V>(tab: {V}): (({V}, number) -> (number?, V), {V}, number)
|
||||||
|
|
||||||
|
declare function pcall<A..., R...>(f: (A...) -> R..., ...: A...): (boolean, R...)
|
||||||
|
|
||||||
|
-- FIXME: The actual type of `xpcall` is:
|
||||||
|
-- <E, A..., R1..., R2...>(f: (A...) -> R1..., err: (E) -> R2..., A...) -> (true, R1...) | (false, R2...)
|
||||||
|
-- Since we can't represent the return value, we use (boolean, R1...).
|
||||||
|
declare function xpcall<E, A..., R1..., R2...>(f: (A...) -> R1..., err: (E) -> R2..., ...: A...): (boolean, R1...)
|
||||||
|
|
||||||
|
-- `select` has a magic function attached to provide more detailed type information
|
||||||
|
declare function select<A...>(i: string | number, ...: A...): ...any
|
||||||
|
|
||||||
|
-- FIXME: This type is not entirely correct - `loadstring` returns a function or
|
||||||
|
-- (nil, string).
|
||||||
|
declare function loadstring<A...>(src: string, chunkname: string?): (((A...) -> any)?, string?)
|
||||||
|
|
||||||
|
@checked declare function newproxy(mt: boolean?): any
|
||||||
|
|
||||||
|
-- Cannot use `typeof` here because it will produce a polytype when we expect a monotype.
|
||||||
|
declare function unpack<V>(tab: {V}, i: number?, j: number?): ...V
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionBit32Src = 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,
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionMathSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
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,
|
||||||
|
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 = {
|
||||||
|
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,
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionCoroutineSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionTableSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
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,
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionDebugSrc = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
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),
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
static const std::string kBuiltinDefinitionUtf8Src = R"BUILTIN_SRC(
|
||||||
|
|
||||||
|
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,
|
||||||
|
}
|
||||||
|
|
||||||
|
)BUILTIN_SRC";
|
||||||
|
|
||||||
|
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 +454,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 +527,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 +549,25 @@ declare vector: {
|
||||||
|
|
||||||
std::string getBuiltinDefinitionSource()
|
std::string getBuiltinDefinitionSource()
|
||||||
{
|
{
|
||||||
std::string result = FFlag::LuauMathMap ? kBuiltinDefinitionLuaSrcChecked : kBuiltinDefinitionLuaSrcChecked_DEPRECATED;
|
std::string result = FFlag::LuauMathMapDefinition ? kBuiltinDefinitionBaseSrc : kBuiltinDefinitionLuaSrcChecked_DEPRECATED;
|
||||||
|
|
||||||
if (FFlag::LuauVectorDefinitionsExtra)
|
if (FFlag::LuauMathMapDefinition)
|
||||||
|
{
|
||||||
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -143,6 +149,61 @@ static bool isTerminal(const EType& node)
|
||||||
node.get<TNever>() || node.get<TNoRefine>();
|
node.get<TNever>() || node.get<TNoRefine>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool areTerminalAndDefinitelyDisjoint(const EType& lhs, const EType& rhs)
|
||||||
|
{
|
||||||
|
// If either node is non-terminal, then we early exit: we're not going to
|
||||||
|
// do a state space search for whether something like:
|
||||||
|
// (A | B | C | D) & (E | F | G | H)
|
||||||
|
// ... is a disjoint intersection.
|
||||||
|
if (!isTerminal(lhs) || !isTerminal(rhs))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// Special case some types that aren't strict, disjoint subsets.
|
||||||
|
if (lhs.get<TTopClass>() || lhs.get<TClass>())
|
||||||
|
return !(rhs.get<TTopClass>() || rhs.get<TClass>());
|
||||||
|
|
||||||
|
// Handling strings / booleans: these are the types for which we
|
||||||
|
// expect something like:
|
||||||
|
//
|
||||||
|
// "foo" & ~"bar"
|
||||||
|
//
|
||||||
|
// ... to simplify to "foo".
|
||||||
|
if (lhs.get<TString>())
|
||||||
|
return !(rhs.get<TString>() || rhs.get<SString>());
|
||||||
|
|
||||||
|
if (lhs.get<TBoolean>())
|
||||||
|
return !(rhs.get<TBoolean>() || rhs.get<SBoolean>());
|
||||||
|
|
||||||
|
if (auto lhsSString = lhs.get<SString>())
|
||||||
|
{
|
||||||
|
auto rhsSString = rhs.get<SString>();
|
||||||
|
if (!rhsSString)
|
||||||
|
return !rhs.get<TString>();
|
||||||
|
return lhsSString->value() != rhsSString->value();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto lhsSBoolean = lhs.get<SBoolean>())
|
||||||
|
{
|
||||||
|
auto rhsSBoolean = rhs.get<SBoolean>();
|
||||||
|
if (!rhsSBoolean)
|
||||||
|
return !rhs.get<TBoolean>();
|
||||||
|
return lhsSBoolean->value() != rhsSBoolean->value();
|
||||||
|
}
|
||||||
|
|
||||||
|
// At this point:
|
||||||
|
// - We know both nodes are terminal
|
||||||
|
// - We know that the LHS is not any boolean, string, or class
|
||||||
|
// At this point, we have two classes of checks left:
|
||||||
|
// - Whether the two enodes are exactly the same set (now that the static
|
||||||
|
// sets have been covered).
|
||||||
|
// - Whether one of the enodes is a large semantic set such as TAny,
|
||||||
|
// TUnknown, or TError.
|
||||||
|
return !(
|
||||||
|
lhs.index() == rhs.index() || lhs.get<TUnknown>() || rhs.get<TUnknown>() || lhs.get<TAny>() || rhs.get<TAny>() || lhs.get<TNoRefine>() ||
|
||||||
|
rhs.get<TNoRefine>() || lhs.get<TError>() || rhs.get<TError>() || lhs.get<TOpaque>() || rhs.get<TOpaque>()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
static bool isTerminal(const EGraph& egraph, Id eclass)
|
static bool isTerminal(const EGraph& egraph, Id eclass)
|
||||||
{
|
{
|
||||||
const auto& nodes = egraph[eclass].nodes;
|
const auto& nodes = egraph[eclass].nodes;
|
||||||
|
@ -151,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);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -335,11 +396,31 @@ 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());
|
||||||
for (TypeId part : tfun->typeArguments)
|
for (TypeId part : tfun->typeArguments)
|
||||||
parts.push_back(toId(egraph, builtinTypes, mappingIdToClass, typeToMappingId, boundNodes, strings, part));
|
parts.push_back(toId(egraph, builtinTypes, mappingIdToClass, typeToMappingId, boundNodes, strings, part));
|
||||||
|
|
||||||
return cache(egraph.add(TTypeFun{tfun->function.get(), std::move(parts)}));
|
// This looks sily, but we're making a copy of the specific
|
||||||
|
// `TypeFunctionInstanceType` outside of the provided arena so that
|
||||||
|
// we can access the members without fear of the specific TFIT being
|
||||||
|
// overwritten with a bound type.
|
||||||
|
return cache(egraph.add(TTypeFun{
|
||||||
|
std::make_shared<const TypeFunctionInstanceType>(
|
||||||
|
tfun->function, tfun->typeArguments, tfun->packArguments, tfun->userFuncName, tfun->userFuncData
|
||||||
|
),
|
||||||
|
std::move(parts)
|
||||||
|
}));
|
||||||
}
|
}
|
||||||
else if (get<NoRefineType>(ty))
|
else if (get<NoRefineType>(ty))
|
||||||
return egraph.add(TNoRefine{});
|
return egraph.add(TNoRefine{});
|
||||||
|
@ -399,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();
|
||||||
|
@ -416,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;
|
||||||
|
@ -468,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)
|
||||||
|
@ -574,28 +655,34 @@ TypeId flattenTableNode(
|
||||||
// If a TTable is its own basis, it must be the case that some other
|
// If a TTable is its own basis, it must be the case that some other
|
||||||
// node on this eclass is a TImportedTable. Let's use that.
|
// node on this eclass is a TImportedTable. Let's use that.
|
||||||
|
|
||||||
|
bool found = false;
|
||||||
|
|
||||||
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;
|
||||||
index = i;
|
index = i;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
// If we couldn't find one, we don't know what to do. Use ErrorType.
|
// If we couldn't find one, we don't know what to do. Use ErrorType.
|
||||||
LUAU_ASSERT(0);
|
LUAU_ASSERT(0);
|
||||||
return builtinTypes->errorType;
|
return builtinTypes->errorType;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
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;
|
||||||
|
@ -622,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)};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -646,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;
|
||||||
|
@ -703,7 +791,20 @@ TypeId fromId(
|
||||||
if (parts.empty())
|
if (parts.empty())
|
||||||
return builtinTypes->neverType;
|
return builtinTypes->neverType;
|
||||||
else if (parts.size() == 1)
|
else if (parts.size() == 1)
|
||||||
return fromId(egraph, strings, builtinTypes, arena, bestNodes, seen, newTypeFunctions, parts[0]);
|
{
|
||||||
|
TypeId placeholder = arena->addType(BlockedType{});
|
||||||
|
seen[rootId] = placeholder;
|
||||||
|
auto result = fromId(egraph, strings, builtinTypes, arena, bestNodes, seen, newTypeFunctions, parts[0]);
|
||||||
|
if (follow(result) == placeholder)
|
||||||
|
{
|
||||||
|
emplaceType<GenericType>(asMutable(placeholder), "EGRAPH-SINGLETON-CYCLE");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
emplaceType<BoundType>(asMutable(placeholder), result);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
TypeId res = arena->addType(BlockedType{});
|
TypeId res = arena->addType(BlockedType{});
|
||||||
|
@ -768,7 +869,11 @@ TypeId fromId(
|
||||||
for (Id part : tfun->operands())
|
for (Id part : tfun->operands())
|
||||||
args.push_back(fromId(egraph, strings, builtinTypes, arena, bestNodes, seen, newTypeFunctions, part));
|
args.push_back(fromId(egraph, strings, builtinTypes, arena, bestNodes, seen, newTypeFunctions, part));
|
||||||
|
|
||||||
asMutable(res)->ty.emplace<TypeFunctionInstanceType>(*tfun->value(), std::move(args));
|
auto oldInstance = tfun->value();
|
||||||
|
|
||||||
|
asMutable(res)->ty.emplace<TypeFunctionInstanceType>(
|
||||||
|
oldInstance->function, std::move(args), std::vector<TypePackId>(), oldInstance->userFuncName, oldInstance->userFuncData
|
||||||
|
);
|
||||||
|
|
||||||
newTypeFunctions.push_back(res);
|
newTypeFunctions.push_back(res);
|
||||||
|
|
||||||
|
@ -853,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 "";
|
||||||
|
@ -906,7 +1019,7 @@ static std::string getNodeName(const StringCache& strings, const EType& node)
|
||||||
else if (node.get<TNever>())
|
else if (node.get<TNever>())
|
||||||
return "never";
|
return "never";
|
||||||
else if (auto tfun = node.get<TTypeFun>())
|
else if (auto tfun = node.get<TTypeFun>())
|
||||||
return "tfun " + tfun->value()->name;
|
return "tfun " + tfun->value()->function->name;
|
||||||
else if (node.get<Negation>())
|
else if (node.get<Negation>())
|
||||||
return "~";
|
return "~";
|
||||||
else if (node.get<Invalid>())
|
else if (node.get<Invalid>())
|
||||||
|
@ -928,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())
|
||||||
|
@ -950,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);
|
||||||
|
@ -965,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);
|
||||||
|
@ -1001,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;
|
||||||
|
@ -1037,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;
|
||||||
|
@ -1225,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;
|
||||||
|
|
||||||
|
@ -1342,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;
|
||||||
|
@ -1387,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)
|
||||||
|
@ -1433,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1467,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]))
|
||||||
|
@ -1517,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}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1552,11 +1686,6 @@ std::optional<EType> intersectOne(EGraph& egraph, Id hereId, const EType* hereNo
|
||||||
thereNode->get<Intersection>() || thereNode->get<Negation>() || hereNode->get<TOpaque>() || thereNode->get<TOpaque>())
|
thereNode->get<Intersection>() || thereNode->get<Negation>() || hereNode->get<TOpaque>() || thereNode->get<TOpaque>())
|
||||||
return std::nullopt;
|
return std::nullopt;
|
||||||
|
|
||||||
if (hereNode->get<TAny>())
|
|
||||||
return *thereNode;
|
|
||||||
if (thereNode->get<TAny>())
|
|
||||||
return *hereNode;
|
|
||||||
|
|
||||||
if (hereNode->get<TUnknown>())
|
if (hereNode->get<TUnknown>())
|
||||||
return *thereNode;
|
return *thereNode;
|
||||||
if (thereNode->get<TUnknown>())
|
if (thereNode->get<TUnknown>())
|
||||||
|
@ -1732,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))
|
||||||
|
@ -1755,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}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1788,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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1808,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
|
||||||
|
@ -1821,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}}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1839,6 +1988,74 @@ void Simplifier::intersectWithNegatedClass(Id id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Simplifier::intersectWithNegatedAtom(Id id)
|
||||||
|
{
|
||||||
|
// Let I and ~J be two arbitrary distinct operands of an intersection where
|
||||||
|
// I and J are terminal but are not type variables. (free, generic, or
|
||||||
|
// otherwise opaque)
|
||||||
|
//
|
||||||
|
// If I and J are equal, then the whole intersection is equivalent to never.
|
||||||
|
//
|
||||||
|
// If I and J are inequal, then J & ~I == J
|
||||||
|
|
||||||
|
for (const auto [intersection, intersectionIndex] : Query<Intersection>(&egraph, id))
|
||||||
|
{
|
||||||
|
const Slice<const Id>& intersectionOperands = intersection->operands();
|
||||||
|
for (size_t i = 0; i < intersectionOperands.size(); ++i)
|
||||||
|
{
|
||||||
|
for (const auto [negation, negationIndex] : Query<Negation>(&egraph, intersectionOperands[i]))
|
||||||
|
{
|
||||||
|
for (size_t negationOperandIndex = 0; negationOperandIndex < egraph[negation->operands()[0]].nodes.size(); ++negationOperandIndex)
|
||||||
|
{
|
||||||
|
const EType* negationOperand = &egraph[negation->operands()[0]].nodes[negationOperandIndex].node;
|
||||||
|
if (!isTerminal(*negationOperand) || negationOperand->get<TOpaque>())
|
||||||
|
continue;
|
||||||
|
|
||||||
|
for (size_t j = 0; j < intersectionOperands.size(); ++j)
|
||||||
|
{
|
||||||
|
if (j == i)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
for (size_t jNodeIndex = 0; jNodeIndex < egraph[intersectionOperands[j]].nodes.size(); ++jNodeIndex)
|
||||||
|
{
|
||||||
|
const EType* jNode = &egraph[intersectionOperands[j]].nodes[jNodeIndex].node;
|
||||||
|
if (!isTerminal(*jNode) || jNode->get<TOpaque>())
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (*negationOperand == *jNode)
|
||||||
|
{
|
||||||
|
// eg "Hello" & ~"Hello"
|
||||||
|
// or boolean & ~boolean
|
||||||
|
subst(
|
||||||
|
id,
|
||||||
|
egraph.add(TNever{}),
|
||||||
|
"intersectWithNegatedAtom",
|
||||||
|
{{id, intersectionIndex}, {intersectionOperands[i], negationIndex}, {intersectionOperands[j], jNodeIndex}}
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else if (areTerminalAndDefinitelyDisjoint(*jNode, *negationOperand))
|
||||||
|
{
|
||||||
|
// eg "Hello" & ~"World"
|
||||||
|
// or boolean & ~string
|
||||||
|
std::vector<Id> newOperands(intersectionOperands.begin(), intersectionOperands.end());
|
||||||
|
newOperands.erase(newOperands.begin() + std::vector<Id>::difference_type(i));
|
||||||
|
|
||||||
|
subst(
|
||||||
|
id,
|
||||||
|
mkIntersection(egraph, std::move(newOperands)),
|
||||||
|
"intersectWithNegatedAtom",
|
||||||
|
{{id, intersectionIndex}, {intersectionOperands[i], negationIndex}, {intersectionOperands[j], jNodeIndex}}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void Simplifier::intersectWithNoRefine(Id id)
|
void Simplifier::intersectWithNoRefine(Id id)
|
||||||
{
|
{
|
||||||
for (const auto pair : Query<Intersection>(&egraph, id))
|
for (const auto pair : Query<Intersection>(&egraph, id))
|
||||||
|
@ -2003,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}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2160,7 +2377,7 @@ void Simplifier::intersectTableProperty(Id id)
|
||||||
|
|
||||||
subst(
|
subst(
|
||||||
id,
|
id,
|
||||||
egraph.add(Intersection{std::move(newIntersectionParts)}),
|
mkIntersection(egraph, std::move(newIntersectionParts)),
|
||||||
"intersectTableProperty",
|
"intersectTableProperty",
|
||||||
{{id, intersectionIndex}, {iId, table1Index}, {jId, table2Index}}
|
{{id, intersectionIndex}, {iId, table1Index}, {jId, table2Index}}
|
||||||
);
|
);
|
||||||
|
@ -2250,7 +2467,7 @@ void Simplifier::builtinTypeFunctions(Id id)
|
||||||
if (args.size() != 2)
|
if (args.size() != 2)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
const std::string& name = tfun->value()->name;
|
const std::string& name = tfun->value()->function->name;
|
||||||
if (name == "add" || name == "sub" || name == "mul" || name == "div" || name == "idiv" || name == "pow" || name == "mod")
|
if (name == "add" || name == "sub" || name == "mul" || name == "div" || name == "idiv" || name == "pow" || name == "mod")
|
||||||
{
|
{
|
||||||
if (isTag<TNumber>(args[0]) && isTag<TNumber>(args[1]))
|
if (isTag<TNumber>(args[0]) && isTag<TNumber>(args[1]))
|
||||||
|
@ -2272,15 +2489,43 @@ void Simplifier::iffyTypeFunctions(Id id)
|
||||||
{
|
{
|
||||||
const Slice<const Id>& args = tfun->operands();
|
const Slice<const Id>& args = tfun->operands();
|
||||||
|
|
||||||
const std::string& name = tfun->value()->name;
|
const std::string& name = tfun->value()->function->name;
|
||||||
|
|
||||||
if (name == "union")
|
if (name == "union")
|
||||||
subst(id, add(Union{std::vector(args.begin(), args.end())}), "iffyTypeFunctions", {{id, index}});
|
subst(id, add(Union{std::vector(args.begin(), args.end())}), "iffyTypeFunctions", {{id, index}});
|
||||||
else if (name == "intersect" || name == "refine")
|
else if (name == "intersect")
|
||||||
subst(id, add(Intersection{std::vector(args.begin(), args.end())}), "iffyTypeFunctions", {{id, index}});
|
subst(id, add(Intersection{std::vector(args.begin(), args.end())}), "iffyTypeFunctions", {{id, index}});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Replace instances of `lt<X, Y>` and `le<X, Y>` when either X or Y is `number`
|
||||||
|
// or `string` with `boolean`. Lua semantics are that if we see the expression:
|
||||||
|
//
|
||||||
|
// x < y
|
||||||
|
//
|
||||||
|
// ... we error if `x` and `y` don't have the same type. We know that for
|
||||||
|
// `string` and `number`, comparisons will always return a boolean. So if either
|
||||||
|
// of the arguments to `lt<>` are equivalent to `number` or `string`, then the
|
||||||
|
// type is effectively `boolean`: either the other type is equivalent, in which
|
||||||
|
// case we eval to `boolean`, or we diverge (raise an error).
|
||||||
|
void Simplifier::strictMetamethods(Id id)
|
||||||
|
{
|
||||||
|
for (const auto [tfun, index] : Query<TTypeFun>(&egraph, id))
|
||||||
|
{
|
||||||
|
const Slice<const Id>& args = tfun->operands();
|
||||||
|
|
||||||
|
const std::string& name = tfun->value()->function->name;
|
||||||
|
|
||||||
|
if (!(name == "lt" || name == "le") || args.size() != 2)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if (isTag<TNumber>(args[0]) || isTag<TString>(args[0]) || isTag<TNumber>(args[1]) || isTag<TString>(args[1]))
|
||||||
|
{
|
||||||
|
subst(id, add(TBoolean{}), __FUNCTION__, {{id, index}});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void deleteSimplifier(Simplifier* s)
|
static void deleteSimplifier(Simplifier* s)
|
||||||
{
|
{
|
||||||
delete s;
|
delete s;
|
||||||
|
@ -2308,6 +2553,7 @@ std::optional<EqSatSimplificationResult> eqSatSimplify(NotNull<Simplifier> simpl
|
||||||
&Simplifier::simplifyUnion,
|
&Simplifier::simplifyUnion,
|
||||||
&Simplifier::uninhabitedIntersection,
|
&Simplifier::uninhabitedIntersection,
|
||||||
&Simplifier::intersectWithNegatedClass,
|
&Simplifier::intersectWithNegatedClass,
|
||||||
|
&Simplifier::intersectWithNegatedAtom,
|
||||||
&Simplifier::intersectWithNoRefine,
|
&Simplifier::intersectWithNoRefine,
|
||||||
&Simplifier::cyclicIntersectionOfUnion,
|
&Simplifier::cyclicIntersectionOfUnion,
|
||||||
&Simplifier::cyclicUnionOfIntersection,
|
&Simplifier::cyclicUnionOfIntersection,
|
||||||
|
@ -2318,6 +2564,7 @@ std::optional<EqSatSimplificationResult> eqSatSimplify(NotNull<Simplifier> simpl
|
||||||
&Simplifier::unneededTableModification,
|
&Simplifier::unneededTableModification,
|
||||||
&Simplifier::builtinTypeFunctions,
|
&Simplifier::builtinTypeFunctions,
|
||||||
&Simplifier::iffyTypeFunctions,
|
&Simplifier::iffyTypeFunctions,
|
||||||
|
&Simplifier::strictMetamethods,
|
||||||
};
|
};
|
||||||
|
|
||||||
std::unordered_set<Id> seen;
|
std::unordered_set<Id> seen;
|
||||||
|
@ -2371,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;
|
||||||
|
@ -2383,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)
|
||||||
|
|
||||||
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,
|
||||||
|
@ -245,6 +341,9 @@ FragmentParseResult parseFragment(
|
||||||
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 +357,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 +408,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 +446,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 +475,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 +496,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 +563,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 +572,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 +594,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,7 +641,14 @@ 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
|
||||||
|
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();
|
auto globalScope = (opts && opts->forAutocomplete) ? frontend.globalsForAutocomplete.globalScope.get() : frontend.globals.globalScope.get();
|
||||||
|
|
||||||
TypeArena arenaForFragmentAutocomplete;
|
TypeArena arenaForFragmentAutocomplete;
|
||||||
|
|
|
@ -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,9 +47,14 @@ 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(LuauSelectivelyRetainDFGArena)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -135,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;
|
||||||
|
@ -817,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
|
||||||
{
|
{
|
||||||
|
@ -1046,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();
|
||||||
|
@ -1099,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);
|
||||||
|
@ -1109,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;
|
||||||
|
@ -1144,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);
|
||||||
|
@ -1157,6 +1218,31 @@ bool Frontend::isDirty(const ModuleName& name, bool forAutocomplete) const
|
||||||
* It would be nice for this function to be O(1)
|
* It would be nice for this function to be O(1)
|
||||||
*/
|
*/
|
||||||
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;
|
||||||
|
@ -1197,6 +1283,33 @@ void Frontend::markDirty(const ModuleName& name, std::vector<ModuleName>* marked
|
||||||
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)
|
||||||
{
|
{
|
||||||
|
@ -1317,6 +1430,11 @@ ModulePtr check(
|
||||||
result->mode = mode;
|
result->mode = mode;
|
||||||
result->internalTypes.owningModule = result.get();
|
result->internalTypes.owningModule = result.get();
|
||||||
result->interfaceTypes.owningModule = result.get();
|
result->interfaceTypes.owningModule = result.get();
|
||||||
|
if (FFlag::LuauReferenceAllocatorInNewSolver)
|
||||||
|
{
|
||||||
|
result->allocator = sourceModule.allocator;
|
||||||
|
result->names = sourceModule.names;
|
||||||
|
}
|
||||||
|
|
||||||
iceHandler->moduleName = sourceModule.name;
|
iceHandler->moduleName = sourceModule.name;
|
||||||
|
|
||||||
|
@ -1331,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;
|
||||||
|
@ -1427,6 +1545,7 @@ ModulePtr check(
|
||||||
case Mode::Nonstrict:
|
case Mode::Nonstrict:
|
||||||
Luau::checkNonStrict(
|
Luau::checkNonStrict(
|
||||||
builtinTypes,
|
builtinTypes,
|
||||||
|
NotNull{simplifier.get()},
|
||||||
NotNull{&typeFunctionRuntime},
|
NotNull{&typeFunctionRuntime},
|
||||||
iceHandler,
|
iceHandler,
|
||||||
NotNull{&unifierState},
|
NotNull{&unifierState},
|
||||||
|
@ -1440,7 +1559,14 @@ ModulePtr check(
|
||||||
// fallthrough intentional
|
// fallthrough intentional
|
||||||
case Mode::Strict:
|
case Mode::Strict:
|
||||||
Luau::check(
|
Luau::check(
|
||||||
builtinTypes, NotNull{&typeFunctionRuntime}, NotNull{&unifierState}, NotNull{&limits}, logger.get(), sourceModule, result.get()
|
builtinTypes,
|
||||||
|
NotNull{simplifier.get()},
|
||||||
|
NotNull{&typeFunctionRuntime},
|
||||||
|
NotNull{&unifierState},
|
||||||
|
NotNull{&limits},
|
||||||
|
logger.get(),
|
||||||
|
sourceModule,
|
||||||
|
result.get()
|
||||||
);
|
);
|
||||||
break;
|
break;
|
||||||
case Mode::NoCheck:
|
case Mode::NoCheck:
|
||||||
|
@ -1622,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;
|
||||||
|
@ -1743,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()
|
||||||
|
|
|
@ -10,12 +10,14 @@
|
||||||
#include "Luau/VisitType.h"
|
#include "Luau/VisitType.h"
|
||||||
|
|
||||||
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
LUAU_FASTFLAG(LuauAutocompleteRefactorsForIncrementalAutocomplete)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauGeneralizationRemoveRecursiveUpperBound)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
struct MutatingGeneralizer : TypeOnceVisitor
|
struct MutatingGeneralizer : TypeOnceVisitor
|
||||||
{
|
{
|
||||||
|
NotNull<TypeArena> arena;
|
||||||
NotNull<BuiltinTypes> builtinTypes;
|
NotNull<BuiltinTypes> builtinTypes;
|
||||||
|
|
||||||
NotNull<Scope> scope;
|
NotNull<Scope> scope;
|
||||||
|
@ -29,6 +31,7 @@ struct MutatingGeneralizer : TypeOnceVisitor
|
||||||
bool avoidSealingTables = false;
|
bool avoidSealingTables = false;
|
||||||
|
|
||||||
MutatingGeneralizer(
|
MutatingGeneralizer(
|
||||||
|
NotNull<TypeArena> arena,
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<Scope> scope,
|
NotNull<Scope> scope,
|
||||||
NotNull<DenseHashSet<TypeId>> cachedTypes,
|
NotNull<DenseHashSet<TypeId>> cachedTypes,
|
||||||
|
@ -37,6 +40,7 @@ struct MutatingGeneralizer : TypeOnceVisitor
|
||||||
bool avoidSealingTables
|
bool avoidSealingTables
|
||||||
)
|
)
|
||||||
: TypeOnceVisitor(/* skipBoundTypes */ true)
|
: TypeOnceVisitor(/* skipBoundTypes */ true)
|
||||||
|
, arena(arena)
|
||||||
, builtinTypes(builtinTypes)
|
, builtinTypes(builtinTypes)
|
||||||
, scope(scope)
|
, scope(scope)
|
||||||
, cachedTypes(cachedTypes)
|
, cachedTypes(cachedTypes)
|
||||||
|
@ -229,6 +233,53 @@ struct MutatingGeneralizer : TypeOnceVisitor
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
TypeId ub = follow(ft->upperBound);
|
TypeId ub = follow(ft->upperBound);
|
||||||
|
if (FFlag::LuauGeneralizationRemoveRecursiveUpperBound)
|
||||||
|
{
|
||||||
|
|
||||||
|
// If the upper bound is a union type or an intersection type,
|
||||||
|
// and one of it's members is the free type we're
|
||||||
|
// generalizing, don't include it in the upper bound. For a
|
||||||
|
// free type such as:
|
||||||
|
//
|
||||||
|
// t1 where t1 = D <: 'a <: (A | B | C | t1)
|
||||||
|
//
|
||||||
|
// Naively replacing it with it's upper bound creates:
|
||||||
|
//
|
||||||
|
// t1 where t1 = A | B | C | t1
|
||||||
|
//
|
||||||
|
// It makes sense to just optimize this and exclude the
|
||||||
|
// recursive component by semantic subtyping rules.
|
||||||
|
|
||||||
|
if (auto itv = get<IntersectionType>(ub))
|
||||||
|
{
|
||||||
|
std::vector<TypeId> newIds;
|
||||||
|
newIds.reserve(itv->parts.size());
|
||||||
|
for (auto part : itv)
|
||||||
|
{
|
||||||
|
if (part != ty)
|
||||||
|
newIds.push_back(part);
|
||||||
|
}
|
||||||
|
if (newIds.size() == 1)
|
||||||
|
ub = newIds[0];
|
||||||
|
else if (newIds.size() > 0)
|
||||||
|
ub = arena->addType(IntersectionType{std::move(newIds)});
|
||||||
|
}
|
||||||
|
else if (auto utv = get<UnionType>(ub))
|
||||||
|
{
|
||||||
|
std::vector<TypeId> newIds;
|
||||||
|
newIds.reserve(utv->options.size());
|
||||||
|
for (auto part : utv)
|
||||||
|
{
|
||||||
|
if (part != ty)
|
||||||
|
newIds.push_back(part);
|
||||||
|
}
|
||||||
|
if (newIds.size() == 1)
|
||||||
|
ub = newIds[0];
|
||||||
|
else if (newIds.size() > 0)
|
||||||
|
ub = arena->addType(UnionType{std::move(newIds)});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (FreeType* upperFree = getMutable<FreeType>(ub); upperFree && upperFree->lowerBound == ty)
|
if (FreeType* upperFree = getMutable<FreeType>(ub); upperFree && upperFree->lowerBound == ty)
|
||||||
upperFree->lowerBound = builtinTypes->neverType;
|
upperFree->lowerBound = builtinTypes->neverType;
|
||||||
else
|
else
|
||||||
|
@ -926,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);
|
||||||
|
@ -969,7 +1021,7 @@ std::optional<TypeId> generalize(
|
||||||
FreeTypeSearcher fts{scope, cachedTypes};
|
FreeTypeSearcher fts{scope, cachedTypes};
|
||||||
fts.traverse(ty);
|
fts.traverse(ty);
|
||||||
|
|
||||||
MutatingGeneralizer gen{builtinTypes, scope, cachedTypes, std::move(fts.positiveTypes), std::move(fts.negativeTypes), avoidSealingTables};
|
MutatingGeneralizer gen{arena, builtinTypes, scope, cachedTypes, std::move(fts.positiveTypes), std::move(fts.negativeTypes), avoidSealingTables};
|
||||||
|
|
||||||
gen.traverse(ty);
|
gen.traverse(ty);
|
||||||
|
|
||||||
|
|
|
@ -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,12 @@
|
||||||
#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 bool contains_DEPRECATED(Position pos, Comment comment)
|
||||||
{
|
{
|
||||||
if (comment.location.contains(pos))
|
if (comment.location.contains(pos))
|
||||||
return true;
|
return true;
|
||||||
|
@ -32,7 +33,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 +56,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 +68,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
|
||||||
|
|
|
@ -19,8 +19,8 @@
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <iterator>
|
#include <iterator>
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunNonstrict)
|
|
||||||
LUAU_FASTFLAGVARIABLE(LuauCountSelfCallsNonstrict)
|
LUAU_FASTFLAGVARIABLE(LuauCountSelfCallsNonstrict)
|
||||||
|
LUAU_FASTFLAG(LuauFreeTypesMustHaveBounds)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -158,6 +158,7 @@ private:
|
||||||
struct NonStrictTypeChecker
|
struct NonStrictTypeChecker
|
||||||
{
|
{
|
||||||
NotNull<BuiltinTypes> builtinTypes;
|
NotNull<BuiltinTypes> builtinTypes;
|
||||||
|
NotNull<Simplifier> simplifier;
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime;
|
||||||
const NotNull<InternalErrorReporter> ice;
|
const NotNull<InternalErrorReporter> ice;
|
||||||
NotNull<TypeArena> arena;
|
NotNull<TypeArena> arena;
|
||||||
|
@ -174,6 +175,7 @@ struct NonStrictTypeChecker
|
||||||
NonStrictTypeChecker(
|
NonStrictTypeChecker(
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
const NotNull<InternalErrorReporter> ice,
|
const NotNull<InternalErrorReporter> ice,
|
||||||
NotNull<UnifierSharedState> unifierState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
|
@ -182,12 +184,13 @@ struct NonStrictTypeChecker
|
||||||
Module* module
|
Module* module
|
||||||
)
|
)
|
||||||
: builtinTypes(builtinTypes)
|
: builtinTypes(builtinTypes)
|
||||||
|
, simplifier(simplifier)
|
||||||
, typeFunctionRuntime(typeFunctionRuntime)
|
, typeFunctionRuntime(typeFunctionRuntime)
|
||||||
, ice(ice)
|
, ice(ice)
|
||||||
, arena(arena)
|
, arena(arena)
|
||||||
, module(module)
|
, module(module)
|
||||||
, normalizer{arena, builtinTypes, unifierState, /* cache inhabitance */ true}
|
, normalizer{arena, builtinTypes, unifierState, /* cache inhabitance */ true}
|
||||||
, subtyping{builtinTypes, arena, NotNull(&normalizer), typeFunctionRuntime, ice}
|
, subtyping{builtinTypes, arena, simplifier, NotNull(&normalizer), typeFunctionRuntime, ice}
|
||||||
, dfg(dfg)
|
, dfg(dfg)
|
||||||
, limits(limits)
|
, limits(limits)
|
||||||
{
|
{
|
||||||
|
@ -209,7 +212,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));
|
||||||
|
@ -232,10 +235,11 @@ struct NonStrictTypeChecker
|
||||||
if (noTypeFunctionErrors.find(instance))
|
if (noTypeFunctionErrors.find(instance))
|
||||||
return instance;
|
return instance;
|
||||||
|
|
||||||
ErrorVec errors = reduceTypeFunctions(
|
ErrorVec errors =
|
||||||
|
reduceTypeFunctions(
|
||||||
instance,
|
instance,
|
||||||
location,
|
location,
|
||||||
TypeFunctionContext{arena, builtinTypes, stack.back(), NotNull{&normalizer}, typeFunctionRuntime, ice, limits},
|
TypeFunctionContext{arena, builtinTypes, stack.back(), simplifier, NotNull{&normalizer}, typeFunctionRuntime, ice, limits},
|
||||||
true
|
true
|
||||||
)
|
)
|
||||||
.errors;
|
.errors;
|
||||||
|
@ -424,9 +428,6 @@ struct NonStrictTypeChecker
|
||||||
|
|
||||||
NonStrictContext visit(AstStatTypeFunction* typeFunc)
|
NonStrictContext visit(AstStatTypeFunction* typeFunc)
|
||||||
{
|
{
|
||||||
if (!FFlag::LuauUserTypeFunNonstrict)
|
|
||||||
reportError(GenericError{"This syntax is not supported"}, typeFunc->location);
|
|
||||||
|
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -888,6 +889,7 @@ private:
|
||||||
|
|
||||||
void checkNonStrict(
|
void checkNonStrict(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> ice,
|
NotNull<InternalErrorReporter> ice,
|
||||||
NotNull<UnifierSharedState> unifierState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
|
@ -899,7 +901,9 @@ void checkNonStrict(
|
||||||
{
|
{
|
||||||
LUAU_TIMETRACE_SCOPE("checkNonStrict", "Typechecking");
|
LUAU_TIMETRACE_SCOPE("checkNonStrict", "Typechecking");
|
||||||
|
|
||||||
NonStrictTypeChecker typeChecker{NotNull{&module->internalTypes}, builtinTypes, typeFunctionRuntime, ice, unifierState, dfg, limits, module};
|
NonStrictTypeChecker typeChecker{
|
||||||
|
NotNull{&module->internalTypes}, builtinTypes, simplifier, typeFunctionRuntime, ice, unifierState, dfg, limits, module
|
||||||
|
};
|
||||||
typeChecker.visit(sourceModule.root);
|
typeChecker.visit(sourceModule.root);
|
||||||
unfreeze(module->interfaceTypes);
|
unfreeze(module->interfaceTypes);
|
||||||
copyErrors(module->errors, module->interfaceTypes, builtinTypes);
|
copyErrors(module->errors, module->interfaceTypes, builtinTypes);
|
||||||
|
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
@ -3041,13 +3051,10 @@ 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};
|
||||||
|
@ -3465,7 +3473,14 @@ TypeId Normalizer::typeFromNormal(const NormalizedType& norm)
|
||||||
return arena->addType(UnionType{std::move(result)});
|
return arena->addType(UnionType{std::move(result)});
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isSubtype(TypeId subTy, TypeId superTy, NotNull<Scope> scope, NotNull<BuiltinTypes> builtinTypes, InternalErrorReporter& ice)
|
bool isSubtype(
|
||||||
|
TypeId subTy,
|
||||||
|
TypeId superTy,
|
||||||
|
NotNull<Scope> scope,
|
||||||
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
|
InternalErrorReporter& ice
|
||||||
|
)
|
||||||
{
|
{
|
||||||
UnifierSharedState sharedState{&ice};
|
UnifierSharedState sharedState{&ice};
|
||||||
TypeArena arena;
|
TypeArena arena;
|
||||||
|
@ -3478,7 +3493,7 @@ bool isSubtype(TypeId subTy, TypeId superTy, NotNull<Scope> scope, NotNull<Built
|
||||||
// Subtyping under DCR is not implemented using unification!
|
// Subtyping under DCR is not implemented using unification!
|
||||||
if (FFlag::LuauSolverV2)
|
if (FFlag::LuauSolverV2)
|
||||||
{
|
{
|
||||||
Subtyping subtyping{builtinTypes, NotNull{&arena}, NotNull{&normalizer}, NotNull{&typeFunctionRuntime}, NotNull{&ice}};
|
Subtyping subtyping{builtinTypes, NotNull{&arena}, simplifier, NotNull{&normalizer}, NotNull{&typeFunctionRuntime}, NotNull{&ice}};
|
||||||
|
|
||||||
return subtyping.isSubtype(subTy, superTy, scope).isSubtype;
|
return subtyping.isSubtype(subTy, superTy, scope).isSubtype;
|
||||||
}
|
}
|
||||||
|
@ -3491,7 +3506,14 @@ bool isSubtype(TypeId subTy, TypeId superTy, NotNull<Scope> scope, NotNull<Built
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isSubtype(TypePackId subPack, TypePackId superPack, NotNull<Scope> scope, NotNull<BuiltinTypes> builtinTypes, InternalErrorReporter& ice)
|
bool isSubtype(
|
||||||
|
TypePackId subPack,
|
||||||
|
TypePackId superPack,
|
||||||
|
NotNull<Scope> scope,
|
||||||
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
|
InternalErrorReporter& ice
|
||||||
|
)
|
||||||
{
|
{
|
||||||
UnifierSharedState sharedState{&ice};
|
UnifierSharedState sharedState{&ice};
|
||||||
TypeArena arena;
|
TypeArena arena;
|
||||||
|
@ -3504,7 +3526,7 @@ bool isSubtype(TypePackId subPack, TypePackId superPack, NotNull<Scope> scope, N
|
||||||
// Subtyping under DCR is not implemented using unification!
|
// Subtyping under DCR is not implemented using unification!
|
||||||
if (FFlag::LuauSolverV2)
|
if (FFlag::LuauSolverV2)
|
||||||
{
|
{
|
||||||
Subtyping subtyping{builtinTypes, NotNull{&arena}, NotNull{&normalizer}, NotNull{&typeFunctionRuntime}, NotNull{&ice}};
|
Subtyping subtyping{builtinTypes, NotNull{&arena}, simplifier, NotNull{&normalizer}, NotNull{&typeFunctionRuntime}, NotNull{&ice}};
|
||||||
|
|
||||||
return subtyping.isSubtype(subPack, superPack, scope).isSubtype;
|
return subtyping.isSubtype(subPack, superPack, scope).isSubtype;
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,6 +16,7 @@ namespace Luau
|
||||||
OverloadResolver::OverloadResolver(
|
OverloadResolver::OverloadResolver(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<Scope> scope,
|
NotNull<Scope> scope,
|
||||||
|
@ -25,12 +26,13 @@ OverloadResolver::OverloadResolver(
|
||||||
)
|
)
|
||||||
: builtinTypes(builtinTypes)
|
: builtinTypes(builtinTypes)
|
||||||
, arena(arena)
|
, arena(arena)
|
||||||
|
, simplifier(simplifier)
|
||||||
, normalizer(normalizer)
|
, normalizer(normalizer)
|
||||||
, typeFunctionRuntime(typeFunctionRuntime)
|
, typeFunctionRuntime(typeFunctionRuntime)
|
||||||
, scope(scope)
|
, scope(scope)
|
||||||
, ice(reporter)
|
, ice(reporter)
|
||||||
, limits(limits)
|
, limits(limits)
|
||||||
, subtyping({builtinTypes, arena, normalizer, typeFunctionRuntime, ice})
|
, subtyping({builtinTypes, arena, simplifier, normalizer, typeFunctionRuntime, ice})
|
||||||
, callLoc(callLocation)
|
, callLoc(callLocation)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -202,7 +204,7 @@ std::pair<OverloadResolver::Analysis, ErrorVec> OverloadResolver::checkOverload_
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
FunctionGraphReductionResult result = reduceTypeFunctions(
|
FunctionGraphReductionResult result = reduceTypeFunctions(
|
||||||
fnTy, callLoc, TypeFunctionContext{arena, builtinTypes, scope, normalizer, typeFunctionRuntime, ice, limits}, /*force=*/true
|
fnTy, callLoc, TypeFunctionContext{arena, builtinTypes, scope, simplifier, normalizer, typeFunctionRuntime, ice, limits}, /*force=*/true
|
||||||
);
|
);
|
||||||
if (!result.errors.empty())
|
if (!result.errors.empty())
|
||||||
return {OverloadIsNonviable, result.errors};
|
return {OverloadIsNonviable, result.errors};
|
||||||
|
@ -404,9 +406,10 @@ void OverloadResolver::add(Analysis analysis, TypeId ty, ErrorVec&& errors)
|
||||||
|
|
||||||
// we wrap calling the overload resolver in a separate function to reduce overall stack pressure in `solveFunctionCall`.
|
// we wrap calling the overload resolver in a separate function to reduce overall stack pressure in `solveFunctionCall`.
|
||||||
// this limits the lifetime of `OverloadResolver`, a large type, to only as long as it is actually needed.
|
// this limits the lifetime of `OverloadResolver`, a large type, to only as long as it is actually needed.
|
||||||
std::optional<TypeId> selectOverload(
|
static std::optional<TypeId> selectOverload(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<Scope> scope,
|
NotNull<Scope> scope,
|
||||||
|
@ -417,7 +420,8 @@ std::optional<TypeId> selectOverload(
|
||||||
TypePackId argsPack
|
TypePackId argsPack
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
auto resolver = std::make_unique<OverloadResolver>(builtinTypes, arena, 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)
|
||||||
|
@ -432,6 +436,7 @@ std::optional<TypeId> selectOverload(
|
||||||
SolveResult solveFunctionCall(
|
SolveResult solveFunctionCall(
|
||||||
NotNull<TypeArena> arena,
|
NotNull<TypeArena> arena,
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> iceReporter,
|
NotNull<InternalErrorReporter> iceReporter,
|
||||||
|
@ -443,7 +448,7 @@ SolveResult solveFunctionCall(
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
std::optional<TypeId> overloadToUse =
|
std::optional<TypeId> overloadToUse =
|
||||||
selectOverload(builtinTypes, arena, normalizer, typeFunctionRuntime, scope, iceReporter, limits, location, fn, argsPack);
|
selectOverload(builtinTypes, arena, simplifier, normalizer, typeFunctionRuntime, scope, iceReporter, limits, location, fn, argsPack);
|
||||||
if (!overloadToUse)
|
if (!overloadToUse)
|
||||||
return {SolveResult::NoMatchingOverload};
|
return {SolveResult::NoMatchingOverload};
|
||||||
|
|
||||||
|
|
|
@ -211,6 +211,16 @@ void Scope::inheritRefinements(const ScopePtr& childScope)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Scope::shouldWarnGlobal(std::string name) const
|
||||||
|
{
|
||||||
|
for (const Scope* current = this; current; current = current->parent.get())
|
||||||
|
{
|
||||||
|
if (current->globalsToWarn.contains(name))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool subsumesStrict(Scope* left, Scope* right)
|
bool subsumesStrict(Scope* left, Scope* right)
|
||||||
{
|
{
|
||||||
while (right)
|
while (right)
|
||||||
|
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
@ -396,12 +395,14 @@ TypePackId* SubtypingEnvironment::getMappedPackBounds(TypePackId tp)
|
||||||
Subtyping::Subtyping(
|
Subtyping::Subtyping(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
NotNull<TypeArena> typeArena,
|
NotNull<TypeArena> typeArena,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<Normalizer> normalizer,
|
NotNull<Normalizer> normalizer,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<InternalErrorReporter> iceReporter
|
NotNull<InternalErrorReporter> iceReporter
|
||||||
)
|
)
|
||||||
: builtinTypes(builtinTypes)
|
: builtinTypes(builtinTypes)
|
||||||
, arena(typeArena)
|
, arena(typeArena)
|
||||||
|
, simplifier(simplifier)
|
||||||
, normalizer(normalizer)
|
, normalizer(normalizer)
|
||||||
, typeFunctionRuntime(typeFunctionRuntime)
|
, typeFunctionRuntime(typeFunctionRuntime)
|
||||||
, iceReporter(iceReporter)
|
, iceReporter(iceReporter)
|
||||||
|
@ -1472,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));
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1861,7 +1861,7 @@ TypeId Subtyping::makeAggregateType(const Container& container, TypeId orElse)
|
||||||
|
|
||||||
std::pair<TypeId, ErrorVec> Subtyping::handleTypeFunctionReductionResult(const TypeFunctionInstanceType* functionInstance, NotNull<Scope> scope)
|
std::pair<TypeId, ErrorVec> Subtyping::handleTypeFunctionReductionResult(const TypeFunctionInstanceType* functionInstance, NotNull<Scope> scope)
|
||||||
{
|
{
|
||||||
TypeFunctionContext context{arena, builtinTypes, scope, normalizer, typeFunctionRuntime, iceReporter, NotNull{&limits}};
|
TypeFunctionContext context{arena, builtinTypes, scope, simplifier, normalizer, typeFunctionRuntime, iceReporter, NotNull{&limits}};
|
||||||
TypeId function = arena->addType(*functionInstance);
|
TypeId function = arena->addType(*functionInstance);
|
||||||
FunctionGraphReductionResult result = reduceTypeFunctions(function, {}, context, true);
|
FunctionGraphReductionResult result = reduceTypeFunctions(function, {}, context, true);
|
||||||
ErrorVec errors;
|
ErrorVec errors;
|
||||||
|
|
|
@ -9,6 +9,8 @@
|
||||||
#include "Luau/TypeUtils.h"
|
#include "Luau/TypeUtils.h"
|
||||||
#include "Luau/Unifier2.h"
|
#include "Luau/Unifier2.h"
|
||||||
|
|
||||||
|
LUAU_FASTFLAGVARIABLE(LuauDontInPlaceMutateTableType)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -236,6 +238,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))
|
||||||
|
@ -280,6 +284,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 +394,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});
|
||||||
|
|
||||||
|
|
|
@ -386,8 +386,12 @@ public:
|
||||||
}
|
}
|
||||||
AstType* operator()(const NegationType& ntv)
|
AstType* operator()(const NegationType& ntv)
|
||||||
{
|
{
|
||||||
// FIXME: do the same thing we do with ErrorType
|
AstArray<AstTypeOrPack> params;
|
||||||
throw InternalCompilerError("Cannot convert NegationType into AstNode");
|
params.size = 1;
|
||||||
|
params.data = static_cast<AstTypeOrPack*>(allocator->allocate(sizeof(AstType*)));
|
||||||
|
params.data[0] = AstTypeOrPack{Luau::visit(*this, ntv.ty->ty), nullptr};
|
||||||
|
|
||||||
|
return allocator->alloc<AstTypeReference>(Location(), std::nullopt, AstName("negate"), std::nullopt, Location(), true, params);
|
||||||
}
|
}
|
||||||
AstType* operator()(const TypeFunctionInstanceType& tfit)
|
AstType* operator()(const TypeFunctionInstanceType& tfit)
|
||||||
{
|
{
|
||||||
|
|
|
@ -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,12 +26,12 @@
|
||||||
#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
|
||||||
{
|
{
|
||||||
|
@ -175,7 +174,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)
|
||||||
|
@ -268,6 +267,7 @@ struct InternalTypeFunctionFinder : TypeOnceVisitor
|
||||||
|
|
||||||
void check(
|
void check(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<UnifierSharedState> unifierState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
NotNull<TypeCheckLimits> limits,
|
NotNull<TypeCheckLimits> limits,
|
||||||
|
@ -278,7 +278,7 @@ void check(
|
||||||
{
|
{
|
||||||
LUAU_TIMETRACE_SCOPE("check", "Typechecking");
|
LUAU_TIMETRACE_SCOPE("check", "Typechecking");
|
||||||
|
|
||||||
TypeChecker2 typeChecker{builtinTypes, typeFunctionRuntime, unifierState, limits, logger, &sourceModule, module};
|
TypeChecker2 typeChecker{builtinTypes, simplifier, typeFunctionRuntime, unifierState, limits, logger, &sourceModule, module};
|
||||||
|
|
||||||
typeChecker.visit(sourceModule.root);
|
typeChecker.visit(sourceModule.root);
|
||||||
|
|
||||||
|
@ -295,6 +295,7 @@ void check(
|
||||||
|
|
||||||
TypeChecker2::TypeChecker2(
|
TypeChecker2::TypeChecker2(
|
||||||
NotNull<BuiltinTypes> builtinTypes,
|
NotNull<BuiltinTypes> builtinTypes,
|
||||||
|
NotNull<Simplifier> simplifier,
|
||||||
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
NotNull<TypeFunctionRuntime> typeFunctionRuntime,
|
||||||
NotNull<UnifierSharedState> unifierState,
|
NotNull<UnifierSharedState> unifierState,
|
||||||
NotNull<TypeCheckLimits> limits,
|
NotNull<TypeCheckLimits> limits,
|
||||||
|
@ -303,6 +304,7 @@ TypeChecker2::TypeChecker2(
|
||||||
Module* module
|
Module* module
|
||||||
)
|
)
|
||||||
: builtinTypes(builtinTypes)
|
: builtinTypes(builtinTypes)
|
||||||
|
, simplifier(simplifier)
|
||||||
, typeFunctionRuntime(typeFunctionRuntime)
|
, typeFunctionRuntime(typeFunctionRuntime)
|
||||||
, logger(logger)
|
, logger(logger)
|
||||||
, limits(limits)
|
, limits(limits)
|
||||||
|
@ -310,7 +312,7 @@ TypeChecker2::TypeChecker2(
|
||||||
, sourceModule(sourceModule)
|
, sourceModule(sourceModule)
|
||||||
, module(module)
|
, module(module)
|
||||||
, normalizer{&module->internalTypes, builtinTypes, unifierState, /* cacheInhabitance */ true}
|
, normalizer{&module->internalTypes, builtinTypes, unifierState, /* cacheInhabitance */ true}
|
||||||
, _subtyping{builtinTypes, NotNull{&module->internalTypes}, NotNull{&normalizer}, typeFunctionRuntime, NotNull{unifierState->iceHandler}}
|
, _subtyping{builtinTypes, NotNull{&module->internalTypes}, simplifier, NotNull{&normalizer}, typeFunctionRuntime, NotNull{unifierState->iceHandler}}
|
||||||
, subtyping(&_subtyping)
|
, subtyping(&_subtyping)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
@ -492,7 +494,9 @@ TypeId TypeChecker2::checkForTypeFunctionInhabitance(TypeId instance, Location l
|
||||||
reduceTypeFunctions(
|
reduceTypeFunctions(
|
||||||
instance,
|
instance,
|
||||||
location,
|
location,
|
||||||
TypeFunctionContext{NotNull{&module->internalTypes}, builtinTypes, stack.back(), NotNull{&normalizer}, typeFunctionRuntime, ice, limits},
|
TypeFunctionContext{
|
||||||
|
NotNull{&module->internalTypes}, builtinTypes, stack.back(), simplifier, NotNull{&normalizer}, typeFunctionRuntime, ice, limits
|
||||||
|
},
|
||||||
true
|
true
|
||||||
)
|
)
|
||||||
.errors;
|
.errors;
|
||||||
|
@ -501,7 +505,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
|
||||||
|
@ -551,7 +555,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)
|
||||||
|
@ -559,7 +563,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);
|
||||||
|
@ -567,7 +571,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});
|
||||||
|
@ -591,7 +595,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();
|
||||||
|
|
||||||
|
@ -1014,7 +1018,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};
|
||||||
|
|
||||||
|
@ -1349,8 +1354,18 @@ void TypeChecker2::visit(AstExprGlobal* expr)
|
||||||
{
|
{
|
||||||
NotNull<Scope> scope = stack.back();
|
NotNull<Scope> scope = stack.back();
|
||||||
if (!scope->lookup(expr->name))
|
if (!scope->lookup(expr->name))
|
||||||
|
{
|
||||||
reportError(UnknownSymbol{expr->name.value, UnknownSymbol::Binding}, expr->location);
|
reportError(UnknownSymbol{expr->name.value, UnknownSymbol::Binding}, expr->location);
|
||||||
}
|
}
|
||||||
|
else if (FFlag::InferGlobalTypes)
|
||||||
|
{
|
||||||
|
if (scope->shouldWarnGlobal(expr->name.value) && !warnedGlobals.contains(expr->name.value))
|
||||||
|
{
|
||||||
|
reportError(UnknownSymbol{expr->name.value, UnknownSymbol::Binding}, expr->location);
|
||||||
|
warnedGlobals.insert(expr->name.value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void TypeChecker2::visit(AstExprVarargs* expr)
|
void TypeChecker2::visit(AstExprVarargs* expr)
|
||||||
{
|
{
|
||||||
|
@ -1437,9 +1452,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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1448,6 +1464,7 @@ void TypeChecker2::visitCall(AstExprCall* call)
|
||||||
OverloadResolver resolver{
|
OverloadResolver resolver{
|
||||||
builtinTypes,
|
builtinTypes,
|
||||||
NotNull{&module->internalTypes},
|
NotNull{&module->internalTypes},
|
||||||
|
simplifier,
|
||||||
NotNull{&normalizer},
|
NotNull{&normalizer},
|
||||||
typeFunctionRuntime,
|
typeFunctionRuntime,
|
||||||
NotNull{stack.back()},
|
NotNull{stack.back()},
|
||||||
|
@ -1545,7 +1562,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))
|
||||||
{
|
{
|
||||||
|
@ -2089,7 +2106,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));
|
||||||
|
@ -2341,7 +2361,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));
|
||||||
|
@ -2403,6 +2424,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)
|
||||||
|
@ -3024,7 +3047,7 @@ PropertyType TypeChecker2::hasIndexTypeFromType(
|
||||||
{
|
{
|
||||||
TypeId indexType = follow(tt->indexer->indexType);
|
TypeId indexType = follow(tt->indexer->indexType);
|
||||||
TypeId givenType = module->internalTypes.addType(SingletonType{StringSingleton{prop}});
|
TypeId givenType = module->internalTypes.addType(SingletonType{StringSingleton{prop}});
|
||||||
if (isSubtype(givenType, indexType, NotNull{module->getModuleScope().get()}, builtinTypes, *ice))
|
if (isSubtype(givenType, indexType, NotNull{module->getModuleScope().get()}, builtinTypes, simplifier, *ice))
|
||||||
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
return {NormalizationResult::True, {tt->indexer->indexResultType}};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -14,9 +14,9 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
LUAU_DYNAMIC_FASTINT(LuauTypeFunctionSerdeIterationLimit)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixRegister)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixInner)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunFixNoReadWrite)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunGenerics)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunThreadBuffer)
|
LUAU_FASTFLAGVARIABLE(LuauUserTypeFunCloneTail)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -134,11 +134,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";
|
||||||
|
@ -160,6 +158,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 (FFlag::LuauUserTypeFunGenerics && 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");
|
||||||
|
@ -266,6 +266,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)
|
||||||
|
@ -413,10 +427,21 @@ 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 (FFlag::LuauUserTypeFunFixInner)
|
||||||
|
{
|
||||||
|
if (auto tfnt = get<TypeFunctionNegationType>(self); tfnt)
|
||||||
|
allocTypeUserData(L, tfnt->type->type);
|
||||||
|
else
|
||||||
|
luaL_error(L, "type.inner: cannot call inner method on non-negation type: `%s` type", getTag(L, self).c_str());
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
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());
|
||||||
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -657,10 +682,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;
|
||||||
}
|
}
|
||||||
|
@ -697,10 +720,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;
|
||||||
}
|
}
|
||||||
|
@ -768,9 +789,159 @@ 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)
|
||||||
// Returns the type instance representing a function
|
{
|
||||||
static int createFunction(lua_State* L)
|
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");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int createFunction_DEPRECATED(lua_State* L)
|
||||||
{
|
{
|
||||||
int argumentCount = lua_gettop(L);
|
int argumentCount = lua_gettop(L);
|
||||||
if (argumentCount > 2)
|
if (argumentCount > 2)
|
||||||
|
@ -858,7 +1029,62 @@ static int createFunction(lua_State* L)
|
||||||
else if (!lua_isnoneornil(L, 2))
|
else if (!lua_isnoneornil(L, 2))
|
||||||
luaL_typeerrorL(L, 2, "table");
|
luaL_typeerrorL(L, 2, "table");
|
||||||
|
|
||||||
allocTypeUserData(L, TypeFunctionFunctionType{argTypes, retTypes});
|
allocTypeUserData(L, TypeFunctionFunctionType{{}, {}, argTypes, retTypes});
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Luau: `types.newfunction(parameters: {head: {type}?, tail: type?}, returns: {head: {type}?, tail: type?}, generics: {type}?) -> type`
|
||||||
|
// Returns the type instance representing a function
|
||||||
|
static int createFunction(lua_State* L)
|
||||||
|
{
|
||||||
|
int argumentCount = lua_gettop(L);
|
||||||
|
if (argumentCount > 3)
|
||||||
|
luaL_error(L, "types.newfunction: expected 0-3 arguments, but got %d", argumentCount);
|
||||||
|
|
||||||
|
TypeFunctionTypePackId argTypes = nullptr;
|
||||||
|
|
||||||
|
if (lua_istable(L, 1))
|
||||||
|
{
|
||||||
|
lua_getfield(L, 1, "head");
|
||||||
|
lua_getfield(L, 1, "tail");
|
||||||
|
|
||||||
|
argTypes = getTypePack(L, -2, -1);
|
||||||
|
|
||||||
|
lua_pop(L, 2);
|
||||||
|
}
|
||||||
|
else if (!lua_isnoneornil(L, 1))
|
||||||
|
{
|
||||||
|
luaL_typeerrorL(L, 1, "table");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
||||||
|
}
|
||||||
|
|
||||||
|
TypeFunctionTypePackId retTypes = nullptr;
|
||||||
|
|
||||||
|
if (lua_istable(L, 2))
|
||||||
|
{
|
||||||
|
lua_getfield(L, 2, "head");
|
||||||
|
lua_getfield(L, 2, "tail");
|
||||||
|
|
||||||
|
retTypes = getTypePack(L, -2, -1);
|
||||||
|
|
||||||
|
lua_pop(L, 2);
|
||||||
|
}
|
||||||
|
else if (!lua_isnoneornil(L, 2))
|
||||||
|
{
|
||||||
|
luaL_typeerrorL(L, 2, "table");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{});
|
||||||
|
}
|
||||||
|
|
||||||
|
auto [genericTypes, genericPacks] = getGenerics(L, 3, "types.newfunction");
|
||||||
|
|
||||||
|
allocTypeUserData(L, TypeFunctionFunctionType{std::move(genericTypes), std::move(genericPacks), argTypes, retTypes});
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -876,6 +1102,12 @@ 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());
|
||||||
|
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
tfft->argTypes = getTypePack(L, 2, 3);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
std::vector<TypeFunctionTypeId> head{};
|
std::vector<TypeFunctionTypeId> head{};
|
||||||
if (lua_istable(L, 2))
|
if (lua_istable(L, 2))
|
||||||
{
|
{
|
||||||
|
@ -908,6 +1140,7 @@ static int setFunctionParameters(lua_State* L)
|
||||||
tfft->argTypes = *tail;
|
tfft->argTypes = *tail;
|
||||||
else // Make argTypes a type pack
|
else // Make argTypes a type pack
|
||||||
tfft->argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
tfft->argTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -925,6 +1158,12 @@ 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 (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
pushTypePack(L, tfft->argTypes);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (auto tftp = get<TypeFunctionTypePack>(tfft->argTypes))
|
if (auto tftp = get<TypeFunctionTypePack>(tfft->argTypes))
|
||||||
{
|
{
|
||||||
int size = 0;
|
int size = 0;
|
||||||
|
@ -971,6 +1210,8 @@ static int getFunctionParameters(lua_State* L)
|
||||||
}
|
}
|
||||||
|
|
||||||
lua_createtable(L, 0, 0);
|
lua_createtable(L, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -987,6 +1228,12 @@ 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());
|
||||||
|
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
tfft->retTypes = getTypePack(L, 2, 3);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
std::vector<TypeFunctionTypeId> head{};
|
std::vector<TypeFunctionTypeId> head{};
|
||||||
if (lua_istable(L, 2))
|
if (lua_istable(L, 2))
|
||||||
{
|
{
|
||||||
|
@ -1019,6 +1266,7 @@ static int setFunctionReturns(lua_State* L)
|
||||||
tfft->retTypes = *tail;
|
tfft->retTypes = *tail;
|
||||||
else // Make retTypes a type pack
|
else // Make retTypes a type pack
|
||||||
tfft->retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
tfft->retTypes = allocateTypeFunctionTypePack(L, TypeFunctionTypePack{head, tail});
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1036,6 +1284,12 @@ 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 (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
pushTypePack(L, tfft->retTypes);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (auto tftp = get<TypeFunctionTypePack>(tfft->retTypes))
|
if (auto tftp = get<TypeFunctionTypePack>(tfft->retTypes))
|
||||||
{
|
{
|
||||||
int size = 0;
|
int size = 0;
|
||||||
|
@ -1082,6 +1336,57 @@ static int getFunctionReturns(lua_State* L)
|
||||||
}
|
}
|
||||||
|
|
||||||
lua_createtable(L, 0, 0);
|
lua_createtable(L, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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)
|
||||||
|
{
|
||||||
|
allocTypeUserData(L, el->type);
|
||||||
|
lua_rawseti(L, -2, pos++);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const auto& el : tfft->genericPacks)
|
||||||
|
{
|
||||||
|
auto gty = get<TypeFunctionGenericTypePack>(el);
|
||||||
|
LUAU_ASSERT(gty);
|
||||||
|
allocTypeUserData(L, TypeFunctionGenericType{gty->isNamed, true, gty->name});
|
||||||
|
lua_rawseti(L, -2, pos++);
|
||||||
|
}
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1107,6 +1412,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)
|
||||||
|
@ -1376,7 +1711,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)
|
||||||
{
|
{
|
||||||
|
@ -1408,8 +1743,6 @@ static int isEqualToType(lua_State* L)
|
||||||
|
|
||||||
void registerTypesLibrary(lua_State* L)
|
void registerTypesLibrary(lua_State* L)
|
||||||
{
|
{
|
||||||
LUAU_ASSERT(FFlag::LuauUserTypeFunFixRegister);
|
|
||||||
|
|
||||||
luaL_Reg fields[] = {
|
luaL_Reg fields[] = {
|
||||||
{"unknown", createUnknown},
|
{"unknown", createUnknown},
|
||||||
{"never", createNever},
|
{"never", createNever},
|
||||||
|
@ -1417,8 +1750,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}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1428,8 +1761,9 @@ void registerTypesLibrary(lua_State* L)
|
||||||
{"unionof", createUnion},
|
{"unionof", createUnion},
|
||||||
{"intersectionof", createIntersection},
|
{"intersectionof", createIntersection},
|
||||||
{"newtable", createTable},
|
{"newtable", createTable},
|
||||||
{"newfunction", createFunction},
|
{"newfunction", FFlag::LuauUserTypeFunGenerics ? createFunction : createFunction_DEPRECATED},
|
||||||
{"copy", deepCopy},
|
{"copy", deepCopy},
|
||||||
|
{FFlag::LuauUserTypeFunGenerics ? "generic" : nullptr, FFlag::LuauUserTypeFunGenerics ? createGeneric : nullptr},
|
||||||
|
|
||||||
{nullptr, nullptr}
|
{nullptr, nullptr}
|
||||||
};
|
};
|
||||||
|
@ -1463,8 +1797,6 @@ static int typeUserdataIndex(lua_State* L)
|
||||||
}
|
}
|
||||||
|
|
||||||
void registerTypeUserData(lua_State* L)
|
void registerTypeUserData(lua_State* L)
|
||||||
{
|
|
||||||
if (FFlag::LuauUserTypeFunFixRegister)
|
|
||||||
{
|
{
|
||||||
luaL_Reg typeUserdataMethods[] = {
|
luaL_Reg typeUserdataMethods[] = {
|
||||||
{"is", checkTag},
|
{"is", checkTag},
|
||||||
|
@ -1496,6 +1828,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},
|
||||||
|
@ -1503,6 +1837,14 @@ void registerTypeUserData(lua_State* L)
|
||||||
// Class type methods
|
// Class type methods
|
||||||
{"parent", getClassParent},
|
{"parent", getClassParent},
|
||||||
|
|
||||||
|
// Function type methods (cont.)
|
||||||
|
{FFlag::LuauUserTypeFunGenerics ? "setgenerics" : nullptr, FFlag::LuauUserTypeFunGenerics ? setFunctionGenerics : nullptr},
|
||||||
|
{FFlag::LuauUserTypeFunGenerics ? "generics" : nullptr, FFlag::LuauUserTypeFunGenerics ? getFunctionGenerics : nullptr},
|
||||||
|
|
||||||
|
// Generic type methods
|
||||||
|
{FFlag::LuauUserTypeFunGenerics ? "name" : nullptr, FFlag::LuauUserTypeFunGenerics ? getGenericName : nullptr},
|
||||||
|
{FFlag::LuauUserTypeFunGenerics ? "ispack" : nullptr, FFlag::LuauUserTypeFunGenerics ? getGenericIsPack : nullptr},
|
||||||
|
|
||||||
{nullptr, nullptr}
|
{nullptr, nullptr}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1525,109 +1867,40 @@ void registerTypeUserData(lua_State* L)
|
||||||
|
|
||||||
lua_setreadonly(L, -1, true);
|
lua_setreadonly(L, -1, true);
|
||||||
lua_pop(L, 1);
|
lua_pop(L, 1);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// List of fields for type userdata
|
|
||||||
luaL_Reg typeUserdataFields[] = {
|
|
||||||
{"unknown", createUnknown},
|
|
||||||
{"never", createNever},
|
|
||||||
{"any", createAny},
|
|
||||||
{"boolean", createBoolean},
|
|
||||||
{"number", createNumber},
|
|
||||||
{"string", createString},
|
|
||||||
{nullptr, nullptr}
|
|
||||||
};
|
|
||||||
|
|
||||||
// List of methods for type userdata
|
|
||||||
luaL_Reg typeUserdataMethods[] = {
|
|
||||||
{"singleton", createSingleton},
|
|
||||||
{"negationof", createNegation},
|
|
||||||
{"unionof", createUnion},
|
|
||||||
{"intersectionof", createIntersection},
|
|
||||||
{"newtable", createTable},
|
|
||||||
{"newfunction", createFunction},
|
|
||||||
{"copy", deepCopy},
|
|
||||||
|
|
||||||
// Common methods
|
|
||||||
{"is", checkTag},
|
|
||||||
|
|
||||||
// Negation type methods
|
|
||||||
{"inner", getNegatedValue},
|
|
||||||
|
|
||||||
// Singleton type methods
|
|
||||||
{"value", getSingletonValue},
|
|
||||||
|
|
||||||
// Table type methods
|
|
||||||
{"setproperty", setTableProp},
|
|
||||||
{"setreadproperty", setReadTableProp},
|
|
||||||
{"setwriteproperty", setWriteTableProp},
|
|
||||||
{"readproperty", readTableProp},
|
|
||||||
{"writeproperty", writeTableProp},
|
|
||||||
{"properties", getProps},
|
|
||||||
{"setindexer", setTableIndexer},
|
|
||||||
{"setreadindexer", setTableReadIndexer},
|
|
||||||
{"setwriteindexer", setTableWriteIndexer},
|
|
||||||
{"indexer", getIndexer},
|
|
||||||
{"readindexer", getReadIndexer},
|
|
||||||
{"writeindexer", getWriteIndexer},
|
|
||||||
{"setmetatable", setTableMetatable},
|
|
||||||
{"metatable", getMetatable},
|
|
||||||
|
|
||||||
// Function type methods
|
|
||||||
{"setparameters", setFunctionParameters},
|
|
||||||
{"parameters", getFunctionParameters},
|
|
||||||
{"setreturns", setFunctionReturns},
|
|
||||||
{"returns", getFunctionReturns},
|
|
||||||
|
|
||||||
// Union and Intersection type methods
|
|
||||||
{"components", getComponents},
|
|
||||||
|
|
||||||
// Class type methods
|
|
||||||
{"parent", getClassParent},
|
|
||||||
{"indexer", getIndexer},
|
|
||||||
{nullptr, nullptr}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Create and register metatable for type userdata
|
|
||||||
luaL_newmetatable(L, "type");
|
|
||||||
|
|
||||||
// Protect metatable from being fetched.
|
|
||||||
lua_pushstring(L, "The metatable is locked");
|
|
||||||
lua_setfield(L, -2, "__metatable");
|
|
||||||
|
|
||||||
// Set type userdata metatable's __eq to type_equals()
|
|
||||||
lua_pushcfunction(L, isEqualToType, "__eq");
|
|
||||||
lua_setfield(L, -2, "__eq");
|
|
||||||
|
|
||||||
// Set type userdata metatable's __index to itself
|
|
||||||
lua_pushvalue(L, -1); // Push a copy of type userdata metatable
|
|
||||||
lua_setfield(L, -2, "__index");
|
|
||||||
|
|
||||||
luaL_register(L, nullptr, typeUserdataMethods);
|
|
||||||
|
|
||||||
// Set fields for type userdata
|
|
||||||
for (luaL_Reg* l = typeUserdataFields; l->name; l++)
|
|
||||||
{
|
|
||||||
l->func(L);
|
|
||||||
lua_setfield(L, -2, l->name);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set types library as a global name "types"
|
|
||||||
lua_setglobal(L, "types");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sets up a destructor for the type userdata.
|
// Sets up a destructor for the type userdata.
|
||||||
lua_setuserdatadtor(L, kTypeUserdataTag, deallocTypeUserData);
|
lua_setuserdatadtor(L, kTypeUserdataTag, deallocTypeUserData);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Used to redirect all the removed global functions to say "this function is unsupported"
|
// Used to redirect all the removed global functions to say "this function is unsupported"
|
||||||
int unsupportedFunction(lua_State* L)
|
static int unsupportedFunction(lua_State* L)
|
||||||
{
|
{
|
||||||
luaL_errorL(L, "this function is not supported in type functions");
|
luaL_errorL(L, "this function is not supported in type functions");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int print(lua_State* L)
|
||||||
|
{
|
||||||
|
std::string result;
|
||||||
|
|
||||||
|
int n = lua_gettop(L);
|
||||||
|
for (int i = 1; i <= n; i++)
|
||||||
|
{
|
||||||
|
size_t l = 0;
|
||||||
|
const char* s = luaL_tolstring(L, i, &l); // convert to string using __tostring et al
|
||||||
|
if (i > 1)
|
||||||
|
result.append('\t', 1);
|
||||||
|
result.append(s, l);
|
||||||
|
lua_pop(L, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
auto ctx = getTypeFunctionRuntime(L);
|
||||||
|
|
||||||
|
ctx->messages.push_back(std::move(result));
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
// Add libraries / globals for type function environment
|
// Add libraries / globals for type function environment
|
||||||
void setTypeFunctionEnvironment(lua_State* L)
|
void setTypeFunctionEnvironment(lua_State* L)
|
||||||
{
|
{
|
||||||
|
@ -1660,12 +1933,15 @@ void setTypeFunctionEnvironment(lua_State* L)
|
||||||
lua_pop(L, 1);
|
lua_pop(L, 1);
|
||||||
|
|
||||||
// Remove certain global functions from the base library
|
// Remove certain global functions from the base library
|
||||||
static const std::string 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)
|
||||||
{
|
{
|
||||||
lua_pushcfunction(L, unsupportedFunction, "Removing global function from type function environment");
|
lua_pushcfunction(L, unsupportedFunction, name);
|
||||||
lua_setglobal(L, name.c_str());
|
lua_setglobal(L, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
lua_pushcfunction(L, print, "print");
|
||||||
|
lua_setglobal(L, "print");
|
||||||
}
|
}
|
||||||
|
|
||||||
void resetTypeFunctionState(lua_State* L)
|
void resetTypeFunctionState(lua_State* L)
|
||||||
|
@ -1821,6 +2097,27 @@ bool areEqual(SeenSet& seen, const TypeFunctionFunctionType& lhs, const TypeFunc
|
||||||
if (seenSetContains(seen, &lhs, &rhs))
|
if (seenSetContains(seen, &lhs, &rhs))
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
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;
|
||||||
|
|
||||||
|
@ -1921,6 +2218,16 @@ bool areEqual(SeenSet& seen, const TypeFunctionType& lhs, const TypeFunctionType
|
||||||
return areEqual(seen, *lf, *rf);
|
return areEqual(seen, *lf, *rf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
{
|
||||||
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1967,6 +2274,16 @@ bool areEqual(SeenSet& seen, const TypeFunctionTypePackVar& lhs, const TypeFunct
|
||||||
return areEqual(seen, *lv, *rv);
|
return areEqual(seen, *lv, *rv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
{
|
||||||
|
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2156,11 +2473,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:
|
||||||
|
@ -2191,10 +2506,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); FFlag::LuauUserTypeFunGenerics && g)
|
||||||
|
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);
|
||||||
|
@ -2212,6 +2531,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); gPack && FFlag::LuauUserTypeFunGenerics)
|
||||||
|
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);
|
||||||
|
@ -2242,6 +2565,9 @@ 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)};
|
||||||
|
FFlag::LuauUserTypeFunGenerics && 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
|
||||||
}
|
}
|
||||||
|
@ -2253,6 +2579,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)};
|
||||||
|
FFlag::LuauUserTypeFunGenerics && 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
|
||||||
}
|
}
|
||||||
|
@ -2333,6 +2662,17 @@ private:
|
||||||
|
|
||||||
void cloneChildren(TypeFunctionFunctionType* f1, TypeFunctionFunctionType* f2)
|
void cloneChildren(TypeFunctionFunctionType* f1, TypeFunctionFunctionType* f2)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
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);
|
||||||
}
|
}
|
||||||
|
@ -2342,16 +2682,32 @@ 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 (FFlag::LuauUserTypeFunCloneTail)
|
||||||
|
{
|
||||||
|
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,7 @@
|
||||||
// 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_FASTFLAGVARIABLE(LuauUserTypeFunFixMetatable)
|
LUAU_FASTFLAG(LuauUserTypeFunGenerics)
|
||||||
LUAU_FASTFLAG(LuauUserTypeFunThreadBuffer)
|
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
@ -161,26 +160,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:
|
||||||
|
@ -222,13 +205,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); FFlag::LuauUserTypeFunGenerics && g)
|
||||||
|
{
|
||||||
|
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());
|
||||||
|
@ -253,6 +245,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); FFlag::LuauUserTypeFunGenerics && gPack)
|
||||||
|
{
|
||||||
|
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());
|
||||||
|
@ -290,6 +291,9 @@ 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)};
|
||||||
|
FFlag::LuauUserTypeFunGenerics && 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());
|
||||||
|
@ -303,6 +307,9 @@ 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)};
|
||||||
|
FFlag::LuauUserTypeFunGenerics && 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());
|
||||||
|
@ -382,28 +389,27 @@ private:
|
||||||
}
|
}
|
||||||
|
|
||||||
void serializeChildren(const MetatableType* m1, TypeFunctionTableType* m2)
|
void serializeChildren(const MetatableType* m1, TypeFunctionTableType* m2)
|
||||||
{
|
|
||||||
if (FFlag::LuauUserTypeFunFixMetatable)
|
|
||||||
{
|
{
|
||||||
// Serialize main part of the metatable immediately
|
// Serialize main part of the metatable immediately
|
||||||
if (auto tableTy = get<TableType>(m1->table))
|
if (auto tableTy = get<TableType>(m1->table))
|
||||||
serializeChildren(tableTy, m2);
|
serializeChildren(tableTy, m2);
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
auto tmpTable = get<TypeFunctionTableType>(shallowSerialize(m1->table));
|
|
||||||
if (!tmpTable)
|
|
||||||
state->ctx->ice->ice("Serializing user defined type function arguments: metatable's table is not a TableType");
|
|
||||||
|
|
||||||
m2->props = tmpTable->props;
|
|
||||||
m2->indexer = tmpTable->indexer;
|
|
||||||
}
|
|
||||||
|
|
||||||
m2->metatable = shallowSerialize(m1->metatable);
|
m2->metatable = shallowSerialize(m1->metatable);
|
||||||
}
|
}
|
||||||
|
|
||||||
void serializeChildren(const FunctionType* f1, TypeFunctionFunctionType* f2)
|
void serializeChildren(const FunctionType* f1, TypeFunctionFunctionType* f2)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
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);
|
||||||
}
|
}
|
||||||
|
@ -433,6 +439,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)
|
||||||
|
@ -446,6 +457,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
|
||||||
|
@ -466,6 +496,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
|
||||||
|
|
||||||
|
@ -477,7 +516,9 @@ public:
|
||||||
, typeFunctionRuntime(state->ctx->typeFunctionRuntime)
|
, typeFunctionRuntime(state->ctx->typeFunctionRuntime)
|
||||||
, queue({})
|
, queue({})
|
||||||
, types({})
|
, types({})
|
||||||
, packs({}){};
|
, packs({})
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
TypeId deserialize(TypeFunctionTypeId ty)
|
TypeId deserialize(TypeFunctionTypeId ty)
|
||||||
{
|
{
|
||||||
|
@ -531,6 +572,16 @@ private:
|
||||||
queue.pop_back();
|
queue.pop_back();
|
||||||
|
|
||||||
deserializeChildren(tfti, ty);
|
deserializeChildren(tfti, ty);
|
||||||
|
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
// 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();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -563,6 +614,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))
|
||||||
|
@ -587,16 +653,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");
|
||||||
|
@ -642,6 +702,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); FFlag::LuauUserTypeFunGenerics && g)
|
||||||
|
{
|
||||||
|
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");
|
||||||
|
|
||||||
|
@ -658,11 +745,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); FFlag::LuauUserTypeFunGenerics && gPack)
|
||||||
|
{
|
||||||
|
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);
|
||||||
|
@ -697,6 +809,9 @@ 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)};
|
||||||
|
FFlag::LuauUserTypeFunGenerics && 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");
|
||||||
}
|
}
|
||||||
|
@ -708,6 +823,9 @@ 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)};
|
||||||
|
FFlag::LuauUserTypeFunGenerics && 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");
|
||||||
}
|
}
|
||||||
|
@ -791,6 +909,64 @@ private:
|
||||||
|
|
||||||
void deserializeChildren(TypeFunctionFunctionType* f2, FunctionType* f1)
|
void deserializeChildren(TypeFunctionFunctionType* f2, FunctionType* f1)
|
||||||
{
|
{
|
||||||
|
if (FFlag::LuauUserTypeFunGenerics)
|
||||||
|
{
|
||||||
|
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);
|
||||||
|
|
||||||
|
@ -803,6 +979,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)
|
||||||
|
@ -816,6 +997,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)
|
||||||
|
|
|
@ -32,7 +32,9 @@ LUAU_FASTINTVARIABLE(LuauVisitRecursionLimit, 500)
|
||||||
LUAU_FASTFLAG(LuauKnowsTheDataModel3)
|
LUAU_FASTFLAG(LuauKnowsTheDataModel3)
|
||||||
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeDuringUnification)
|
LUAU_FASTFLAGVARIABLE(DebugLuauFreezeDuringUnification)
|
||||||
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
LUAU_FASTFLAG(LuauInstantiateInSubtyping)
|
||||||
LUAU_FASTFLAGVARIABLE(LuauMetatableFollow)
|
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;
|
||||||
|
@ -2866,7 +2873,7 @@ TypeId TypeChecker::checkRelationalOperation(
|
||||||
std::optional<TypeId> metamethod = findMetatableEntry(lhsType, metamethodName, expr.location, /* addErrors= */ true);
|
std::optional<TypeId> metamethod = findMetatableEntry(lhsType, metamethodName, expr.location, /* addErrors= */ true);
|
||||||
if (metamethod)
|
if (metamethod)
|
||||||
{
|
{
|
||||||
if (const FunctionType* ftv = get<FunctionType>(FFlag::LuauMetatableFollow ? follow(*metamethod) : *metamethod))
|
if (const FunctionType* ftv = get<FunctionType>(follow(*metamethod)))
|
||||||
{
|
{
|
||||||
if (isEquality)
|
if (isEquality)
|
||||||
{
|
{
|
||||||
|
@ -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));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5205,6 +5212,13 @@ LUAU_NOINLINE void TypeChecker::reportErrorCodeTooComplex(const Location& locati
|
||||||
ScopePtr TypeChecker::childFunctionScope(const ScopePtr& parent, const Location& location, int subLevel)
|
ScopePtr TypeChecker::childFunctionScope(const ScopePtr& parent, const Location& location, int subLevel)
|
||||||
{
|
{
|
||||||
ScopePtr scope = std::make_shared<Scope>(parent, subLevel);
|
ScopePtr scope = std::make_shared<Scope>(parent, subLevel);
|
||||||
|
if (FFlag::LuauOldSolverCreatesChildScopePointers)
|
||||||
|
{
|
||||||
|
scope->location = location;
|
||||||
|
scope->returnType = parent->returnType;
|
||||||
|
parent->children.emplace_back(scope.get());
|
||||||
|
}
|
||||||
|
|
||||||
currentModule->scopes.push_back(std::make_pair(location, scope));
|
currentModule->scopes.push_back(std::make_pair(location, scope));
|
||||||
return scope;
|
return scope;
|
||||||
}
|
}
|
||||||
|
@ -5215,6 +5229,12 @@ ScopePtr TypeChecker::childScope(const ScopePtr& parent, const Location& locatio
|
||||||
ScopePtr scope = std::make_shared<Scope>(parent);
|
ScopePtr scope = std::make_shared<Scope>(parent);
|
||||||
scope->level = parent->level;
|
scope->level = parent->level;
|
||||||
scope->varargPack = parent->varargPack;
|
scope->varargPack = parent->varargPack;
|
||||||
|
if (FFlag::LuauOldSolverCreatesChildScopePointers)
|
||||||
|
{
|
||||||
|
scope->location = location;
|
||||||
|
scope->returnType = parent->returnType;
|
||||||
|
parent->children.emplace_back(scope.get());
|
||||||
|
}
|
||||||
|
|
||||||
currentModule->scopes.push_back(std::make_pair(location, scope));
|
currentModule->scopes.push_back(std::make_pair(location, scope));
|
||||||
return scope;
|
return scope;
|
||||||
|
@ -5260,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)
|
||||||
|
@ -5705,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));
|
||||||
|
@ -5713,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);
|
||||||
|
|
|
@ -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);
|
||||||
|
@ -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
|
||||||
|
|
|
@ -1204,6 +1204,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:
|
||||||
|
@ -1470,6 +1482,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));
|
||||||
|
|
334
Ast/include/Luau/Cst.h
Normal file
334
Ast/include/Luau/Cst.h
Normal file
|
@ -0,0 +1,334 @@
|
||||||
|
// 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 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;
|
||||||
};
|
};
|
||||||
|
@ -230,17 +238,6 @@ private:
|
||||||
bool skipComments;
|
bool skipComments;
|
||||||
bool readNames;
|
bool readNames;
|
||||||
|
|
||||||
// This offset represents a column offset to be applied to any positions created by the lexer until the next new line.
|
|
||||||
// For example:
|
|
||||||
// local x = 4
|
|
||||||
// local y = 5
|
|
||||||
// If we start lexing from the position of `l` in `local x = 4`, the line number will be 1, and the column will be 4
|
|
||||||
// However, because the lexer calculates line offsets by 'index in source buffer where there is a newline', the column
|
|
||||||
// count will start at 0. For this reason, for just the first line, we'll need to store the offset.
|
|
||||||
unsigned int lexResumeOffset;
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
enum class BraceType
|
enum class BraceType
|
||||||
{
|
{
|
||||||
InterpolatedString,
|
InterpolatedString,
|
||||||
|
|
|
@ -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,7 +206,17 @@ 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(
|
||||||
|
@ -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
|
||||||
|
@ -280,9 +297,13 @@ private:
|
||||||
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
|
||||||
|
|
|
@ -1091,6 +1091,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 +1163,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)
|
||||||
|
|
169
Ast/src/Cst.cpp
Normal file
169
Ast/src/Cst.cpp
Normal file
|
@ -0,0 +1,169 @@
|
||||||
|
// 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)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
|
@ -8,7 +8,9 @@
|
||||||
|
|
||||||
#include <limits.h>
|
#include <limits.h>
|
||||||
|
|
||||||
LUAU_FASTFLAGVARIABLE(LexerResumesFromPosition)
|
LUAU_FASTFLAGVARIABLE(LexerResumesFromPosition2)
|
||||||
|
LUAU_FASTFLAGVARIABLE(LexerFixInterpStringStart)
|
||||||
|
|
||||||
namespace Luau
|
namespace Luau
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@ -304,20 +306,51 @@ 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)
|
||||||
, offset(0)
|
, offset(0)
|
||||||
, line(FFlag::LexerResumesFromPosition ? startPosition.line : 0)
|
, line(FFlag::LexerResumesFromPosition2 ? startPosition.line : 0)
|
||||||
, lineOffset(0)
|
, lineOffset(FFlag::LexerResumesFromPosition2 ? 0u - startPosition.column : 0)
|
||||||
, lexeme(
|
, lexeme(
|
||||||
(FFlag::LexerResumesFromPosition ? Location(Position(startPosition.line, startPosition.column), 0) : Location(Position(0, 0), 0)),
|
(FFlag::LexerResumesFromPosition2 ? Location(Position(startPosition.line, startPosition.column), 0) : Location(Position(0, 0), 0)),
|
||||||
Lexeme::Eof
|
Lexeme::Eof
|
||||||
)
|
)
|
||||||
, names(names)
|
, names(names)
|
||||||
, skipComments(false)
|
, skipComments(false)
|
||||||
, readNames(true)
|
, readNames(true)
|
||||||
, lexResumeOffset(FFlag::LexerResumesFromPosition ? startPosition.column : 0)
|
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -372,7 +405,6 @@ Lexeme Lexer::lookahead()
|
||||||
Location currentPrevLocation = prevLocation;
|
Location currentPrevLocation = prevLocation;
|
||||||
size_t currentBraceStackSize = braceStack.size();
|
size_t currentBraceStackSize = braceStack.size();
|
||||||
BraceType currentBraceType = braceStack.empty() ? BraceType::Normal : braceStack.back();
|
BraceType currentBraceType = braceStack.empty() ? BraceType::Normal : braceStack.back();
|
||||||
unsigned int currentLexResumeOffset = lexResumeOffset;
|
|
||||||
|
|
||||||
Lexeme result = next();
|
Lexeme result = next();
|
||||||
|
|
||||||
|
@ -381,7 +413,6 @@ Lexeme Lexer::lookahead()
|
||||||
lineOffset = currentLineOffset;
|
lineOffset = currentLineOffset;
|
||||||
lexeme = currentLexeme;
|
lexeme = currentLexeme;
|
||||||
prevLocation = currentPrevLocation;
|
prevLocation = currentPrevLocation;
|
||||||
lexResumeOffset = currentLexResumeOffset;
|
|
||||||
|
|
||||||
if (braceStack.size() < currentBraceStackSize)
|
if (braceStack.size() < currentBraceStackSize)
|
||||||
braceStack.push_back(currentBraceType);
|
braceStack.push_back(currentBraceType);
|
||||||
|
@ -412,9 +443,10 @@ char Lexer::peekch(unsigned int lookahead) const
|
||||||
return (offset + lookahead < bufferSize) ? buffer[offset + lookahead] : 0;
|
return (offset + lookahead < bufferSize) ? buffer[offset + lookahead] : 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LUAU_FORCEINLINE
|
||||||
Position Lexer::position() const
|
Position Lexer::position() const
|
||||||
{
|
{
|
||||||
return Position(line, offset - lineOffset + (FFlag::LexerResumesFromPosition ? lexResumeOffset : 0));
|
return Position(line, offset - lineOffset);
|
||||||
}
|
}
|
||||||
|
|
||||||
LUAU_FORCEINLINE
|
LUAU_FORCEINLINE
|
||||||
|
@ -433,9 +465,6 @@ void Lexer::consumeAny()
|
||||||
{
|
{
|
||||||
line++;
|
line++;
|
||||||
lineOffset = offset + 1;
|
lineOffset = offset + 1;
|
||||||
// every new line, we reset
|
|
||||||
if (FFlag::LexerResumesFromPosition)
|
|
||||||
lexResumeOffset = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
offset++;
|
offset++;
|
||||||
|
@ -764,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:
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue