mirror of
https://github.com/luau-lang/luau.git
synced 2025-08-26 11:27:08 +01:00
190 lines
5.4 KiB
C++
190 lines
5.4 KiB
C++
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
|
|
|
|
#include "Luau/Constraint.h"
|
|
#include "Luau/TypeFunction.h"
|
|
#include "Luau/VisitType.h"
|
|
|
|
LUAU_FASTFLAG(LuauEagerGeneralization4)
|
|
LUAU_FASTFLAG(LuauForceSimplifyConstraint2)
|
|
LUAU_FASTFLAGVARIABLE(LuauExplicitSkipBoundTypes)
|
|
|
|
namespace Luau
|
|
{
|
|
|
|
Constraint::Constraint(NotNull<Scope> scope, const Location& location, ConstraintV&& c)
|
|
: scope(scope)
|
|
, location(location)
|
|
, c(std::move(c))
|
|
{
|
|
}
|
|
|
|
struct ReferenceCountInitializer : TypeOnceVisitor
|
|
{
|
|
NotNull<TypeIds> result;
|
|
bool traverseIntoTypeFunctions = true;
|
|
|
|
explicit ReferenceCountInitializer(NotNull<TypeIds> result)
|
|
: TypeOnceVisitor("ReferenceCountInitializer", FFlag::LuauExplicitSkipBoundTypes)
|
|
, result(result)
|
|
{
|
|
}
|
|
|
|
bool visit(TypeId ty, const FreeType&) override
|
|
{
|
|
result->insert(ty);
|
|
return false;
|
|
}
|
|
|
|
bool visit(TypeId ty, const BlockedType&) override
|
|
{
|
|
result->insert(ty);
|
|
return false;
|
|
}
|
|
|
|
bool visit(TypeId ty, const PendingExpansionType&) override
|
|
{
|
|
result->insert(ty);
|
|
return false;
|
|
}
|
|
|
|
bool visit(TypeId ty, const TableType& tt) override
|
|
{
|
|
if (FFlag::LuauEagerGeneralization4)
|
|
{
|
|
if (tt.state == TableState::Unsealed || tt.state == TableState::Free)
|
|
result->insert(ty);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool visit(TypeId ty, const ExternType&) override
|
|
{
|
|
// ExternTypes never contain free types.
|
|
return false;
|
|
}
|
|
|
|
bool visit(TypeId, const TypeFunctionInstanceType& tfit) override
|
|
{
|
|
if (FFlag::LuauForceSimplifyConstraint2)
|
|
return tfit.function->canReduceGenerics;
|
|
else
|
|
return FFlag::LuauEagerGeneralization4 && traverseIntoTypeFunctions;
|
|
}
|
|
};
|
|
|
|
bool isReferenceCountedType(const TypeId typ)
|
|
{
|
|
if (FFlag::LuauEagerGeneralization4)
|
|
{
|
|
if (auto tt = get<TableType>(typ))
|
|
return tt->state == TableState::Free || tt->state == TableState::Unsealed;
|
|
}
|
|
|
|
// n.b. this should match whatever `ReferenceCountInitializer` includes.
|
|
return get<FreeType>(typ) || get<BlockedType>(typ) || get<PendingExpansionType>(typ);
|
|
}
|
|
|
|
TypeIds Constraint::getMaybeMutatedFreeTypes() const
|
|
{
|
|
// For the purpose of this function and reference counting in general, we are only considering
|
|
// mutations that affect the _bounds_ of the free type, and not something that may bind the free
|
|
// type itself to a new type. As such, `ReduceConstraint` and `GeneralizationConstraint` have no
|
|
// contribution to the output set here.
|
|
|
|
TypeIds types;
|
|
ReferenceCountInitializer rci{NotNull{&types}};
|
|
|
|
if (auto ec = get<EqualityConstraint>(*this))
|
|
{
|
|
rci.traverse(ec->resultType);
|
|
// `EqualityConstraints` should not mutate `assignmentType`.
|
|
}
|
|
else if (auto sc = get<SubtypeConstraint>(*this))
|
|
{
|
|
rci.traverse(sc->subType);
|
|
rci.traverse(sc->superType);
|
|
}
|
|
else if (auto psc = get<PackSubtypeConstraint>(*this))
|
|
{
|
|
rci.traverse(psc->subPack);
|
|
rci.traverse(psc->superPack);
|
|
}
|
|
else if (auto itc = get<IterableConstraint>(*this))
|
|
{
|
|
for (TypeId ty : itc->variables)
|
|
rci.traverse(ty);
|
|
// `IterableConstraints` should not mutate `iterator`.
|
|
}
|
|
else if (auto nc = get<NameConstraint>(*this))
|
|
{
|
|
rci.traverse(nc->namedType);
|
|
}
|
|
else if (auto taec = get<TypeAliasExpansionConstraint>(*this))
|
|
{
|
|
rci.traverse(taec->target);
|
|
}
|
|
else if (auto fchc = get<FunctionCheckConstraint>(*this))
|
|
{
|
|
rci.traverse(fchc->argsPack);
|
|
}
|
|
else if (auto fcc = get<FunctionCallConstraint>(*this); fcc && FFlag::LuauEagerGeneralization4)
|
|
{
|
|
rci.traverseIntoTypeFunctions = false;
|
|
rci.traverse(fcc->fn);
|
|
rci.traverse(fcc->argsPack);
|
|
rci.traverseIntoTypeFunctions = true;
|
|
}
|
|
else if (auto ptc = get<PrimitiveTypeConstraint>(*this))
|
|
{
|
|
rci.traverse(ptc->freeType);
|
|
}
|
|
else if (auto hpc = get<HasPropConstraint>(*this))
|
|
{
|
|
rci.traverse(hpc->resultType);
|
|
if (FFlag::LuauEagerGeneralization4)
|
|
rci.traverse(hpc->subjectType);
|
|
}
|
|
else if (auto hic = get<HasIndexerConstraint>(*this))
|
|
{
|
|
if (FFlag::LuauEagerGeneralization4)
|
|
rci.traverse(hic->subjectType);
|
|
rci.traverse(hic->resultType);
|
|
// `HasIndexerConstraint` should not mutate `indexType`.
|
|
}
|
|
else if (auto apc = get<AssignPropConstraint>(*this))
|
|
{
|
|
rci.traverse(apc->lhsType);
|
|
rci.traverse(apc->rhsType);
|
|
}
|
|
else if (auto aic = get<AssignIndexConstraint>(*this))
|
|
{
|
|
rci.traverse(aic->lhsType);
|
|
rci.traverse(aic->indexType);
|
|
rci.traverse(aic->rhsType);
|
|
}
|
|
else if (auto uc = get<UnpackConstraint>(*this))
|
|
{
|
|
for (TypeId ty : uc->resultPack)
|
|
rci.traverse(ty);
|
|
// `UnpackConstraint` should not mutate `sourcePack`.
|
|
}
|
|
else if (auto rpc = get<ReducePackConstraint>(*this))
|
|
{
|
|
rci.traverse(rpc->tp);
|
|
}
|
|
else if (auto tcc = get<TableCheckConstraint>(*this))
|
|
{
|
|
rci.traverse(tcc->exprType);
|
|
}
|
|
|
|
// NOTE: this should probably be in an if-else chain with the above.
|
|
if (auto pftc = get<PushFunctionTypeConstraint>(*this))
|
|
{
|
|
rci.traverse(pftc->functionType);
|
|
}
|
|
|
|
return types;
|
|
}
|
|
|
|
} // namespace Luau
|