mirror of
https://github.com/luau-lang/luau.git
synced 2025-01-19 17:28:06 +00:00
ce8495a69e
# What's Changed? - Code refactoring with a new clang-format - More bug fixes / test case fixes in the new solver ## New Solver - More precise telemetry collection of `any` types - Simplification of two completely disjoint tables combines them into a single table that inherits all properties / indexers - Refining a `never & <anything>` does not produce type family types nor constraints - Silence "inference failed to complete" error when it is the only error reported --- ### Internal Contributors Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Dibri Nsofor <dnsofor@roblox.com> Co-authored-by: Jeremy Yoo <jyoo@roblox.com> Co-authored-by: Vighnesh Vijay <vvijay@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com> --------- Co-authored-by: Aaron Weiss <aaronweiss@roblox.com> Co-authored-by: Alexander McCord <amccord@roblox.com> Co-authored-by: Andy Friesen <afriesen@roblox.com> Co-authored-by: Vighnesh <vvijay@roblox.com> Co-authored-by: Aviral Goel <agoel@roblox.com> Co-authored-by: David Cope <dcope@roblox.com> Co-authored-by: Lily Brown <lbrown@roblox.com> Co-authored-by: Vyacheslav Egorov <vegorov@roblox.com>
240 lines
6.2 KiB
C++
240 lines
6.2 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/Quantify.h"
|
|
|
|
#include "Luau/Scope.h"
|
|
#include "Luau/Substitution.h"
|
|
#include "Luau/TxnLog.h"
|
|
#include "Luau/Type.h"
|
|
#include "Luau/VisitType.h"
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
struct Quantifier final : TypeOnceVisitor
|
|
{
|
|
TypeLevel level;
|
|
std::vector<TypeId> generics;
|
|
std::vector<TypePackId> genericPacks;
|
|
Scope* scope = nullptr;
|
|
bool seenGenericType = false;
|
|
bool seenMutableType = false;
|
|
|
|
explicit Quantifier(TypeLevel level)
|
|
: level(level)
|
|
{
|
|
}
|
|
|
|
/// @return true if outer encloses inner
|
|
bool subsumes(Scope* outer, Scope* inner)
|
|
{
|
|
while (inner)
|
|
{
|
|
if (inner == outer)
|
|
return true;
|
|
inner = inner->parent.get();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool visit(TypeId ty, const FreeType& ftv) override
|
|
{
|
|
seenMutableType = true;
|
|
|
|
if (!level.subsumes(ftv.level))
|
|
return false;
|
|
|
|
*asMutable(ty) = GenericType{level};
|
|
|
|
generics.push_back(ty);
|
|
|
|
return false;
|
|
}
|
|
|
|
bool visit(TypeId ty, const TableType&) override
|
|
{
|
|
LUAU_ASSERT(getMutable<TableType>(ty));
|
|
TableType& ttv = *getMutable<TableType>(ty);
|
|
|
|
if (ttv.state == TableState::Generic)
|
|
seenGenericType = true;
|
|
|
|
if (ttv.state == TableState::Free)
|
|
seenMutableType = true;
|
|
|
|
if (!level.subsumes(ttv.level))
|
|
{
|
|
if (ttv.state == TableState::Unsealed)
|
|
seenMutableType = true;
|
|
return false;
|
|
}
|
|
|
|
if (ttv.state == TableState::Free)
|
|
{
|
|
ttv.state = TableState::Generic;
|
|
seenGenericType = true;
|
|
}
|
|
else if (ttv.state == TableState::Unsealed)
|
|
ttv.state = TableState::Sealed;
|
|
|
|
ttv.level = level;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool visit(TypePackId tp, const FreeTypePack& ftp) override
|
|
{
|
|
seenMutableType = true;
|
|
|
|
if (!level.subsumes(ftp.level))
|
|
return false;
|
|
|
|
*asMutable(tp) = GenericTypePack{level};
|
|
genericPacks.push_back(tp);
|
|
return true;
|
|
}
|
|
};
|
|
|
|
void quantify(TypeId ty, TypeLevel level)
|
|
{
|
|
Quantifier q{level};
|
|
q.traverse(ty);
|
|
|
|
FunctionType* ftv = getMutable<FunctionType>(ty);
|
|
LUAU_ASSERT(ftv);
|
|
ftv->generics.insert(ftv->generics.end(), q.generics.begin(), q.generics.end());
|
|
ftv->genericPacks.insert(ftv->genericPacks.end(), q.genericPacks.begin(), q.genericPacks.end());
|
|
}
|
|
|
|
struct PureQuantifier : Substitution
|
|
{
|
|
Scope* scope;
|
|
OrderedMap<TypeId, TypeId> insertedGenerics;
|
|
OrderedMap<TypePackId, TypePackId> insertedGenericPacks;
|
|
bool seenMutableType = false;
|
|
bool seenGenericType = false;
|
|
|
|
PureQuantifier(TypeArena* arena, Scope* scope)
|
|
: Substitution(TxnLog::empty(), arena)
|
|
, scope(scope)
|
|
{
|
|
}
|
|
|
|
bool isDirty(TypeId ty) override
|
|
{
|
|
LUAU_ASSERT(ty == follow(ty));
|
|
|
|
if (auto ftv = get<FreeType>(ty))
|
|
{
|
|
bool result = subsumes(scope, ftv->scope);
|
|
seenMutableType |= result;
|
|
return result;
|
|
}
|
|
else if (auto ttv = get<TableType>(ty))
|
|
{
|
|
if (ttv->state == TableState::Free)
|
|
seenMutableType = true;
|
|
else if (ttv->state == TableState::Generic)
|
|
seenGenericType = true;
|
|
|
|
return (ttv->state == TableState::Unsealed || ttv->state == TableState::Free) && subsumes(scope, ttv->scope);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool isDirty(TypePackId tp) override
|
|
{
|
|
if (auto ftp = get<FreeTypePack>(tp))
|
|
{
|
|
return subsumes(scope, ftp->scope);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
TypeId clean(TypeId ty) override
|
|
{
|
|
if (auto ftv = get<FreeType>(ty))
|
|
{
|
|
TypeId result = arena->addType(GenericType{scope});
|
|
insertedGenerics.push(ty, result);
|
|
return result;
|
|
}
|
|
else if (auto ttv = get<TableType>(ty))
|
|
{
|
|
TypeId result = arena->addType(TableType{});
|
|
TableType* resultTable = getMutable<TableType>(result);
|
|
LUAU_ASSERT(resultTable);
|
|
|
|
*resultTable = *ttv;
|
|
resultTable->level = TypeLevel{};
|
|
resultTable->scope = scope;
|
|
|
|
if (ttv->state == TableState::Free)
|
|
{
|
|
resultTable->state = TableState::Generic;
|
|
insertedGenerics.push(ty, result);
|
|
}
|
|
else if (ttv->state == TableState::Unsealed)
|
|
resultTable->state = TableState::Sealed;
|
|
|
|
return result;
|
|
}
|
|
|
|
return ty;
|
|
}
|
|
|
|
TypePackId clean(TypePackId tp) override
|
|
{
|
|
if (auto ftp = get<FreeTypePack>(tp))
|
|
{
|
|
TypePackId result = arena->addTypePack(TypePackVar{GenericTypePack{scope}});
|
|
insertedGenericPacks.push(tp, result);
|
|
return result;
|
|
}
|
|
|
|
return tp;
|
|
}
|
|
|
|
bool ignoreChildren(TypeId ty) override
|
|
{
|
|
if (get<ClassType>(ty))
|
|
return true;
|
|
|
|
return ty->persistent;
|
|
}
|
|
bool ignoreChildren(TypePackId ty) override
|
|
{
|
|
return ty->persistent;
|
|
}
|
|
};
|
|
|
|
std::optional<QuantifierResult> quantify(TypeArena* arena, TypeId ty, Scope* scope)
|
|
{
|
|
PureQuantifier quantifier{arena, scope};
|
|
std::optional<TypeId> result = quantifier.substitute(ty);
|
|
if (!result)
|
|
return std::nullopt;
|
|
|
|
FunctionType* ftv = getMutable<FunctionType>(*result);
|
|
LUAU_ASSERT(ftv);
|
|
ftv->scope = scope;
|
|
|
|
for (auto k : quantifier.insertedGenerics.keys)
|
|
{
|
|
TypeId g = quantifier.insertedGenerics.pairings[k];
|
|
if (get<GenericType>(g))
|
|
ftv->generics.push_back(g);
|
|
}
|
|
|
|
for (auto k : quantifier.insertedGenericPacks.keys)
|
|
ftv->genericPacks.push_back(quantifier.insertedGenericPacks.pairings[k]);
|
|
|
|
ftv->hasNoFreeOrGenericTypes = ftv->generics.empty() && ftv->genericPacks.empty() && !quantifier.seenGenericType && !quantifier.seenMutableType;
|
|
|
|
return std::optional<QuantifierResult>({*result, std::move(quantifier.insertedGenerics), std::move(quantifier.insertedGenericPacks)});
|
|
}
|
|
|
|
} // namespace Luau
|