luau/Analysis/src/Quantify.cpp

283 lines
7.6 KiB
C++
Raw Normal View History

// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#include "Luau/Quantify.h"
2022-06-24 02:44:07 +01:00
#include "Luau/Scope.h"
2022-06-17 01:54:42 +01:00
#include "Luau/Substitution.h"
2022-06-24 02:44:07 +01:00
#include "Luau/TxnLog.h"
2023-01-03 17:33:19 +00:00
#include "Luau/Type.h"
#include "Luau/VisitType.h"
2022-07-01 00:29:02 +01:00
LUAU_FASTFLAG(DebugLuauSharedSelf)
2022-04-14 22:57:15 +01:00
namespace Luau
{
2023-01-03 17:33:19 +00:00
struct Quantifier final : TypeOnceVisitor
{
TypeLevel level;
std::vector<TypeId> generics;
std::vector<TypePackId> genericPacks;
2022-07-29 04:41:13 +01:00
Scope* scope = nullptr;
2022-04-14 22:57:15 +01:00
bool seenGenericType = false;
bool seenMutableType = false;
2022-05-06 00:52:48 +01:00
explicit Quantifier(TypeLevel level)
2022-02-18 00:41:20 +00:00
: level(level)
{
2022-06-03 21:32:20 +01:00
}
/// @return true if outer encloses inner
2022-07-29 04:41:13 +01:00
bool subsumes(Scope* outer, Scope* inner)
2022-06-03 21:32:20 +01:00
{
while (inner)
{
if (inner == outer)
return true;
2022-07-29 04:41:13 +01:00
inner = inner->parent.get();
2022-06-03 21:32:20 +01:00
}
return false;
}
2023-01-03 17:33:19 +00:00
bool visit(TypeId ty, const FreeType& ftv) override
2022-05-06 00:52:48 +01:00
{
2022-05-20 00:46:52 +01:00
seenMutableType = true;
2022-05-06 00:52:48 +01:00
2022-09-29 23:11:54 +01:00
if (!level.subsumes(ftv.level))
2022-05-06 00:52:48 +01:00
return false;
2023-01-03 17:33:19 +00:00
*asMutable(ty) = GenericType{level};
2022-06-03 21:32:20 +01:00
2022-05-06 00:52:48 +01:00
generics.push_back(ty);
return false;
}
2023-01-03 17:33:19 +00:00
bool visit(TypeId ty, const TableType&) override
2022-05-06 00:52:48 +01:00
{
2023-01-03 17:33:19 +00:00
LUAU_ASSERT(getMutable<TableType>(ty));
TableType& ttv = *getMutable<TableType>(ty);
2022-05-20 00:46:52 +01:00
if (ttv.state == TableState::Generic)
seenGenericType = true;
2022-04-14 22:57:15 +01:00
2022-05-20 00:46:52 +01:00
if (ttv.state == TableState::Free)
seenMutableType = true;
2022-04-14 22:57:15 +01:00
2022-09-29 23:11:54 +01:00
if (!level.subsumes(ttv.level))
2022-04-14 22:57:15 +01:00
{
2022-05-20 00:46:52 +01:00
if (ttv.state == TableState::Unsealed)
2022-04-14 22:57:15 +01:00
seenMutableType = true;
return false;
2022-04-14 22:57:15 +01:00
}
if (ttv.state == TableState::Free)
2022-04-14 22:57:15 +01:00
{
ttv.state = TableState::Generic;
2022-05-20 00:46:52 +01:00
seenGenericType = true;
2022-04-14 22:57:15 +01:00
}
else if (ttv.state == TableState::Unsealed)
ttv.state = TableState::Sealed;
ttv.level = level;
return true;
}
2022-05-06 00:52:48 +01:00
bool visit(TypePackId tp, const FreeTypePack& ftp) override
{
2022-05-20 00:46:52 +01:00
seenMutableType = true;
2022-04-14 22:57:15 +01:00
2022-09-29 23:11:54 +01:00
if (!level.subsumes(ftp.level))
return false;
*asMutable(tp) = GenericTypePack{level};
genericPacks.push_back(tp);
return true;
}
};
2022-02-18 00:41:20 +00:00
void quantify(TypeId ty, TypeLevel level)
{
2022-07-01 00:29:02 +01:00
if (FFlag::DebugLuauSharedSelf)
2022-05-20 00:46:52 +01:00
{
2022-07-01 00:29:02 +01:00
ty = follow(ty);
if (auto ttv = getTableType(ty); ttv && ttv->selfTy)
{
Quantifier selfQ{level};
selfQ.traverse(*ttv->selfTy);
Quantifier q{level};
q.traverse(ty);
for (const auto& [_, prop] : ttv->props)
{
2023-04-28 12:55:55 +01:00
auto ftv = getMutable<FunctionType>(follow(prop.type()));
2022-07-01 00:29:02 +01:00
if (!ftv || !ftv->hasSelf)
continue;
if (Luau::first(ftv->argTypes) == ttv->selfTy)
{
ftv->generics.insert(ftv->generics.end(), selfQ.generics.begin(), selfQ.generics.end());
ftv->genericPacks.insert(ftv->genericPacks.end(), selfQ.genericPacks.begin(), selfQ.genericPacks.end());
}
}
}
2023-01-03 17:33:19 +00:00
else if (auto ftv = getMutable<FunctionType>(ty))
2022-07-01 00:29:02 +01:00
{
Quantifier q{level};
q.traverse(ty);
ftv->generics.insert(ftv->generics.end(), q.generics.begin(), q.generics.end());
ftv->genericPacks.insert(ftv->genericPacks.end(), q.genericPacks.begin(), q.genericPacks.end());
if (ftv->generics.empty() && ftv->genericPacks.empty() && !q.seenMutableType && !q.seenGenericType)
2023-05-19 19:59:59 +01:00
ftv->hasNoFreeOrGenericTypes = true;
2022-07-01 00:29:02 +01:00
}
2022-05-20 00:46:52 +01:00
}
else
{
2022-07-01 00:29:02 +01:00
Quantifier q{level};
q.traverse(ty);
2022-04-14 22:57:15 +01:00
2023-01-03 17:33:19 +00:00
FunctionType* ftv = getMutable<FunctionType>(ty);
2022-07-01 00:29:02 +01:00
LUAU_ASSERT(ftv);
2022-08-04 22:27:28 +01:00
ftv->generics.insert(ftv->generics.end(), q.generics.begin(), q.generics.end());
ftv->genericPacks.insert(ftv->genericPacks.end(), q.genericPacks.begin(), q.genericPacks.end());
2022-07-01 00:29:02 +01:00
}
2022-06-03 21:32:20 +01:00
}
2022-06-17 01:54:42 +01:00
struct PureQuantifier : Substitution
{
2022-07-29 04:41:13 +01:00
Scope* scope;
2023-05-12 13:15:01 +01:00
OrderedMap<TypeId, TypeId> insertedGenerics;
OrderedMap<TypePackId, TypePackId> insertedGenericPacks;
2023-01-20 12:02:39 +00:00
bool seenMutableType = false;
bool seenGenericType = false;
2022-06-17 01:54:42 +01:00
2022-07-29 04:41:13 +01:00
PureQuantifier(TypeArena* arena, Scope* scope)
2022-07-01 00:29:02 +01:00
: Substitution(TxnLog::empty(), arena)
2022-06-17 01:54:42 +01:00
, scope(scope)
{
}
bool isDirty(TypeId ty) override
{
LUAU_ASSERT(ty == follow(ty));
2023-01-03 17:33:19 +00:00
if (auto ftv = get<FreeType>(ty))
2022-06-17 01:54:42 +01:00
{
2023-01-20 12:02:39 +00:00
bool result = subsumes(scope, ftv->scope);
seenMutableType |= result;
return result;
2022-06-17 01:54:42 +01:00
}
2023-01-03 17:33:19 +00:00
else if (auto ttv = get<TableType>(ty))
2022-06-17 01:54:42 +01:00
{
2023-01-20 12:02:39 +00:00
if (ttv->state == TableState::Free)
seenMutableType = true;
else if (ttv->state == TableState::Generic)
seenGenericType = true;
2023-02-24 18:24:22 +00:00
return (ttv->state == TableState::Unsealed || ttv->state == TableState::Free) && subsumes(scope, ttv->scope);
2022-06-17 01:54:42 +01:00
}
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
{
2023-01-03 17:33:19 +00:00
if (auto ftv = get<FreeType>(ty))
2022-06-17 01:54:42 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId result = arena->addType(GenericType{scope});
2023-05-12 13:15:01 +01:00
insertedGenerics.push(ty, result);
2022-06-17 01:54:42 +01:00
return result;
}
2023-01-03 17:33:19 +00:00
else if (auto ttv = get<TableType>(ty))
2022-06-17 01:54:42 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId result = arena->addType(TableType{});
TableType* resultTable = getMutable<TableType>(result);
2022-06-17 01:54:42 +01:00
LUAU_ASSERT(resultTable);
*resultTable = *ttv;
2022-09-29 23:11:54 +01:00
resultTable->level = TypeLevel{};
resultTable->scope = scope;
2023-01-20 12:02:39 +00:00
if (ttv->state == TableState::Free)
2023-05-12 13:15:01 +01:00
{
2023-01-20 12:02:39 +00:00
resultTable->state = TableState::Generic;
2023-05-12 13:15:01 +01:00
insertedGenerics.push(ty, result);
}
2023-01-20 12:02:39 +00:00
else if (ttv->state == TableState::Unsealed)
resultTable->state = TableState::Sealed;
2022-06-17 01:54:42 +01:00
return result;
}
return ty;
}
TypePackId clean(TypePackId tp) override
{
if (auto ftp = get<FreeTypePack>(tp))
{
2023-05-12 13:15:01 +01:00
TypePackId result = arena->addTypePack(TypePackVar{GenericTypePack{scope}});
insertedGenericPacks.push(tp, result);
2022-06-17 01:54:42 +01:00
return result;
}
return tp;
}
bool ignoreChildren(TypeId ty) override
{
2023-06-24 06:33:44 +01:00
if (get<ClassType>(ty))
2022-08-11 21:42:54 +01:00
return true;
2022-06-17 01:54:42 +01:00
return ty->persistent;
}
bool ignoreChildren(TypePackId ty) override
{
return ty->persistent;
}
};
2023-05-12 13:15:01 +01:00
std::optional<QuantifierResult> quantify(TypeArena* arena, TypeId ty, Scope* scope)
2022-06-17 01:54:42 +01:00
{
2022-07-01 00:29:02 +01:00
PureQuantifier quantifier{arena, scope};
2022-06-17 01:54:42 +01:00
std::optional<TypeId> result = quantifier.substitute(ty);
2023-03-10 19:20:04 +00:00
if (!result)
return std::nullopt;
2022-06-17 01:54:42 +01:00
2023-01-03 17:33:19 +00:00
FunctionType* ftv = getMutable<FunctionType>(*result);
2022-06-17 01:54:42 +01:00
LUAU_ASSERT(ftv);
2022-09-29 23:11:54 +01:00
ftv->scope = scope;
2023-05-12 13:15:01 +01:00
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]);
2023-05-19 19:59:59 +01:00
ftv->hasNoFreeOrGenericTypes = ftv->generics.empty() && ftv->genericPacks.empty() && !quantifier.seenGenericType && !quantifier.seenMutableType;
2022-06-17 01:54:42 +01:00
2023-05-12 13:15:01 +01:00
return std::optional<QuantifierResult>({*result, std::move(quantifier.insertedGenerics), std::move(quantifier.insertedGenericPacks)});
2022-06-17 01:54:42 +01:00
}
} // namespace Luau