luau/Analysis/src/Clone.cpp

542 lines
16 KiB
C++
Raw Normal View History

2022-04-07 21:53:47 +01:00
// This file is part of the Luau programming language and is licensed under MIT License; see LICENSE.txt for details
#include "Luau/Clone.h"
2022-10-21 18:33:43 +01:00
2022-04-07 21:53:47 +01:00
#include "Luau/RecursionCounter.h"
2022-06-03 21:32:20 +01:00
#include "Luau/TxnLog.h"
2022-04-07 21:53:47 +01:00
#include "Luau/TypePack.h"
#include "Luau/Unifiable.h"
2022-04-14 22:57:15 +01:00
LUAU_FASTFLAG(DebugLuauCopyBeforeNormalizing)
2023-04-28 12:55:55 +01:00
LUAU_FASTFLAG(DebugLuauReadWriteProperties)
2022-04-14 22:57:15 +01:00
LUAU_FASTFLAG(DebugLuauDeferredConstraintResolution)
2022-04-07 21:53:47 +01:00
LUAU_FASTINTVARIABLE(LuauTypeCloneRecursionLimit, 300)
2023-05-19 19:59:59 +01:00
LUAU_FASTFLAGVARIABLE(LuauCloneCyclicUnions, false)
2022-04-07 21:53:47 +01:00
namespace Luau
{
namespace
{
2023-04-28 12:55:55 +01:00
Property clone(const Property& prop, TypeArena& dest, CloneState& cloneState)
{
if (FFlag::DebugLuauReadWriteProperties)
{
std::optional<TypeId> cloneReadTy;
if (auto ty = prop.readType())
cloneReadTy = clone(*ty, dest, cloneState);
std::optional<TypeId> cloneWriteTy;
if (auto ty = prop.writeType())
cloneWriteTy = clone(*ty, dest, cloneState);
std::optional<Property> cloned = Property::create(cloneReadTy, cloneWriteTy);
LUAU_ASSERT(cloned);
cloned->deprecated = prop.deprecated;
cloned->deprecatedSuggestion = prop.deprecatedSuggestion;
cloned->location = prop.location;
cloned->tags = prop.tags;
cloned->documentationSymbol = prop.documentationSymbol;
return *cloned;
}
else
{
return Property{
clone(prop.type(), dest, cloneState),
prop.deprecated,
prop.deprecatedSuggestion,
prop.location,
prop.tags,
prop.documentationSymbol,
};
}
}
2023-05-12 13:15:01 +01:00
static TableIndexer clone(const TableIndexer& indexer, TypeArena& dest, CloneState& cloneState)
{
return TableIndexer{clone(indexer.indexType, dest, cloneState), clone(indexer.indexResultType, dest, cloneState)};
}
2022-04-07 21:53:47 +01:00
struct TypePackCloner;
/*
* Both TypeCloner and TypePackCloner work by depositing the requested type variable into the appropriate 'seen' set.
* They do not return anything because their sole consumer (the deepClone function) already has a pointer into this storage.
*/
struct TypeCloner
{
2022-04-14 22:57:15 +01:00
TypeCloner(TypeArena& dest, TypeId typeId, CloneState& cloneState)
2022-04-07 21:53:47 +01:00
: dest(dest)
, typeId(typeId)
2022-04-14 22:57:15 +01:00
, seenTypes(cloneState.seenTypes)
, seenTypePacks(cloneState.seenTypePacks)
2022-04-07 21:53:47 +01:00
, cloneState(cloneState)
{
}
TypeArena& dest;
TypeId typeId;
SeenTypes& seenTypes;
SeenTypePacks& seenTypePacks;
CloneState& cloneState;
template<typename T>
void defaultClone(const T& t);
2023-04-07 20:56:27 +01:00
void operator()(const FreeType& t);
void operator()(const GenericType& t);
void operator()(const BoundType& t);
void operator()(const ErrorType& t);
2023-01-03 17:33:19 +00:00
void operator()(const BlockedType& t);
void operator()(const PendingExpansionType& t);
void operator()(const PrimitiveType& t);
void operator()(const SingletonType& t);
void operator()(const FunctionType& t);
void operator()(const TableType& t);
void operator()(const MetatableType& t);
void operator()(const ClassType& t);
void operator()(const AnyType& t);
void operator()(const UnionType& t);
void operator()(const IntersectionType& t);
void operator()(const LazyType& t);
void operator()(const UnknownType& t);
void operator()(const NeverType& t);
void operator()(const NegationType& t);
2023-05-12 13:15:01 +01:00
void operator()(const TypeFamilyInstanceType& t);
2022-04-07 21:53:47 +01:00
};
struct TypePackCloner
{
TypeArena& dest;
TypePackId typePackId;
SeenTypes& seenTypes;
SeenTypePacks& seenTypePacks;
CloneState& cloneState;
2022-04-14 22:57:15 +01:00
TypePackCloner(TypeArena& dest, TypePackId typePackId, CloneState& cloneState)
2022-04-07 21:53:47 +01:00
: dest(dest)
, typePackId(typePackId)
2022-04-14 22:57:15 +01:00
, seenTypes(cloneState.seenTypes)
, seenTypePacks(cloneState.seenTypePacks)
2022-04-07 21:53:47 +01:00
, cloneState(cloneState)
{
}
template<typename T>
void defaultClone(const T& t)
{
TypePackId cloned = dest.addTypePack(TypePackVar{t});
seenTypePacks[typePackId] = cloned;
}
2023-04-07 20:56:27 +01:00
void operator()(const FreeTypePack& t)
2022-04-07 21:53:47 +01:00
{
2022-05-20 00:46:52 +01:00
defaultClone(t);
2022-04-07 21:53:47 +01:00
}
2023-04-07 20:56:27 +01:00
void operator()(const GenericTypePack& t)
2022-04-07 21:53:47 +01:00
{
defaultClone(t);
}
2023-04-07 20:56:27 +01:00
void operator()(const ErrorTypePack& t)
2022-04-07 21:53:47 +01:00
{
defaultClone(t);
}
2022-09-02 00:00:14 +01:00
void operator()(const BlockedTypePack& t)
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
// While we are a-cloning, we can flatten out bound Types and make things a bit tighter.
2022-04-07 21:53:47 +01:00
// We just need to be sure that we rewrite pointers both to the binder and the bindee to the same pointer.
void operator()(const Unifiable::Bound<TypePackId>& t)
{
2022-04-14 22:57:15 +01:00
TypePackId cloned = clone(t.boundTo, dest, cloneState);
if (FFlag::DebugLuauCopyBeforeNormalizing)
cloned = dest.addTypePack(TypePackVar{BoundTypePack{cloned}});
2022-04-07 21:53:47 +01:00
seenTypePacks[typePackId] = cloned;
}
void operator()(const VariadicTypePack& t)
{
2022-04-14 22:57:15 +01:00
TypePackId cloned = dest.addTypePack(TypePackVar{VariadicTypePack{clone(t.ty, dest, cloneState), /*hidden*/ t.hidden}});
2022-04-07 21:53:47 +01:00
seenTypePacks[typePackId] = cloned;
}
void operator()(const TypePack& t)
{
TypePackId cloned = dest.addTypePack(TypePack{});
TypePack* destTp = getMutable<TypePack>(cloned);
LUAU_ASSERT(destTp != nullptr);
seenTypePacks[typePackId] = cloned;
for (TypeId ty : t.head)
2022-04-14 22:57:15 +01:00
destTp->head.push_back(clone(ty, dest, cloneState));
2022-04-07 21:53:47 +01:00
if (t.tail)
2022-04-14 22:57:15 +01:00
destTp->tail = clone(*t.tail, dest, cloneState);
2022-04-07 21:53:47 +01:00
}
2023-05-12 13:15:01 +01:00
void operator()(const TypeFamilyInstanceTypePack& t)
{
TypePackId cloned = dest.addTypePack(TypeFamilyInstanceTypePack{t.family, {}, {}});
TypeFamilyInstanceTypePack* destTp = getMutable<TypeFamilyInstanceTypePack>(cloned);
LUAU_ASSERT(destTp);
seenTypePacks[typePackId] = cloned;
destTp->typeArguments.reserve(t.typeArguments.size());
for (TypeId ty : t.typeArguments)
destTp->typeArguments.push_back(clone(ty, dest, cloneState));
destTp->packArguments.reserve(t.packArguments.size());
for (TypePackId tp : t.packArguments)
destTp->packArguments.push_back(clone(tp, dest, cloneState));
}
2022-04-07 21:53:47 +01:00
};
template<typename T>
void TypeCloner::defaultClone(const T& t)
{
TypeId cloned = dest.addType(t);
seenTypes[typeId] = cloned;
}
2023-04-07 20:56:27 +01:00
void TypeCloner::operator()(const FreeType& t)
2022-04-07 21:53:47 +01:00
{
if (FFlag::DebugLuauDeferredConstraintResolution)
{
FreeType ft{t.scope, clone(t.lowerBound, dest, cloneState), clone(t.upperBound, dest, cloneState)};
TypeId res = dest.addType(ft);
seenTypes[typeId] = res;
}
else
defaultClone(t);
2022-04-07 21:53:47 +01:00
}
2023-04-07 20:56:27 +01:00
void TypeCloner::operator()(const GenericType& t)
2022-04-07 21:53:47 +01:00
{
defaultClone(t);
}
void TypeCloner::operator()(const Unifiable::Bound<TypeId>& t)
{
2022-04-14 22:57:15 +01:00
TypeId boundTo = clone(t.boundTo, dest, cloneState);
if (FFlag::DebugLuauCopyBeforeNormalizing)
2023-01-03 17:33:19 +00:00
boundTo = dest.addType(BoundType{boundTo});
2022-04-07 21:53:47 +01:00
seenTypes[typeId] = boundTo;
}
void TypeCloner::operator()(const Unifiable::Error& t)
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const BlockedType& t)
2022-06-17 01:54:42 +01:00
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const PendingExpansionType& t)
2022-08-04 22:27:28 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId res = dest.addType(PendingExpansionType{t.prefix, t.name, t.typeArguments, t.packArguments});
PendingExpansionType* petv = getMutable<PendingExpansionType>(res);
2022-08-04 22:27:28 +01:00
LUAU_ASSERT(petv);
seenTypes[typeId] = res;
std::vector<TypeId> typeArguments;
for (TypeId arg : t.typeArguments)
typeArguments.push_back(clone(arg, dest, cloneState));
std::vector<TypePackId> packArguments;
for (TypePackId arg : t.packArguments)
packArguments.push_back(clone(arg, dest, cloneState));
petv->typeArguments = std::move(typeArguments);
petv->packArguments = std::move(packArguments);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const PrimitiveType& t)
2022-04-07 21:53:47 +01:00
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const SingletonType& t)
2022-04-07 21:53:47 +01:00
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const FunctionType& t)
2022-04-07 21:53:47 +01:00
{
2022-09-29 23:11:54 +01:00
// FISHY: We always erase the scope when we clone things. clone() was
// originally written so that we could copy a module's type surface into an
// export arena. This probably dates to that.
2023-01-03 17:33:19 +00:00
TypeId result = dest.addType(FunctionType{TypeLevel{0, 0}, {}, {}, nullptr, nullptr, t.definition, t.hasSelf});
FunctionType* ftv = getMutable<FunctionType>(result);
2022-04-07 21:53:47 +01:00
LUAU_ASSERT(ftv != nullptr);
seenTypes[typeId] = result;
for (TypeId generic : t.generics)
2022-04-14 22:57:15 +01:00
ftv->generics.push_back(clone(generic, dest, cloneState));
2022-04-07 21:53:47 +01:00
for (TypePackId genericPack : t.genericPacks)
2022-04-14 22:57:15 +01:00
ftv->genericPacks.push_back(clone(genericPack, dest, cloneState));
2022-04-07 21:53:47 +01:00
ftv->tags = t.tags;
2022-04-14 22:57:15 +01:00
ftv->argTypes = clone(t.argTypes, dest, cloneState);
2022-04-07 21:53:47 +01:00
ftv->argNames = t.argNames;
2022-06-17 01:54:42 +01:00
ftv->retTypes = clone(t.retTypes, dest, cloneState);
2023-05-19 19:59:59 +01:00
ftv->hasNoFreeOrGenericTypes = t.hasNoFreeOrGenericTypes;
2022-04-07 21:53:47 +01:00
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const TableType& t)
2022-04-07 21:53:47 +01:00
{
// If table is now bound to another one, we ignore the content of the original
2022-04-14 22:57:15 +01:00
if (!FFlag::DebugLuauCopyBeforeNormalizing && t.boundTo)
2022-04-07 21:53:47 +01:00
{
2022-04-14 22:57:15 +01:00
TypeId boundTo = clone(*t.boundTo, dest, cloneState);
2022-04-07 21:53:47 +01:00
seenTypes[typeId] = boundTo;
return;
}
2023-01-03 17:33:19 +00:00
TypeId result = dest.addType(TableType{});
TableType* ttv = getMutable<TableType>(result);
2022-04-07 21:53:47 +01:00
LUAU_ASSERT(ttv != nullptr);
*ttv = t;
seenTypes[typeId] = result;
ttv->level = TypeLevel{0, 0};
2022-04-14 22:57:15 +01:00
if (FFlag::DebugLuauCopyBeforeNormalizing && t.boundTo)
ttv->boundTo = clone(*t.boundTo, dest, cloneState);
2022-04-07 21:53:47 +01:00
for (const auto& [name, prop] : t.props)
2023-04-28 12:55:55 +01:00
ttv->props[name] = clone(prop, dest, cloneState);
2022-04-07 21:53:47 +01:00
2023-07-07 18:14:35 +01:00
if (t.indexer)
ttv->indexer = clone(*t.indexer, dest, cloneState);
2022-04-07 21:53:47 +01:00
for (TypeId& arg : ttv->instantiatedTypeParams)
2022-04-14 22:57:15 +01:00
arg = clone(arg, dest, cloneState);
2022-04-07 21:53:47 +01:00
for (TypePackId& arg : ttv->instantiatedTypePackParams)
2022-04-14 22:57:15 +01:00
arg = clone(arg, dest, cloneState);
2022-04-07 21:53:47 +01:00
ttv->definitionModuleName = t.definitionModuleName;
2023-01-20 12:02:39 +00:00
ttv->definitionLocation = t.definitionLocation;
2022-04-07 21:53:47 +01:00
ttv->tags = t.tags;
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const MetatableType& t)
2022-04-07 21:53:47 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId result = dest.addType(MetatableType{});
MetatableType* mtv = getMutable<MetatableType>(result);
2022-04-07 21:53:47 +01:00
seenTypes[typeId] = result;
2022-04-14 22:57:15 +01:00
mtv->table = clone(t.table, dest, cloneState);
mtv->metatable = clone(t.metatable, dest, cloneState);
2022-04-07 21:53:47 +01:00
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const ClassType& t)
2022-04-07 21:53:47 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId result = dest.addType(ClassType{t.name, {}, std::nullopt, std::nullopt, t.tags, t.userData, t.definitionModuleName});
ClassType* ctv = getMutable<ClassType>(result);
2022-04-07 21:53:47 +01:00
seenTypes[typeId] = result;
for (const auto& [name, prop] : t.props)
2023-04-28 12:55:55 +01:00
ctv->props[name] = clone(prop, dest, cloneState);
2022-04-07 21:53:47 +01:00
if (t.parent)
2022-04-14 22:57:15 +01:00
ctv->parent = clone(*t.parent, dest, cloneState);
2022-04-07 21:53:47 +01:00
if (t.metatable)
2022-04-14 22:57:15 +01:00
ctv->metatable = clone(*t.metatable, dest, cloneState);
2023-05-12 13:15:01 +01:00
2023-07-07 18:14:35 +01:00
if (t.indexer)
ctv->indexer = clone(*t.indexer, dest, cloneState);
2022-04-07 21:53:47 +01:00
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const AnyType& t)
2022-04-07 21:53:47 +01:00
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const UnionType& t)
2022-04-07 21:53:47 +01:00
{
2023-05-19 19:59:59 +01:00
if (FFlag::LuauCloneCyclicUnions)
{
// We're just using this FreeType as a placeholder until we've finished
// cloning the parts of this union so it is okay that its bounds are
// nullptr. We'll never indirect them.
TypeId result = dest.addType(FreeType{nullptr, /*lowerBound*/nullptr, /*upperBound*/nullptr});
2023-05-19 19:59:59 +01:00
seenTypes[typeId] = result;
2022-04-07 21:53:47 +01:00
2023-05-19 19:59:59 +01:00
std::vector<TypeId> options;
options.reserve(t.options.size());
2022-04-07 21:53:47 +01:00
2023-05-19 19:59:59 +01:00
for (TypeId ty : t.options)
options.push_back(clone(ty, dest, cloneState));
asMutable(result)->ty.emplace<UnionType>(std::move(options));
}
else
{
std::vector<TypeId> options;
options.reserve(t.options.size());
for (TypeId ty : t.options)
options.push_back(clone(ty, dest, cloneState));
TypeId result = dest.addType(UnionType{std::move(options)});
seenTypes[typeId] = result;
}
2022-04-07 21:53:47 +01:00
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const IntersectionType& t)
2022-04-07 21:53:47 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId result = dest.addType(IntersectionType{});
2022-04-07 21:53:47 +01:00
seenTypes[typeId] = result;
2023-01-03 17:33:19 +00:00
IntersectionType* option = getMutable<IntersectionType>(result);
2022-04-07 21:53:47 +01:00
LUAU_ASSERT(option != nullptr);
for (TypeId ty : t.parts)
2022-04-14 22:57:15 +01:00
option->parts.push_back(clone(ty, dest, cloneState));
2022-04-07 21:53:47 +01:00
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const LazyType& t)
2022-04-07 21:53:47 +01:00
{
2023-04-21 22:41:03 +01:00
if (TypeId unwrapped = t.unwrapped.load())
{
seenTypes[typeId] = clone(unwrapped, dest, cloneState);
}
else
{
defaultClone(t);
}
2022-04-07 21:53:47 +01:00
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const UnknownType& t)
2022-07-08 02:05:31 +01:00
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const NeverType& t)
2022-07-08 02:05:31 +01:00
{
defaultClone(t);
}
2023-01-03 17:33:19 +00:00
void TypeCloner::operator()(const NegationType& t)
2022-10-21 18:33:43 +01:00
{
2023-01-03 17:33:19 +00:00
TypeId result = dest.addType(AnyType{});
2022-10-21 18:33:43 +01:00
seenTypes[typeId] = result;
TypeId ty = clone(t.ty, dest, cloneState);
2023-01-03 17:33:19 +00:00
asMutable(result)->ty = NegationType{ty};
2022-10-21 18:33:43 +01:00
}
2023-05-12 13:15:01 +01:00
void TypeCloner::operator()(const TypeFamilyInstanceType& t)
{
TypeId result = dest.addType(TypeFamilyInstanceType{
t.family,
{},
{},
});
seenTypes[typeId] = result;
TypeFamilyInstanceType* tfit = getMutable<TypeFamilyInstanceType>(result);
LUAU_ASSERT(tfit != nullptr);
tfit->typeArguments.reserve(t.typeArguments.size());
for (TypeId p : t.typeArguments)
tfit->typeArguments.push_back(clone(p, dest, cloneState));
tfit->packArguments.reserve(t.packArguments.size());
for (TypePackId p : t.packArguments)
tfit->packArguments.push_back(clone(p, dest, cloneState));
}
2022-04-07 21:53:47 +01:00
} // anonymous namespace
2022-04-14 22:57:15 +01:00
TypePackId clone(TypePackId tp, TypeArena& dest, CloneState& cloneState)
2022-04-07 21:53:47 +01:00
{
if (tp->persistent)
return tp;
2022-06-24 02:44:07 +01:00
RecursionLimiter _ra(&cloneState.recursionCount, FInt::LuauTypeCloneRecursionLimit);
2022-04-07 21:53:47 +01:00
2022-04-14 22:57:15 +01:00
TypePackId& res = cloneState.seenTypePacks[tp];
2022-04-07 21:53:47 +01:00
if (res == nullptr)
{
2022-04-14 22:57:15 +01:00
TypePackCloner cloner{dest, tp, cloneState};
2022-04-07 21:53:47 +01:00
Luau::visit(cloner, tp->ty); // Mutates the storage that 'res' points into.
}
return res;
}
2022-04-14 22:57:15 +01:00
TypeId clone(TypeId typeId, TypeArena& dest, CloneState& cloneState)
2022-04-07 21:53:47 +01:00
{
if (typeId->persistent)
return typeId;
2022-06-24 02:44:07 +01:00
RecursionLimiter _ra(&cloneState.recursionCount, FInt::LuauTypeCloneRecursionLimit);
2022-04-07 21:53:47 +01:00
2022-04-14 22:57:15 +01:00
TypeId& res = cloneState.seenTypes[typeId];
2022-04-07 21:53:47 +01:00
if (res == nullptr)
{
2022-04-14 22:57:15 +01:00
TypeCloner cloner{dest, typeId, cloneState};
2022-04-07 21:53:47 +01:00
Luau::visit(cloner, typeId->ty); // Mutates the storage that 'res' points into.
// Persistent types are not being cloned and we get the original type back which might be read-only
if (!res->persistent)
2022-04-21 22:04:22 +01:00
{
2022-04-07 21:53:47 +01:00
asMutable(res)->documentationSymbol = typeId->documentationSymbol;
2022-04-21 22:04:22 +01:00
}
2022-04-07 21:53:47 +01:00
}
return res;
}
2022-04-14 22:57:15 +01:00
TypeFun clone(const TypeFun& typeFun, TypeArena& dest, CloneState& cloneState)
2022-04-07 21:53:47 +01:00
{
TypeFun result;
for (auto param : typeFun.typeParams)
{
2022-04-14 22:57:15 +01:00
TypeId ty = clone(param.ty, dest, cloneState);
2022-04-07 21:53:47 +01:00
std::optional<TypeId> defaultValue;
if (param.defaultValue)
2022-04-14 22:57:15 +01:00
defaultValue = clone(*param.defaultValue, dest, cloneState);
2022-04-07 21:53:47 +01:00
result.typeParams.push_back({ty, defaultValue});
}
for (auto param : typeFun.typePackParams)
{
2022-04-14 22:57:15 +01:00
TypePackId tp = clone(param.tp, dest, cloneState);
2022-04-07 21:53:47 +01:00
std::optional<TypePackId> defaultValue;
if (param.defaultValue)
2022-04-14 22:57:15 +01:00
defaultValue = clone(*param.defaultValue, dest, cloneState);
2022-04-07 21:53:47 +01:00
result.typePackParams.push_back({tp, defaultValue});
}
2022-04-14 22:57:15 +01:00
result.type = clone(typeFun.type, dest, cloneState);
2022-04-07 21:53:47 +01:00
return result;
}
} // namespace Luau